Kotlin协程简述与上下文和调度器(Dispatchers )

news2025/1/12 22:09:19

协程概述

子程序或者称为函数,在所有的语言中都是层级调用,如:A调用B,B在执行过程中又调用了C,C执行完毕返回,B执行完毕返回,最后是A执行完毕。所以子程序是 通过栈来实现的,一个线程就是执行一个子程序。子程序调用总是一个入口,一次返回,调用顺序是明确的。

而协程看上去是子程序,执行的过程中,在 子程序中可中断,去执行其他的子程序,在适当的时候可以回来接着执行。

Kotlin协程工作原理

Kotlin 协程的大致的执行流程如上图所示,这个流程是各种类型的协程执行时都大致遵循的流程,不是一个严格精确的执行流程。

创建并启动协程

fun create.main() {
    //1. 创建协程体
    val coroutine = suspend {
        println("in coroutine")
        5
    }.createCoroutine(object: Continuation<Int> {
        override fun resumeWith(result: Result<Int>) {
            println("coroutine end: $result")
        }
​
        override val context: CoroutineContext
            get() = EmptyCoroutineContext
​
    })
​
    //2. 执行协程
    coroutine.resume(Unit)
}

上面代码的输出结果:

in coroutine
coroutine end: Success(5)

协程的执行过程

调用栈流程如下

  1. 我们通过 suspend block#createCoroutine 得到的 coroutine 实际是 SafeContinuation 对象
  2. SafeContinuation 实际上是代理类,其中的 delegate 属性才是真正的 Continuation 对象
  3. suspend block 中的代码在 BaseContinuationImpl 中执行
  4. 我们的匿名内部类对象 Continuation 被回调

suspend block 是如何变为协程体被执行的?

我们分析调用栈得知,resumeWith 最终是在 BaseContinuationImpl 中执行的,下面来看看代码

@SinceKotlin("1.3")
internal abstract class BaseContinuationImpl(
    public val completion: Continuation<Any?>?
) : Continuation<Any?>, CoroutineStackFrame, Serializable {
    public final override fun resumeWith(result: Result<Any?>) {
        var current = this
        var param = result
        while (true) {
            probeCoroutineResumed(current)
            with(current) {
                val completion = completion!!
                val outcome: Result<Any?> =
                    try {
                        val outcome = invokeSuspend(param) //1.这里执行了 suspend block
                        if (outcome === COROUTINE_SUSPENDED) return
                        Result.success(outcome)
                    } catch (exception: Throwable) {
                        Result.failure(exception)
                    }
                releaseIntercepted()
                if (completion is BaseContinuationImpl) {
                    current = completion
                    param = outcome
                } else {
                    completion.resumeWith(outcome) //2.这里回调了我们的匿名内部类
                    return
                }
            }
        }
    }
​
    protected abstract fun invokeSuspend(result: Result<Any?>): Any? //3. 抽象方法
}

在代码注释 1. 处,调用 current.invokeSuspend,执行了我们定义的协程体,证明 suspend block 其实是 BaseContinuationImpl 的子类

在 2. 处,协程体执行完毕后,我们的代码收到了完成回调

在 3. 处,可以发现 invokeSuspend 是个抽象方法,suspend block 就是这个方法的具体实现

下面我通过断点,进一步分析 suspend block 是通过哪个子类执行的。

可以看到 current 是名为 {文件} 方法 {方法} 方法{变量}$1 格式的对象,证明 kotlin 编译器遇到 suspend 关键字后会帮我们生成一个 BaseContinuationImpl 的子类

那么,这个子类到底是什么呢?将 kt 编译为 .class 再通过 jadx 打开后,得到的 java 代码如下

public final class CreateCoroutineKt {
    public static final void create.main() {
        Continuation coroutine = ContinuationKt.createCoroutine(new CreateCoroutineKt$create.main$coroutine$1(null), new CreateCoroutineKt$create.main$coroutine$2());
        Unit unit = Unit.INSTANCE;
        Result.Companion companion = Result.Companion;
        coroutine.resumeWith(Result.constructor-impl(unit));
    }
}
final class CreateCoroutineKt$create.main$coroutine$1 extends SuspendLambda implements Function1<Continuation<? super Integer>, Object> {
    int label;
​
    CreateCoroutineKt$create.main$coroutine$1(Continuation<? super CreateCoroutineKt$create.main$coroutine$1> continuation) {
        super(1, continuation);
    }
​
    @NotNull
    public final Continuation<Unit> create(@NotNull Continuation<?> continuation) {
        return new CreateCoroutineKt$create.main$coroutine$1(continuation);
    }
​
    @Nullable
    public final Object invoke(@Nullable Continuation<? super Integer> continuation) {
        return create(continuation).invokeSuspend(Unit.INSTANCE);
    }
​
    @Nullable
    public final Object invokeSuspend(@NotNull Object obj) {
        IntrinsicsKt.getCOROUTINE_SUSPENDED();
        switch (this.label) {
            case 0:
                ResultKt.throwOnFailure(obj);
                System.out.println((Object) "in coroutine"); //协程体的逻辑
                return Boxing.boxInt(5);
            default:
                throw new IllegalStateException("call to 'resume' before 'invoke' with coroutine");
        }
    }
}

明显看出,kt 编译器帮助我们把 suspend 关键字变为了 SuspendLambda 的 子类,并重写了 invokeSuspend 方法,不难猜出 SuspendLambda 继承自 BaseContinuationImp。

Kotlin全解析文档:《Kotlin手册》点击可以查看详细类目

协程上下文与调度器

协程的上下文通常是CoroutineContext类型为代表。这个类型是被定义在Kotlin的标准库中。

在协程中,上下文是各种不同元素的集合。而其中主导作用的元素就是Job。

我们在了解协程的并发与调度的时候涉及到了Job。Kotlin 协程 组合挂起函数和async关键字,实现协程的并发操作 (zinyan.com)

这篇继续深入了解Job。

调度器(Dispatchers )与线程

什么是调度器?调度器就是一个决定了协程在哪个线程或者哪些线程上执行的控制对象。

它可以将协程限制在一个特定的线程执行,也可以把协程分配到一个线程池,或者让协程不受限制约束的进行运行。

协程上下文对象:CoroutineContext。

协程调度器对象:CoroutineDispatcher。

而我们通常在使用launch 或者async时可以通过可选参数定义CoroutineContext 对象。然后它会帮我们指定一个调度器对象。也可以使用Dispatchers 对象,定义调度器

示例:

import kotlinx.coroutines.*
​
fun main() = runBlocking<Unit> {
    // 运行在父协程的上下文中,即 runBlocking 主协程
    launch {
        println("main runBlocking      : 我工作的线程 ${Thread.currentThread().name}")
    }
    //调度到主线层中,并且不受限制
    launch(Dispatchers.Unconfined) { // 
        println("Unconfined            : 我工作的线程 ${Thread.currentThread().name}")
        
            println("这个节点是什么事实结束呢")
    }
    //调度到默认线程
    launch(Dispatchers.Default) { 
        println("Default               : 我工作的线程 ${Thread.currentThread().name}")
    }
    // 调度到一个新的线程之中
    launch(newSingleThreadContext("ZinyanThread")) { 
        println("ZinyanThreadContext   : 我工作的线程 ${Thread.currentThread().name}")
    }
}
//输出
Unconfined            : 我工作的线程 main
Default               : 我工作的线程 DefaultDispatcher-worker-1
ZinyanThreadContext   : 我工作的线程 ZinyanThread
main runBlocking      : 我工作的线程 main

下面介绍上面的四种调度逻辑。

launch{…}:默认情况下,它将会从启动它的协程对象中继承上下文以及调度器。

我们上面的例子就是,从main线程中的runBlocking协程对象中继承了上下文,结果显示运行在了main线程之中。

Dispatchers.Unconfined:是特殊的调度器,上面的例子中是运行在了main线层。但是有一个注释,叫做非受限的调度器。然后可以看到,它的输出是最快最早的。但它仅仅只是运行到第一个挂起点。挂起后,它恢复线程中的协程,而这完全由被调用的挂起函数来决定。非受限的调度器非常适用于执行不消耗 CPU 时间的任务,以及不更新局限于特定线程的任何共享数据(如UI)的协程。

它会默认继承外部协程对象。当它被限制在了调用者线程时,继承自它将会有效地限制协程在该线程运行并且具有可预测的 FIFO 调度。

例子:

fun main() = runBlocking<Unit> {
    // 非受限的——将和主线程一起工作
    launch(Dispatchers.Unconfined) {
        println("Unconfined      : 工作线程 ${Thread.currentThread().name}")
        delay(500)
        println("Unconfined      : 线程延迟 ${Thread.currentThread().name}")
    }
    launch { // 父协程的上下文,主 runBlocking 协程
        println("main runBlocking: 工作线程 ${Thread.currentThread().name}")
        delay(1000)
        println("main runBlocking: 线程延迟 ${Thread.currentThread().name}")
    }
}
//输出
Unconfined      : 工作线程 main
main runBlocking: 工作线程 main
Unconfined      : 线程延迟 kotlinx.coroutines.DefaultExecutor
main runBlocking: 线程延迟 main

所以,该协程的上下文继承自 runBlocking {…} 协程并在 main 线程中运行,当 delay 函数调用的时候,非受限的那个协程在默认的执行者线程中恢复执行。

非受限的调度器是一种高级机制,可以在某些极端情况下提供帮助而不需要调度协程以便稍后执行或产生不希望的副作用, 因为某些操作必须立即在协程中执行。非受限调度器不应该在通常的代码中使用。

Dispatchers.Default:默认调度器。默认调度器使用共享的后台线程池。所以 launch(Dispatchers.Default) { …… } 与 GlobalScope.launch { …… } 使用相同的调度器。

newSingleThreadContext(“MyOwnThread”) :自定义协程线层。为协程的运行启动了一个线程。一个专用的线程是一种非常昂贵的资源。在实际开发中两者都必须被释放,当不再需要的时候,使用 close 函数,或存储在一个顶层变量中使它在整个应用程序中被重用。否则就会出现线程泛滥的情况。

不同线程中的跳转

实现两个协程线程的跳转。示例:

fun main() = runBlocking<Unit> {
    newSingleThreadContext("Ctx1").use { ctx1 ->
        newSingleThreadContext("Ctx2").use { ctx2 ->
            runBlocking(ctx1) {
                println("开启 Ctx1")
                withContext(ctx2) {
                    println("Ctx2 开始工作")
                }
                println("返回到 Ctx1 ")
            }
        }
    }
}
//输出
开启 Ctx1
Ctx2 开始工作
返回到 Ctx1

在这个示例中, 使用runBlocking 显式指定了一个上下文。并且之后在协程中使用withContext来改变协程的上下文,而仍然驻留在相同的协程中。

得到上面的输出结果。在这个例子中,使用的都是newSingleThreadContext()创建的线程,而我们使用了标准库中的use函数来释放该线程。避免线程的滥用。

上下文中的Job

协程中的Job是上下文的一部分,并可以使用coroutineContext [Job] 表达式在上下文中检索它。

示例:

fun main() = runBlocking<Unit> {
    println("My job is ${coroutineContext[Job]}")
}
//输出
My job is BlockingCoroutine{Active}@1de0aca6

那么这个有什么作用呢?例如我们可以查询协程的活动状态

示例:

fun main() = runBlocking<Unit> {
    println("My job is ${coroutineContext[Job]}")
    var s = coroutineContext[Job]?.isActive
    println(s)
}
//输出
true

说明我当前的协程对象是活动的。

而为什么要添加“?” 那是因为对象可能为null。

子协程

当一个协程被其他协程在CoroutineScope中被启动的话,它将会通过CoroutineScope.coroutineContext来继承主协程的上下文。并且这个新协程的Job对象将会成为父协程的子Job对象。

当一个父协程被取消的时候,所有它的子协程也会被递归的取消。

但是,当使用 GlobalScope 来启动一个协程时,则新协程的作业没有父作业。因此它与这个启动的作用域无关且独立运作。

示例:

fun main() = runBlocking<Unit> {
    // 启动一个协程来处理某种传入请求(request)
    val request = launch {
        // 孵化了两个子作业, 其中一个通过 GlobalScope 启动
        GlobalScope.launch {
            println("job1: 我运行在GlobalScope启动的协程中")
            delay(1000)
            println("job1: 等待了1秒,你会发现我不受取消方法的影响")
        }
        // 另一个则承袭了父协程的上下文
        launch {
            delay(100)
            println("job2: 我是一个父协程启动的子协程对象")
            delay(1000)
            println("job2: 等待1秒,如果父协程被取消后,我也将会被取消。这行就不应该打印")
        }
    }
    delay(500)
    request.cancel() // 取消请求(request)的执行
    delay(1000) // 延迟一秒钟来看看发生了什么
    println("main: 整个协程全部取消后")
}
//输出
job1: 我运行在GlobalScope启动的协程中
job2: 我是一个父协程启动的子协程对象
job1: 等待了1秒,你会发现我不受取消方法的影响
main: 整个协程全部取消后

我们通过输出结果就可以看看到。只有job1 的两个方法被执行了。而job2 在取消过程中也被跟着进行了取消。

父协程

我们了解了子协程的概念后,才能比较清晰的明白父协程。

一个父协程总是等待所有的子协程执行结束。父协程并不显式的跟踪所有子协程的启动,并且不必使用 Job.join在最后的时候等待它们:

示例:

fun main() = runBlocking<Unit> {
    // 启动一个协程来处理某种传入请求(request)
    val request = launch {
        repeat(3) { i -> // 启动少量的子协程
            launch  {
                delay((i + 1) * 200L) // 延迟 200 毫秒、400 毫秒、600 毫秒的时间
                println("协程:$i 结束")
            }
        }
        println("返回值:父协程本身已经执行完毕了,但我并没有调用方法明确的关闭所有子协程, 子协程的事务还没有结束")
    }
    request.join() // 等待请求的完成,包括其所有子协程
    println("所有的协程结束")
}
//输出
返回值:父协程本身已经执行完毕了,但我并没有调用方法明确的关闭所有子协程, 子协程的事务还没有结束
协程:0 结束
协程:1 结束
协程:2 结束
所有的协程结束

我们可以看到,父协程的代码已经执行完毕并输出了。但是子协程仍然处于活动状态,那么整个协程就仍然属于活动状态。

当然,我们如果主动调用.cancel() 那么子协程还没有运行完也会被强制结束了。

这就是协程中的父子协程之间的关系了。

给协程命名-方便进行调试

协程如果打印日志的时候,是会有默认Id的。但是如果是在处理一些特定的请求或者逻辑的话

我们给协程进行命名,那我们在调试的时候就能更方便的进行调试了。

给协程命名通常是通过CoroutineName进行处理。

示例:

val v1 = async(CoroutineName("v1coroutine")) {
    delay(500)
    log("Computing v1")
    252
}
val v2 = async(CoroutineName("v2coroutine")) {
    delay(1000)
    log("Computing v2")
    6
}

例如上面就是我将两个协程对象 进行了命名。这种命名结果只有在log日志中才能看到结果。

初始协程时,多元素添加

我们学过载协程中初始化调度器,在上一步也学习了添加协程名称。

那么我们如果在启动的时候这两个配置属性都要进行添加,那么该如何处理?

可以通过+进行拼接。示例:

fun main() = runBlocking<Unit> {
    // 启动一个协程来处理某种传入请求(request)
    launch(Dispatchers.Default + CoroutineName("test")) {
        println("我工作的线程:${Thread.currentThread().name}")
    }
}

作用域-CoroutineScope

作用域我们都理解,就是在指定空间和区域内生效而已。而我们如果在Android开发中,使用Activity启动一个协程来处理网络或者异步IO读取等操作。所有的这个协程应该在Activity被销毁后自动取消,来避免内存泄露。

我们除了可以手动处理,并关闭外,我们还可以在协程构建的时候进行声明它的范围。

示例:

class DemoActivity : AppCompatActivity() {
    //MainScope 是使用 Kotlinx协程库自带的工厂函数。
    // 它是使用Dispatchers.Main作为调度器的适配UI线程
    private val mainScope = MainScope()
​
    //关闭的时候 取消作用域
    fun destroy() {
        mainScope.cancel()
    }
​
    fun doSomething() {
        // 在示例中启动了 10 个协程,且每个都工作了不同的时长
        repeat(10) { i ->
            mainScope.launch {
                delay((i + 1) * 200L) // 延迟 200 毫秒、400 毫秒、600 毫秒等等不同的时间
                println("Coroutine $i is done")
            }
        }
    }
}

然后我们如果关闭activity,协程也会自动进行关闭。

Android 现在在所有具有生命周期的实体中(activity,Fragment等),都对协程作用域提供了一级支持。

局部数据传递

我们如果使用协程,特别是子协程,父协程混杂等等情况。那么如果能够将一些数据在协程与协程之间传递。那么将会大大提高效率。

Kotlin 提供了:ThreadLocal,asContextElement 扩展函数来帮助我们,它们创建了额外的上下文元素, 且保留给定 ThreadLocal 的值,并在每次协程切换其上下文时恢复它。

示例:

import kotlinx.coroutines.*
​
val threadLocal = ThreadLocal<String?>() // declare thread-local variable
​
fun main() = runBlocking<Unit> {
    threadLocal.set("main")
    println("Pre-main, current thread: ${Thread.currentThread()}, thread local value: '${threadLocal.get()}'")
    val job = launch(Dispatchers.Default + threadLocal.asContextElement(value = "launch")) {
        println("Launch start, current thread: ${Thread.currentThread()}, thread local value: '${threadLocal.get()}'")
        yield()
        println("After yield, current thread: ${Thread.currentThread()}, thread local value: '${threadLocal.get()}'")
    }
    job.join()
    println("Post-main, current thread: ${Thread.currentThread()}, thread local value: '${threadLocal.get()}'")
}
//输出
Pre-main, current thread: Thread[main,5,main], thread local value: 'main'
Launch start, current thread: Thread[DefaultDispatcher-worker-1,5,main], thread local value: 'launch'
After yield, current thread: Thread[DefaultDispatcher-worker-1,5,main], thread local value: 'launch'
Post-main, current thread: Thread[main,5,main], thread local value: 'main'

在这个示例中,使用Dispatchers.Default 在线程池中启动了一个新的协程。所以它工作在线程池中的不同线程中,但它仍然具有线程局部变量的值,例如上面就是使用asContextElement 修改get的值从main 改为launch。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/944864.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Sar测试简介

1.Sar 简介 SAR即英语“Specific Absorption Rate”的缩写。SAR值一般指手机产品中电磁波所产生的热能&#xff0c;它是对人体产生影响的衡量数据&#xff0c;单位是W/Kg&#xff08;瓦/公斤&#xff09; 对于测量手机产品的“SAR”&#xff0c;通俗地讲&#xff0c;就是测量手…

STM32f103入门(4)对射式红外传感器计次(外部中断)

中断:在主程序运行过程中&#xff0c;出现了特定的中断触发条件 (中断源)&#xff0c;使得CPU暂停当前正在运行的程序&#xff0c;转而去处理中断程序处理完成后又返回原来被暂停的位置继续运行中断优先级:当有多个中断源同时申请中断时&#xff0c;CPU会根据中断源的轻重缓急进…

国内的化妆品核辐射检测

化妆品核辐射物质检测是指检测化妆品中的放射性物质&#xff0c;包括放射性核素和放射性同位素。这些放射性物质主要来源于环境中的放射性污染&#xff0c;如空气、水和土壤中的放射性物质&#xff0c;以及化妆品生产过程中的放射性污染&#xff0c;如原料、设备、工艺等。化妆…

如何理解IaaS、PaaS、SaaS?盘点受欢迎的八大SaaS平台!

如何理解IaaS、PaaS、SaaS&#xff1f;他们各自的优势在哪里&#xff1f;我们又应该如何选择&#xff1f;本篇将带大家深入浅出的了解IaaS、PaaS、SaaS&#xff0c;并为大家盘点2023最受欢迎的SaaS平台。 通过本文你将了解&#xff1a; IaaS、PaaS、SaaS分别是什么我们该如何理…

高德地图api给点设置闪烁动画

首先高德提供了点动画两种选择和掉落弹跳&#xff0c;可以在属性中定义也可以setAnimation 但是没闪烁 所以css&#xff0c;遍历所有点标记找到要闪烁的点给一个class在deepcss一个动画 js function setshadow(params) { //获取地图所有点标记let data map.getAllOverlays(&…

Mac下使用Homebrew安装MySQL5.7

Mac下使用Homebrew安装MySQL5.7 1. 安装Homebrew & Oh-My-Zsh2. 查询软件信息3. 执行安装命令4. 开机启动5. 服务状态查询6. 初始化配置7. 登录测试7.1 终端登录7.2 客户端登录 参考 1. 安装Homebrew & Oh-My-Zsh mac下如何安装homebrew MacOS安装Homebrew与Oh-My-Zsh…

在字节划水5年被优化,太无情了...

前言 先简单说下&#xff0c;涵哥是某不知名 985 的本硕&#xff0c;17 年毕业加入字节&#xff0c;以“人员优化”的名义无情被裁员&#xff0c;之后跳槽到了有赞&#xff0c;一直从事软件测试的工作。还差一个月也6年了吧&#xff0c;算是在这行的资深划水员。6年的时间也让…

广告英语翻译的原则,你了解多少

我们知道&#xff0c;广告语的特点是通俗化、口语化&#xff0c;能够吸引读者注意&#xff0c;刺激其消费欲望。因此&#xff0c;广告英语翻译也要具有强烈的感染力&#xff0c;达到其预期功能。那么&#xff0c;如何做好广告英语翻译&#xff0c;关于广告英语翻译的原则&#…

SpringBoot集成JWT token实现权限验证

先在pom中引入 JWT依赖 <!-- JWT --> <dependency><groupId>com.auth0</groupId><artifactId>java-jwt</artifactId><version>4.3.0</version> </dependency> 然后引入一个生成的 token 的工具类 import cn.hutool.c…

TCP协议基础

一&#xff1a; TCP协议是什么&#xff1f; TCP协议是基于面向连接&#xff0c;可靠传输&#xff0c;基于字节流的传输层通信协议 1. 面向连接 TCP协议是一种面向连接的协议&#xff0c;意味着在双方在建立数据传输之前&#xff0c;需要进行一个逻辑上的连接&#xff0c;且是…

MySQL的mysql-bin.00xx binlog日志文件的清理

目录 引言手工清理配置自动清理 引言 公司一个项目生产环境mysql数据盘占用空间增长得特别快&#xff0c;经过排查发现是开启了mysql的binlog日志。如果把binlog日志关闭&#xff0c;如果操作万一出现问题&#xff0c;就没有办法恢复数据&#xff0c;很不安全&#xff0c;只能…

手机实景无人直播应该如何操作?

实景无人直播是一种创新的直播方式&#xff0c;通过利用实景场景进行直播&#xff0c;为观众带来真实、沉浸式的观赏和参与体验。本文将深入探讨实景无人直播的技术操作和商业思考&#xff0c;帮助读者全面了解如何搭建和运营这一新型直播模式。 一、实景无人直播的技术操作…

【Seata】00 - Seata Server 部署(Windows、Docker 基于 Jpom)

文章目录 前言参考目录版本说明Windows 部署 seata-server1&#xff1a;下载压缩包2&#xff1a;文件存储模式3&#xff1a;db 存储模式3.1&#xff1a;建表3.2&#xff1a;修改配置文件3.3&#xff1a;启动脚本4&#xff1a;源码部署 Docker 部署 seata-server &#xff08;基…

报名开启!2023 SuperMap开发者大会全议程揭幕

2023年9月13日-15日&#xff0c;2023 SuperMap开发者大会(2023 SuperMap Developer Conference&#xff0c;简称SDC 2023)将在线举办&#xff0c;聚焦地理空间智能和GIS前沿技术&#xff0c;与全体开发者一起&#xff0c;探寻利用新技术简化开发流程的方法&#xff0c;商讨如何…

【pyqt5界面化开发-5】网格布局(九宫格)界面

一、网格布局 需要模块&#xff1a;QGridLayout 代码 import sys from PyQt5.QtGui import QIcon from PyQt5.QtWidgets import QApplication, QWidget, QPushButton, QLabel, QLineEdit, QDesktopWidget, QHBoxLayout, QVBoxLayout, QGroupBox, QRadioButton, QGridLayoutc…

集创北方ICN6211 是一款MIPIDSI转RGB视频桥接IC

ICN6211 1.描述&#xff1a; ICN6211是一个桥接芯片&#xff0c;它接收MIPIDSI输入并发送RGB输出。MIPIDSI最多支持4个车道&#xff0c; 每个车道的最大运行频率为1Gbps&#xff1b;总最大输入带宽为4Gbps&#xff1b;并且还支持MIPI定义的ULPS&#xff08;超 低功耗状态&a…

企业级数据共享规模化模式

数据共享正在成为企业数据战略的重要元素。对于公司而言&#xff0c;Amazon Data Exchange 这样的亚马逊云科技服务提供了与其他公司共享增值数据或从这些数据获利的途径。一些企业希望有一个数据共享平台&#xff0c;他们可以在该平台上建立协作和战略方法&#xff0c;在封闭、…

低功耗低成本BLE发射器

低功耗低成本BLE发射器MG223基于巨微薪火架构的蓝牙射频、基带和协议栈&#xff0c;根据自研蓝牙基带和蓝牙5.1广播的特点&#xff0c;做了面向无线传感器的简化和创新&#xff0c;使之更加适合广泛的轻量级蓝牙无线传感的应用。MG223符合蓝牙5.1广播发射规范&#xff0c;现已获…

44、基于51单片机的热敏电阻的数字温度计(AD1674)(程序+Proteus仿真)

编号&#xff1a;44 基于51单片机的热敏电阻的数字温度计&#xff08;AD1674&#xff09; 功能描述&#xff1a; 本设计由51单片机最小系统AD1674模块热敏电阻模块液晶1602显示模块 1、主控制器是89C82单片机 2、AD1674模块模数转换器进行A/D转换&#xff0c;读取热敏电阻值电…

【JavaWeb 专题】15个最经典的JavaWeb面试题

文章目录 HTTP长连接和短连接HTTP/1.1 与 HTTP/1.0 的区别可扩展性缓存带宽优化长连接消息传递Host 头域错误提示 AjaxAjax 的优势&#xff1a; JSP 和 servlet 有什么区别&#xff1f;定义区别 JSP 的9大内置对象及作用JSP 的 4 种作用域&#xff1f;session 和 cookie 有什么…