- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
协程的出现大大降低了异步编程的复杂度,可以让我们像写同步代码一样去写异步代码,如果没有它,那么很多异步的代码都是需要靠回调函数来一层层嵌套,这个在我之前的一篇有介绍 rxjava回调地狱-kotlin协程来帮忙 。 本篇文章主要介绍 。 这里针对kotlin的语法以及协程的具体用法细节不过多介绍,就当你已了解 。 稍微注意下runBlocking函数比较特别, 。 如下图:它接受了一个suspend的block函数 。 所以我上面的demo这里面有其实有三个suspend函数! 。 在idea我们可以把这个kotlin代码反编译成java代码 。 这个反编译后的java代码 有很多报错是无法直接copy出来运行的(这就没有csharp做的好,csharp反编译出来的代码至少不会报红), 。 看代码的确是一个状态机控制函数和一个匿名类,还原成正常的java代码如下: 比如test1函数 。 其他的函数也类似,完整的代码请查看: https://gist.github.com/yuzd/cf67048777f0eb8fc1b3757f5bf9e8f3 。 整个运行流程如下: 不难看出来suspend函数其实在编译后是变成了状态机,将我们顺序执行的代码,转换成了回调的形式 父suspend函数里面调用子suspend函数,其实是把自己传给了子suspend状态机,如果子函数挂起了,等子函数恢复后直接调用父函数(因为通过状态机的label来控制走不同逻辑,去恢复当时的调用堆栈) 。 这就是协程的挂起与恢复机制了 。 demo 。 我们反编译查看下编译器生成了怎样的状态机 。 看反编译的代码比较吃力,我还原成了正常代码, 。 完整代码请查看 https://github.com/yuzd/asyncawait_study 。 可以看出来,和kotlin其实原理差不多,都是生成一个函数加一个状态机 。 区别是csharp的函数就是创建一个状态机且启动它 。 整体的执行流程如下 。 ps:最右边的是展示如果有多个await 那么就会对应这个状态机的多个状态 。 通过查看kotlin和csharp的实现方式,我发现kotlin的生成的状态机(ContinuationImpl的实现)都是有继承关系的, 比如demo中的test2继承了test1,test继承了main(通过构造函数传递的) 。 然而csharp中没有这样的关系 。 这也带来了两者最大的区别,kotlin的协程绑定了scope的概念,一旦scope被取消,那么scope绑定的所有的协程也都被取消. 这点好像在csharp中没有(如果理解有误欢迎指正) 。 这在实际应用中是怎么个区别呢,举个例子 。 在kotlin是可以的 。 。 。 。
kotlin的suspend函数demo
public static Object test1(Continuation continuation) {
CoroutineTest1 continuationTest1;
label20:
{
if (continuation instanceof CoroutineTest1) {
continuationTest1 = (CoroutineTest1) continuation;
int i = continuationTest1.label & Integer.MIN_VALUE;
if (i != 0) {
continuationTest1.label -= Integer.MIN_VALUE;
}
break label20;
}
continuationTest1 = new CoroutineTest1(continuation);
}
Object result = (continuationTest1).result;
Object var4 = IntrinsicsKt.getCOROUTINE_SUSPENDED();
String var1;
switch ((continuationTest1).label) {
case 0:
ResultKt.throwOnFailure(result);
var1 = "test1-start";
System.out.println(var1);
(continuationTest1).label = 1;
if (test2(continuationTest1) == var4) {
return var4;
}
break;
case 1:
ResultKt.throwOnFailure(result);
break;
default:
throw new IllegalStateException("call to 'resume' before 'invoke' with coroutine");
}
var1 = "test1-end";
System.out.println(var1);
return Unit.INSTANCE;
}
final static class CoroutineTest1 extends ContinuationImpl {
Object result;
int label;
public CoroutineTest1(@Nullable Continuation<Object> completion) {
super(completion);
}
@Nullable
public Object invokeSuspend(@NotNull Object $result) {
this.result = $result;
this.label |= Integer.MIN_VALUE;
return test1(this);
}
}
kotlin协程的挂起点是怎么控制的,异步操作执行完后它知道从哪里恢复?
csharp的async&await
static async Task Main(string[] args)
{
await test1();
Console.WriteLine("Let's Go!");
}
async Task test1(){
Console.WriteLine("test1-start");
await test2();
Console.WriteLine("test1-end");
}
async Task test2()
{
Console.WriteLine("test2-start");
await Task.Delay(1000);
Console.WriteLine("test2-end");
}
static Task CreateMainAsyncStateMachine()
{
MainAsyncStateMachine stateMachine = new MainAsyncStateMachine
{
_builder = AsyncTaskMethodBuilder.Create(),
_state = -1
};
stateMachine._builder.Start(ref stateMachine);
return stateMachine._builder.Task;
}
struct MainAsyncStateMachine : IAsyncStateMachine
{
public int _state;
public AsyncTaskMethodBuilder _builder;
public TaskAwaiter _waiter;
public void MoveNext()
{
int num1 = this._state;
try
{
TaskAwaiter awaiter;
int num2;
if (num1 != 0)
{
awaiter = UserQuery.CreateTest1AsyncStateMachine().GetAwaiter();
if (!awaiter.IsCompleted)
{
Console.WriteLine("MainAsyncStateMachine######Test1AsyncStateMachine IsCompleted:false, 注册自己到Test1Async运行结束时运行");
this._state = num2 = 0;
this._waiter = awaiter;
this._builder.AwaitUnsafeOnCompleted(ref awaiter, ref this);
return;
}
}
else
{
Console.WriteLine("MainAsyncStateMachine######Test1AsyncStateMachine IsCompleted:true");
awaiter = this._waiter;
this._waiter = new TaskAwaiter();
this._state = num2 = -1;
}
awaiter.GetResult();
Console.WriteLine("MainAsyncStateMachine######Let's Go!");
}
catch (Exception e)
{
this._state = -2;
this._builder.SetException(e);
return;
}
this._state = -2;
this._builder.SetResult();
}
public void SetStateMachine(IAsyncStateMachine stateMachine)
{
this._builder.SetStateMachine(stateMachine);
}
}
// 当状态机启动时会触发 状态机的MoveNext方法的调用
stateMachine._builder.Start(ref stateMachine);
这两者相比较,引发怎样的思考
async void testAsyncA(){
testAsyncB();
// 我想取消,或者下面运行出异常了 我也无法取消testAsyncB这个任务
}
async void testAsyncB(){
// do long task
}
suspend fun test2() = coroutineScope {
println("test2-start")
async {
delay(100000);
}
delay(1000)
println("test2-end")
// 或者手动取消当前coroutineScope
this.cancel()
}
最后此篇关于kotlin的suspend对比csharp的async&await的文章就讲到这里了,如果你想了解更多关于kotlin的suspend对比csharp的async&await的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
C++ 社区目前正在讨论上挂协程与下挂协程。 例如,这个提案中提到了挂起:http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2015/n4453.p
我刚刚在稳定 channel 上升级了 Flutter,并在尝试启动应用程序(在本地 iOS 模拟器上)时获得了以下 StackTrace。使用 flutter test 运行单元测试也会受到影响。
我想使用插件camera_camera。但是当我尝试编译项目时出现此错误: Flutter:错误:找不到 Getter:“正在暂停”。案例 AppLifecycleState.suspending。
我正在考虑有关 suspend 的事情Arrow 的文档详细解释了:suspend () -> A提供与 IO 相同的保证. 因此,根据文档,只需使用 suspend我们正在将不纯函数转换为纯函数:
Grails应用程序在启动时会以--debug-forked模式挂起。 我正在使用--debug-fork参数运行应用程序,但它会挂起并等待,直到调试器连接到它,然后调试器断开连接,应用程序再次等待调
请考虑以下代码: // program is reading data from an excel file, Sheet1 at cell(0,0). public class Read {
Windows 和 Solaris 线程 API 都允许在“暂停”状态下创建线程。线程仅在稍后“恢复”时才真正开始。我习惯了没有这个概念的 POSIX 线程,我正在努力理解它的动机。谁能建议为什么创建
将方法注释为 @Suspendable 的最佳实践是什么?在 Flow 中,可能有多个查询 vault/compute 业务逻辑的私有(private)方法。这些是否应该用 @Suspendable
有没有办法暂时暂停表单上所有锚定控件的移动或调整大小?即: procedure ScaleFormBy(AForm: TForm; n, d: Integer); begin AForm.Su
我一直在尝试追踪 Jedi VCL 的 JvHidControllerClass.pas 中的内存泄漏,我在源代码历史记录中发现了此更改: 旧版本: constructor TJvHidDeviceR
我正在制作具有过渡-缩放-平移-效果的幻灯片,其性能对于平滑渲染至关重要。这是一个我要移植到Delphi 10.3的旧项目,有一个线程可以在运行渲染到屏幕的同时从jpgeg-images的文件流中加载
我需要在用户继续使用它的同时调试生产服务器。当断点命中时,我希望为其他用户的请求提供服务的其他线程在我检查线程中的变量并决定跳过或进入方法时继续工作。 在 Java 中,当遇到断点时,您可以在“暂停线
大家好我在做秒表。对于暂停,我使用 Thread.suspend() ,对于恢复,我使用 Thread.resume()。但简历并不是恢复工作。代码: pause(){ shouldRun = f
有一个程序在 TCP/IP 端口中监听和应答请求(专有二进制协议(protocol))。但该程序需要更新,因此需要重新启动,然后才能继续在同一端口上执行其工作。 根据其协议(protocol),所有当
这个问题与 Thread.suspend 的替代方案无关。这是关于使用 Thread.suspend 实现偏向锁的可能性,(我相信)不能使用 Thread.interrupt 或类似的替代方案来实现。
所以我已经启动并运行了一台家庭服务器,并使用局域网唤醒对其进行了设置,并且有一位 friend 设法将其从两个州之外的地方唤醒。太棒了,这样我就可以运行文件服务器而无需一直保持开机状态。问题是,唤醒它
有没有办法确定是否已注册指纹但 Touch ID 由于指纹验证失败次数过多而当前处于“暂停”状态? 最佳答案 错误代码可以与 LAError 类进行比较。 func errorMessageForLA
一般问题... 如果我在我的 VBA 代码(基于 Excel)中暂停/退出/中断“For”循环,会有什么危险? 这是我的代码: For i = 1 to Sheets.Count 'check
我有一个 WPF 项目,当我尝试在 RowLoad 事件上运行此代码时,出现以下错误: private void ParentGridView_OnRowLoaded(object sender, E
在我的帐户页面/查看订阅中,在 ACTION 段落中,我有一个 CANCEL 按钮,但没有 SUSPEND 按钮。你知道这是否正常吗? 我跟踪了这个函数,它只返回取消数据:wcs_get_all_
我是一名优秀的程序员,十分优秀!