仓颉原生应用编程语言教程(第10期)

168 阅读6分钟

视频:KCKCJY

try-with-resources

仓颉使用 try-catch-finally 表达式来实现异常处理,该机制和传统语言的异常处理机制十分相似,但仓颉额外提供了 try-with-resources 表达式语法来自动释放非内存资源。

不同于普通 try 表达式,try-with-resources 表达式中的 catch 块和 finally 块均是可选的,并且 try 关键字其后的块之间可以插入一个或者多个变量定义用来申请一系列的资源对象,这些资源对象在 try-with-resources 表达式中会被自动管理起来,当某个资源发生异常或表达式结束后都会自动释放,达到安全管理资源的目的。

如下实例所示,input 和 output 变量会在 try-with-resources 表达式过程中自动管理,开发者不需要关注当中各种情况的资源释放问题。

try (input = MyResource(),    output = MyResource()) {    while (input.hasNextLine()) {        let lineString = input.readLine()        output.writeLine(lineString)    }}

这里资源对象的类型(上面例子中的 MyResource)必须已经实现了 Resource 接口,特别是已经实现了 Resource 接口中要求的 isClosed 和 close 函数,能够判别资源是否已经被释放,以及做对应的释放操作。编译器将会在发生异常时或者 try 代码块正常结束时,插入对相应函数的调用,自动释放资源。

轻松并发

仓颉语言为并发编程提供了一种简单灵活的方式,通过轻量化线程模型和高效易用的无锁并发对象让并发编程变得轻松,将高效并发处理的能力直接置于开发者的手中。这一节将详细介绍仓颉并发编程两大关键技术的核心思想、设计、以及带来的显著优势,揭示仓颉语言如何实现“轻松并发”。

轻量化线程模型

仓颉语言采用用户态线程模型,在该模型下,每个仓颉线程都是极其轻量级的执行实体,拥有独立的执行上下文但共享内存。该模型不仅简化了开发者编写并发代码的过程,还带来了显著的性能优势。

  1. 开发态:仓颉语言的线程模型使开发者能够像编写普通代码一样轻松地实现并发编程。通常,用户态线程模型可分为“无栈”和“有栈”两种实现方案。尽管“无栈”模型可以将内存占用降到极低,但其实现通常需要在语言中引入新语法,最常见的就是 async/await 关键字。然而,这种新语法会显著增加开发者编写并发代码的复杂度。开发者不仅需要在编程过程中手动标记(如用 async 标记异步函数并用 await 标记其调用点),而且这种标记具有“传染性”(包含 await 的函数必须标记为 async),导致经典的“函数染色”问题。仓颉线程拥有独立的执行上下文,因此能够自由切换,开发者无需为标记操心,从而彻底消除这一复杂性。
  2. 运行态:与传统的操作系统线程相比,轻量化线程模型在性能上具有明显优势。由于其实现完全在用户空间进行,不依赖操作系统的线程管理,这从根本上减少了线程创建和销毁的开销,同时简化了线程调度流程。仓颉语言通过这种设计,实现了更高效的资源利用和更快的执行速度,尤其是在高并发场景下,这种优势更为显著。在一台常见的 x86 服务器上,仓颉线程创建的平均耗时为 700ns,这远小于操作系统线程的创建开销(操作系统线程的创建耗时量级一般为百微秒)。此外,一个仓颉线程仅占用 8Kb 内存资源,因此开发者可以在一个程序中同时创建十万级数量的仓颉线程,大大超出操作系统线程的限制。

整体而言,仓颉语言的轻量化线程设计不仅降低系统的负担,而且使得开发者能够在不增加编程复杂度的前提下,轻松实现数千甚至数万个并发任务。其核心优势包括:

  1. 简单的并发编程:不对开发者编写并发代码做过多语法约束,使其方便地使用仓颉线程并专注业务处理。
  2. 轻量级的开销:由于创建和切换用户态线程的开销远远小于传统的内核线程,仓颉语言可以快速地创建和销毁大量用户态线程,使得开发高并发应用变得轻而易举。
  3. 更高的并发能力:仓颉语言通过用户态线程模型,可以实现非常高的并发数,这使得它特别适合于 I/O 密集型和高并发的网络服务场景。
  4. 减少上下文切换成本:在轻量化线程模型中,上下文切换发生在用户空间,避免了传统线程切换需要经过内核态和用户态之间频繁转换的高成本。

基于这样的设计,在仓颉语言中,实现高效并发不再是一项复杂且耗时的任务。开发者可以通过简单的语法构造大量的用户态线程,无需担心传统并发模型中常见的性能瓶颈。假设我们有一个需求:需要同时处理多个网络请求。在仓颉语言中,这可以轻松实现,如下代码所示:

func fetch_data(url: String) {    let response = http_get(url)    process(response)}main() {    let urls = ["example.com/data1", "example.com/data2", ...]    let futures = ArrayList<Future>()    for (url in urls) {        let fut = spawn {fetch_data(url)} // 创建仓颉线程进行网络请求        futures.append(fut)    }    for (fut in futures) { // 等待所有仓颉线程完成        fut.get()    }}

在上述例子中,spawn 关键字用于创建一个新的仓颉线程,每个线程独立地执行 fetch_data 函数。仓颉语言的运行时环境会自动调度这些线程,而开发者只需关注业务逻辑的实现。最后通过获取线程结果来等待所有的仓颉线程完成,确保主线程能够同步地获取所有结果。

仓颉语言的用户态线程模型以其显著的性能优势和轻量级的设计理念,为并发编程提供了一个颇具吸引力的新选择。它使得编写高并发应用变得更加直接和高效,不仅适用于高负载的网络服务,还能满足各种计算密集型任务的需要。通过这种新型并发模型,仓颉语言降低了并发编程的复杂性,同时还充分利用了现代多核处理器的强大能力。