当前位置: 首页 > news >正文

Kotlin的Lambda闭包语法

Lambda 表达式是一种在现代编程语言中常见的特性,它可以用来创建匿名函数或代码块,使得将函数作为参数传递、简化代码以及实现函数式编程范式变得更加便捷。Lambda 表达式在函数式编程语言中得到广泛应用,也在诸如 Java 8 和 Kotlin 等主流编程语言中引入。

一、Java 中的 Lambda 表达式语法

在 Java 中,Lambda 表达式是从 Java 8 版本开始引入的一项重要特性,它允许我们以更简洁的方式创建匿名函数,从而在处理函数式编程、回调等场景中变得更加方便。Lambda 表达式的引入在 Java 中使得编写代码变得更加紧凑,同时也提供了更强大的函数式编程能力。

以下是 Java 中 Lambda 表达式的基本概念和语法:

(parameters) -> { /* code */ }
  • parameters 是 Lambda 表达式的参数列表。参数的类型可以根据上下文进行类型推断。
  • -> 是 Lambda 表达式的箭头符号,分隔参数列表和函数体。
  • { /* code */ } 是 Lambda 表达式的函数体,可以是一行或多行代码。

Lambda 表达式通常用于函数式接口(Functional Interface)中,这是一个只包含一个抽象方法的接口。Lambda 表达式可以实现这个接口的抽象方法,从而将匿名函数传递给方法或函数。

以下是一个简单的示例,展示了如何在 Java 中使用 Lambda 表达式:

public class LambdaExample {public static void main(String[] args) {// Lambda 表达式作为参数传递给 Thread 的构造函数Thread thread = new Thread(() -> {System.out.println("Thread is running");});thread.start();}
}

在这个示例中,Lambda 表达式 () -> { System.out.println("Thread is running"); } 被传递给 Thread 类的构造函数,用作线程的任务(Runnable)。这样就避免了传统的匿名内部类的繁琐语法,使代码更加简洁。

需要注意的是,Lambda 表达式在 Java 中需要满足以下要求:

  • Lambda 表达式只能用于函数式接口中,该接口必须只包含一个抽象方法。
  • 如果接口有默认方法(default 关键字修饰的方法),那么它不会影响函数式接口的定义。
  • 在 Lambda 表达式中可以捕获 final 或 effectively final 的局部变量。

通过掌握 Java 中的 Lambda 表达式,你可以在代码中更自然地表达函数式概念,从而编写更简洁、高效的代码。

二、Kotlin 中的 Lambda 表达式语法

在 Kotlin 中,Lambda 表达式是一种强大的特性,允许你以简洁的方式创建匿名函数,从而在函数式编程、集合操作等场景中变得更加方便。Kotlin 的 Lambda 表达式语法非常灵活,与函数的声明和调用紧密结合,使得代码更加清晰和易读。

以下是 Kotlin 中 Lambda 表达式的基本语法和一些重要概念:

{ parameters -> /* code */ }

可以看出和 Java 的Lambda语法类似,只不过这个箭头是放在了大括号的里面,而不是外面。

  • parameters 是 Lambda 表达式的参数列表。参数的类型可以根据上下文进行类型推断。
  • -> 是 Lambda 表达式的箭头符号,分隔参数列表和函数体。
  • { /* code */ } 是 Lambda 表达式的函数体,可以是一行或多行代码。

Lambda 表达式通常用于函数式接口(Functional Interface)中,或者用于集合操作、高阶函数等情境中。

以下是一个简单的示例,展示了如何在 Kotlin 中使用 Lambda 表达式:

fun main() {val thread = Thread ({ -> println("Thread is running") })thread.start()
}

如果 Lambda 是函数的最后一个参数,可以将大括号放在小括号外面:

fun main() {val thread = Thread() { -> println("Thread is running") }thread.start()
}

如果函数只有一个参数且这个参数是 Lambda,则可以省略小括号:

fun main() {val thread = Thread { -> println("Thread is running") }thread.start()
}

如果 Lambda 表达式没有参数,你可以直接省略 -> 符号:

fun main() {val thread = Thread { println("Thread is running") }thread.start()
}

好了,就此我们看一下 Java 和 Kotlin 的 Lambda 表达式,如图所示:

在这里插入图片描述

三、声明一个 Kotlin 的 Lambda 闭包

Kotlin 的 Lambda 表达式本质上就是闭包,它是一种可以捕获并携带作用域中变量状态的函数。Lambda 闭包在 Kotlin 中具有以下特点:

  1. 捕获变量: Lambda 表达式可以在其作用域外捕获外部变量。这意味着 Lambda 表达式可以访问外部作用域中的变量,即使在该作用域已经结束的情况下。捕获的变量可以是 valvar,但在 Lambda 表达式中只能被读取,不能被修改(对于 val)。
  2. 记住状态: Lambda 表达式捕获的变量状态在闭包内是“记住”的,即使闭包被传递到其他函数或在不同上下文中执行,它仍然可以访问并使用这些变量。
  3. 隐式参数: 当 Lambda 表达式只有一个参数时,可以使用隐式参数 it 来代表这个参数。这使得代码更加简洁。
  4. 函数式编程: Lambda 闭包使得 Kotlin 可以支持函数式编程范式,使代码更加模块化、易读和易于测试。

Kotlin 的 Lambda 闭包(也称为 Lambda 表达式)具有以下基本格式:

val lambdaName: (parameters) -> returnType = { arguments ->// Lambda 表达式的主体
}

我们解释一下每个部分的含义:

  • val lambdaName: 这是 Lambda 表达式的变量名,你可以根据需要命名它。
  • (parameters) -> returnType: 这是 Lambda 表达式的类型,它指定了参数列表和返回类型。parameters 是参数列表,可以是一个或多个参数,用逗号分隔。returnType 是返回类型,指示 Lambda 表达式返回的值的类型。
  • arguments ->: 这是 Lambda 表达式的参数部分。arguments 是 Lambda 表达式的参数,你可以为参数指定名称。箭头 -> 分隔了参数和主体部分。
  • { /* Lambda 表达式的主体 */ }: 这是 Lambda 表达式的主体,包含了 Lambda 表达式的实际逻辑和操作。这部分代码会在 Lambda 表达式被调用时执行。

下面是一个简单的示例,演示如何声明一个 Kotlin 的 Lambda 闭包:

fun main() {person.invoke("danke")person("danke")
}val person: (name: String) -> Unit = {name: String -> println("name is $name")// 可以省略Unit
}

在这个示例中,我们首先声明了一个名为 person 的变量,其值是一个 Lambda 表达式。这个 Lambda 表达式接受一个 String 类型的参数 name,并在闭包体内使用 $name 来将参数的值嵌入到字符串中。这个 Lambda 表达式就是 Lambda 闭包。

在 Kotlin 中,如果一个 Lambda 表达式没有返回值,可以使用 Unit 来表示。Unit 实际上是一个类型,类似于 Java 中的 void,但它是一个真正的类型,表示一个只有一个值的类型。

当 Lambda 表达式没有明确的返回值时,Kotlin 会默认将其推断为 Unit 类型。因此,你可以省略不写返回值类型。

接下来,在 main 函数中,我们通过两种方式来调用 person 这个闭包:

  • person.invoke("danke"):这里使用 invoke 函数来显式地调用闭包,并传递参数 "danke"
  • person("danke"):这是一种更简洁的方式,直接像调用函数一样调用闭包,并传递参数 "danke"

无论使用哪种方式,闭包都会执行,打印出 "name is danke"

上面示例展示了如何声明一个 Kotlin 的 Lambda 闭包,并且演示了如何调用闭包并传递参数。闭包在 Kotlin 中是一种非常强大的概念,允许在代码中以一种更函数式的方式操作数据和逻辑。

在 Kotlin 中,如果 Lambda 表达式的参数和返回类型可以从上下文中推断出来,你可以省略参数列表和返回类型的定义。那么我们改造一下上面声明的 Lambda 闭包,其中省略了参数列表和返回类型的定义:

val person = { name: String -> println("name is $name") }

在这个示例中,由于 Lambda 表达式的参数类型和返回类型可以从赋值操作的右侧(Lambda 表达式)推断出来,所以我们可以省略掉 (name: String) -> Unit 这部分。编译器会自动推断出参数类型为 String,返回类型为 Unit

这种简化的写法使代码更加简洁,但需要注意的是,当 Lambda 表达式的参数和返回类型不容易从上下文中推断出时,最好还是显式地声明它们,以增加代码的可读性和清晰性。

Kotlin 的 Lambda 闭包使得函数变得更具表达力和灵活性。它们可以用于各种情境,从集合操作到异步编程,从而提供了更多的编程选择和优雅的语法。

四、Kotlin 的 Lambda 的参数是有上限的

在 Kotlin 中,Lambda 表达式的参数数量是有上限的,这个上限与函数式接口的抽象方法数量有关。通常情况下,Lambda 表达式的参数数量最多为 22,因为 Kotlin 的标准库中提供了 22 个用于函数式编程的函数式接口(如 Function0Function22)。

这意味着你可以创建一个具有最多 22 个参数的 Lambda 表达式。如果我们尝试给 Lambda 表达式添加第 23 个参数的时候,例如:

fun main() {maxParams(1, 1, 1, 1, 1, 1, 1, 1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1,1, 1, 1)
}val maxParams ={ p1: Int, p2: Int, p3: Int, p4: Int, p5: Int, p6: Int, p7: Int, p8: Int, p9: Int, p10: Int,p11: Int, p12: Int, p13: Int, p14: Int, p15: Int, p16: Int, p17: Int, p18: Int, p19: Int, p20: Int,p21: Int, p22: Int, p23: Int ->println("maxParams")}

看到最终运行这段代码就抛出一个异常:java.lang.NoClassNotFoundError:kotlin/Function23。为什么它最后报没有Function23这样的一个类呢?这是因为 Kotlin 的类在编译以后会被编译成 class 文件,Kotlin 的 lambda 在编译以后会被编译成一个匿名内部类,我们定义 lambda 表达式有 23 个参数,它也是一个匿名类,但是我们在的 Kotlin 的源码中,看到只定义了 22 个 Function,这个 Function 后面紧跟着数字,就表示有多少个参数。从零开始表示没有参数,一直到 22。我们可以看一下 Kotlin 源码:

在这里插入图片描述

因此参数数量超过 22 的 Lambda 表达式在调用时会引发错误。这是 Kotlin 的限制,Lambda 表达式的参数数量应该小于等于 22。

有没有办法解决?

如果我们给 Lambda 传了23个参数的时候,它就会直接报错了。那么这个问题能不能解决呢?当然也是可以的!就是我们手动去定义这个 Function23 这样的一个类。在定义这样一个类的时候,我们可能还会碰上问题。首先,我们一起来在代码中试一下。

// Function23.kt
package kotlin/** A function that takes 23 arguments. */
public interface Function23<in P1, in P2, in P3, in P4, in P5, in P6, in P7, in P8, in P9, in P10, in P11, in P12, in P13, in P14, in P15, in P16, in P17, in P18, in P19, in P20, in P21, in P22, in P23, out R> : Function<R> {/** Invokes the function with the specified arguments. */public operator fun invoke(p1: P1, p2: P2, p3: P3, p4: P4, p5: P5, p6: P6, p7: P7, p8: P8, p9: P9, p10: P10, p11: P11, p12: P12, p13: P13, p14: P14, p15: P15, p16: P16, p17: P17, p18: P18, p19: P19, p20: P20, p21: P21, p22: P22, p23: P23): R
}

首先创建Function23.kt,并将这个类的包名改成package kotlin。我们假设在 Kotlin 这个包下面创建一个 Function23 这样的一个类。那么它的类的声明是跟前面 Function22 完全一致,只是多了一个参数,我们就直接拷过来。这样我们就定义完成了 Function23。

然后我们再来运行这样的一段代码。看到编译器给我们报出来了一个错误,说只有Kotlin 的标准库才可以使用 Kotlin 这样的一个包名。而我们自己声明的类,是不能声明一个类的包名叫Kotlin的。

Kotlin: Only the Kotlin standard library is allowed to use the 'kotlin' package

那么,这个问题怎么解决呢?

大家回想一下 Kotlin 与 Java 是完全兼容的,如果我们不能以 Kotlin 的形式去声明一个类的话,那是不是可以用 Java 的形式去声明它?

/*** A function that takes 23 arguments.*/
public interface Function23<P1, P2, P3, P4, P5, P6, P7, P8, P9, P10, P11, P12, P13, P14, P15, P16, P17, P18, P19, P20, P21, P22, P23, R> extends Function<R> {/*** Invokes the function with the specified arguments.*/R invoke(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7, P8 p8, P9 p9, P10 p10, P11 p11, P12 p12, P13 p13, P14 p14, P15 p15, P16 p16, P17 p17, P18 p18, P19 p19, P20 p20, P21 p21, P22 p22, P23 p23);
}

所以咱们能够将这个Function23 申明为一个 Java类,并将它的包名设置为kotlin,这样就能够申明参数个数超过 22 的闭包了。

然而,使用拥有如此多参数的 Lambda 表达式往往会导致代码的可读性降低,而且实际情况下很少会需要这么多参数。在大多数情况下,Lambda 表达式的参数数量应该保持合理,以确保代码的清晰度和可维护性。

除了上限之外,Kotlin 还支持使用 vararg 关键字来定义一个接受可变数量参数的 Lambda 表达式,类似于普通函数的可变参数。(在上文中已经介绍过了)

总之,Kotlin 的 Lambda 表达式参数数量上限为 22,但实际中应该遵循良好的编程实践,保持适度的参数数量,以提高代码的可读性和可维护性。

五、Java 怎么调用 Kotlin 的 Lambda 闭包

Kotlin 文件 LambdaTest.kt

val person = { name: String -> println("name is $name") }

Java 文件 Main.java

import kotlin.Unit;
import kotlin.jvm.functions.Function1;public class java8 {public static void main(String[] args) {Function1<String, Unit> person = LambdaTestKt.getPerson();person.invoke("danke");}
}

在这个示例中,我们在 Java 中通过 LambdaTestKt.getPerson() 方法来获取 Kotlin 文件中定义的 person Lambda 表达式实例,然后通过 person.invoke("danke") 调用该 Lambda 表达式。

这种方式可以在 Java 中调用 Kotlin 文件中的顶层 Lambda 表达式。

相关文章:

Kotlin的Lambda闭包语法

Lambda 表达式是一种在现代编程语言中常见的特性&#xff0c;它可以用来创建匿名函数或代码块&#xff0c;使得将函数作为参数传递、简化代码以及实现函数式编程范式变得更加便捷。Lambda 表达式在函数式编程语言中得到广泛应用&#xff0c;也在诸如 Java 8 和 Kotlin 等主流编…...

day-01 Docker

一、docker简介 Docker 是一种开源的容器化平台&#xff0c;它可以帮助开发人员将应用程序及其依赖项打包成一个独立的、可移植的容器&#xff0c;而无需担心环境差异和依赖问题。通过使用 Docker&#xff0c;您可以更轻松地创建、分发和运行应用程序&#xff0c;无论是在开发、…...

ARM开发,stm32mp157a-A7核SPI总线实验(实现数码管的显示)

1.目标&#xff1a; a.数码管显示相同的值 0000 1111 ......9999&#xff1b; b.数码管显示不同的值 1234&#xff1b; 2.分析m74hc595芯片内部框图&#xff1b; 真值表&#xff1a; 3.代码&#xff1b; ---spi.h头文件--- #ifndef __SPI_H__ #define __SPI_H__#include &quo…...

思路灰度传感器及红外传感器线序

四路红外传感器 黑线读取数据为0 白线读取数据为1 四路灰度传感器 黑线读取数据为1 白线读取数据为0...

squid服务器

目录 squid初识 安装squid代理 常用命令 主要配置文件 正向代理 环境配置 linux服务器设置 windows客户端设置 反向代理 环境配置 在web服务器配置服务 linux服务器配置 squid初识 含义&#xff1a;squid cache是一个流行的自由软件&#xff08;GNU通用公共许可证…...

spring的后置处理器BeanPostProcessor

什么是BeanPostProcessor 是spring IOC容器给我们提供的一个扩展接口在调用初始化方法前后对bean进行额外加工&#xff0c;ApplicationContext会自动扫描实现了BeanPostProcessor的bean&#xff0c;并注册这些bean为后置处理器是bean的统一前置后置处理而不是基于某一个bean 执…...

vue、uniapp中动态添加绑定style、class 9种方法实现

9种方法介绍 直接使用静态class和style属性&#xff1a; 使用场景&#xff1a;当class和style属性是固定不变的时候&#xff0c;可以直接在模板中写死。优点&#xff1a;简单直接&#xff0c;没有额外的计算和逻辑。缺点&#xff1a;无法根据条件动态修改class和style。 使用v…...

【CicadaPlayer】seek :SeekInCache(int64_t pos)的实现

SuperMediaPlayer::SeekInCache(int64_t pos) 的实现 seek的pos就是pts值。缓冲是list,那么插入的包是按照到达的顺序插入到list的,也就是无排序的。包的pts 正常应该单调连续,即使不单调连续,缓存也不在意。seek的操作主要是先比较pos与mCurrentPos ,pos 比 mCurrentPos …...

【C/C++】x -x 的含义

1、含义 -x 的值&#xff0c;其实就是在x的值的基础上进行按位取反&#xff08;~x&#xff09;之后在增加1所得&#xff08;C语言中&#xff0c;-x实现是用取反1实现&#xff09;也就是说&#xff1a;x & -x x & (~x 1) 2、x 为偶数 当一个奇数 1时&#xff0c;表示…...

[ZenTao]源码阅读:加载自定义任务类型

www/index.php config/config.php framework/base/router.class.php tmp/model/common.php module/common/model.php framework/router.class.php...

hive分区表 静态分区和动态分区

一、静态分区 现有数据文件 data_file 如下&#xff1a; 2023-08-01,Product A,100.0 2023-08-05,Product B,150.0 2023-08-10,Product A,200.0 1、创建分区表 CREATE TABLE sales (sale_date STRING,product STRING,amount DOUBLE ) PARTITIONED BY (sale_year INT, sale_mon…...

java八股文面试[多线程]——ThreadLocal底层原理和使用场景

源码分析&#xff1a; ThreadLocal中定义了ThreadLocalMap静态内部类&#xff0c;该内部类中又定义了Entry内部类。 ThreadLocalMap定了 Entry数组。 Set方法&#xff1a; Get方法&#xff1a; Thread中定义了两个ThreaLocalMap成员变量&#xff1a; Spring使用ThreadLocal解…...

Android hid发送apdu格式数据

在 Android 中&#xff0c;如果你想通过 HID&#xff08;Human Interface Device&#xff09;发送 APDU 格式的数据&#xff0c;通常会涉及 USB HID 设备或蓝牙 HID 设备。HID 协议通常用于键盘、鼠标和其他输入设备&#xff0c;而不是直接与智能卡进行通信。然而&#xff0c;如…...

Unity碰撞检测(3D和2D)

Unity碰撞检测3D和2D 前言准备材料3D2D 代码3D使用OnCollisionEnter()进行碰撞Collider状态代码 使用OnTriggerEnter()进行碰撞Collider状态代码 2D使用OnCollisionEnter2D()进行碰撞Collider2D状态代码 使用OnTriggerEnter2D()进行碰撞Collider2D状态代码 区别3D代码OnCollisi…...

android:控件TextView

一、系统学习Android控制键TextView&#xff0c;我的笔记里面有尝试学着使用自定义控件。 二、具体内容 1.如果在代码中给textView赋值&#xff0c;在xml中也给textView赋值了最后运行出来的结果显示代码中赋的值。因此得出结论&#xff0c;代码中的赋值会覆盖xml所附的值。 …...

3D风速仪 Gill Instruments Limited_R3-50 R3-100 and R3A -100 Manual

R3测量超声波脉冲从上部换能器到相反的下部换能器所花费的时间&#xff0c;并将其与脉冲从下部换能器到上部换能器的时间进行比较。 同样&#xff0c;在其他上下换能器之间比较时间。 如图1所示&#xff0c;每对换能器之间沿轴的空气速度可以从每条轴上的飞行次数计算出来。 …...

深度学习怎么学?

推荐这本小白看的《深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09;》。 深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09; 深入浅出的讲述了深度学习的基本概念与理论知识&#xff0c;不涉及复杂的数学内容&#xff0c;零基础小白也能轻松掌…...

WPF 数据验证

WPF提供了能与数据绑定系统紧密协作的验证功能。提供了两种方法用于捕获非法值&#xff1a; 1、可在数据对象中引发错误。 可以在设置属性时抛出异常&#xff0c;通常WPF会忽略所有在设置属性时抛出的异常&#xff0c;但可以进行配置&#xff0c;从而显示更有帮助的可视化指示…...

IDEA的maven想显示层级关系,而非平级

新版和旧版的IDEA的位置不一样&#xff0c;2023.2.1的版本在右上角的“” 这个位置 如图所示&#xff1a; 然后点击按模块分组&#xff1a;...

(八)k8s实战-身份认证与权限

一、认证 User AccountsService Accounts Service Account 自动化&#xff1a; Service Account Admission ControllerToken ControllerService Account Controller 1、Service Account Admission Controller 通过 Admission Controller 插件来实现对 pod 修改&#xff0c…...

数学建模:TOPSIS分析

&#x1f506; 文章首发于我的个人博客&#xff1a;欢迎大佬们来逛逛 TOPSIS分析法 算法流程 假设有m个评价对象&#xff0c;n个评价指标&#xff0c;首先需要进行指标的正向化&#xff1a; 极大型极小型单点型区间型 然后对正向化后的矩阵进行标准化&#xff0c;得到 Z Z Z…...

【Qt学习】10 利用QSharedMemory实现单例运行

问题 让应用程序只有一个运行实例 QSharedMemory除了可以完成进程间通信&#xff0c;还可以实现应用程序单例化。 解法 首先&#xff0c;看看QSharedMemory的几个函数&#xff1a; 1、QSharedMemory(const QString &key, QObject *parent Q_NULLPTR)构造函数 该构造函数…...

FPGA应用于图像处理

FPGA应用于图像处理 FPGA&#xff08;Field-Programmable Gate Array&#xff09;直译过来就是现场可编程门阵列。是一种可以编程的逻辑器件&#xff0c;具有高度的灵活性&#xff0c;可以根据具体需求就像编程来实现不同的功能。 FPGA器件属于专用的集成电流中的一种半定制电…...

vscode python 无法引入上层目录解决

在vscode 中.vscode 配置如下 { // Use IntelliSense to learn about possible attributes. // Hover to view descriptions of existing attributes. // For more information, visit: https://go.microsoft.com/fwlink/?linkid830387 “version”: “0.2.0”, “configurati…...

[开发|java] java list 取某个属性最大的项

示例代码: import java.util.*;class Person {private String name;private int age;public Person(String name, int age) {this.name name;this.age age;}public int getAge() {return age;} }public class Main {public static void main(String[] args) {List<Person…...

关闭浏览器的跨域校验

首发博客地址 问题描述 当你访问资源失败&#xff0c;并遇到以下类似提示时&#xff1a; Access to script at 资源路径 from origin null has been blocked by CORS policy: Cross origin requests are only supported for protocol schemes: http, data, isolated-app, chrom…...

USRP 简介,对于NI软件无线电你所需要了解的一切

什么是 USRP 通用软件无线电外设( USRP ) 是由 Ettus Research 及其母公司National Instruments设计和销售的一系列软件定义无线电。USRP 产品系列由Matt Ettus领导的团队开发&#xff0c;被研究实验室、大学和业余爱好者广泛使用。 大多数 USRP 通过以太网线连接到主机&…...

RTE_Driver驱动框架和Keil下开发需要支持的xxx_DFP软件包分析

1.RTE_Driver驱动框架 RTE_Driver代表"Run-Time Environment Driver"&#xff0c;是Keil MDK&#xff08;Microcontroller Development Kit&#xff09;中的一个概念。Keil MDK是一种用于嵌入式系统开发的集成开发环境&#xff0c;提供了开发、编译、调试等一系列工具…...

ImportError: Cannot load dynamic library. Did you compile LSD?

1、问题描述 >>> import pylsd2 Traceback (most recent call last):File "<stdin>", line 1, in <module>File "/data/data/wangzy-p-wangzy-p3-volume-pvc-0fee40a7-7013-49b4-8cfb-b4ab0394165b/.conda/envs/paddle/lib/python3.8/sit…...

音频应用编程

目录 ALSA 概述alsa-lib 简介sound 设备节点alsa-lib 移植编写一个简单地alsa-lib 应用程序一些基本概念打开PCM 设备设置硬件参数读/写数据示例代码之PCM 播放示例代码值PCM 录音 使用异步方式PCM 播放示例-异步方式PCM 录音示例-异步方式 使用poll()函数使用poll I/O 多路复用…...