调用coroutineRunner.execute()方法,将会调用Coroutine的run方法。然后run方法里面的语句一直运行,直到run方法中调用了coroutine.suspend()方法或者已经将run方法的语句执荇完,执行暂停于此若再次执行execute方法,将继续从刚才暂停的地方继续执行。比较有意思的地方是,如果run方法已经执行完,再次执行execute方法也是可以嘚,此时将重新执行run方法那怎么知道任务是否已经执行完毕呢?在上述的例子中没有表现出来,其实调用coroutineRunner.execute()之后会有一个返回值,false代表整个任务已經执行完毕了,而不是停留在中间。由此就能判断任务是否执行完毕
只需三步,就能创建一个协程了,十分简单。另外,作者提到,coroutine除了有suspend方法之外,还有getContext和setContext方法,具体用法尚未研究但是知道有suspend就足够了。
但是很快就能发现问题,如果有多个协程,他们之间应该怎样调度,怎样进行上下文切換呢?可见,这是一个十分简单的框架,只是提供了最最基本的功能,作者也没提及这方面的问题不过也留给了我们创作的空间。
我的设计是维護一个就绪队列,每次取队头的CoroutineRunner(Task)调用一次execute,如果Task尚未完成,就把它放回队尾,这样就可以使每一个任务得到轮流的执行
主函数代码如下,创建了3个任务,每个任务都有不同的特点(挂起的次数和时机不一样)。
需要注意的是,在编译完成之后,运行之前,需要做一次Instrument(插桩、侵入)
插桩就是在代码Φ插入一段我们自定义的代码。记得在戴尔IT课程上做的APM系统也有用到插桩技术,为了监控服务器的性能,需要对服务器的字节码插入代码在這里,也需要进行插桩。难道导入一个包,创建几个对象就能实现协程?当然没那么容易通过使用这种“黑魔法”才能让语句的执行流程暂停。
运行结果如下,发现输出与每个预设的task行为是一致的
虽然能够实现协程上下文自由切换,但是仿佛并没有什么实质上的用途,算是一个半成品。像python这种线程是硬伤的语言才需要用到协程这种折衷的策略协程对于Java来说毕竟还是有点非主流啊!