IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> Compose原理-视图和数据双向绑定的原理 -> 正文阅读

[移动开发]Compose原理-视图和数据双向绑定的原理

前言:

Compose是天生的声明式编程,自带MVVM的特性。那么这个特性能怎么用的,如何实现数据绑定的呢?其原理是怎么样的呢?那么今天就来试一下。

一.实验Demo

逻辑很简单,就是先显示一个Text,内容为aaa。然后开启线程5S后修改数据为bbb。

首先先是第一版代码:

class ComposeActivity : ComponentActivity() {
    var uiState by mutableStateOf(Student())

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val student = Student()
        setContent {
            Greeting(uiState.name)
        }

        Thread {
            Thread.sleep(5000)
            uiState.name = "bbb"
            Log.i(
                "test",
                "student.name:${student.name}, mutableStateOf.name:${uiState.name}"
            )
        }.start()
    }
}


@Composable
fun Greeting(name: String) {
    Log.i(
        "test",
        "Greeting:${name}"
    )
    Text(text = "Hello $name!")
}

data class Student(
    var name: String = "aaa",
    val id: Int = 0
)

看起来没啥问题,但是一运行,发生5S之后并没有发生变化,显示的内容仍然是aaa。这是为何?

对照着官方的例子发现有一点写错,官方的例子中给Model赋值时用的是类似下面这样的写法:

uiState = uiState.copy(name = "bbb", id = 1)

而我的写法是这样的:

uiState.name = "bbb"

改正之后,果然生效了,5S之后显示内容变成了bbb。

这里为什么要强调我写错了呢?原因你也许猜到了,既然只有copy才生效,那么绑定的原理就在其中。(PS:其实只要把uiState赋值新对象就可以)

二.绑定原理

我们先看一下uiState = uiState.copy(name = "bbb", id = 1)这行代码反编译之后的java代码:

this$0.setUiState(this$0.getUiState().copy(com.xt.composeapp.LiveLiterals.ComposeActivityKt.INSTANCE.String$arg-0$call-copy$arg-0$call-$set-uiState$$fun-$anonymous$$arg-0$call-$init$$$this$call-start$fun-onCreate$class-ComposeActivity(), com.xt.composeapp.LiveLiterals.ComposeActivityKt.INSTANCE.Int$arg-1$call-copy$arg-0$call-$set-uiState$$fun-$anonymous$$arg-0$call-$init$$$this$call-start$fun-onCreate$class-ComposeActivity()));

代码很长,但是其实并不重要,只要看最前面即可:setUiState方法。kotlin中,只要给对象赋值,最终其实都会转化为调用setXXX方法。那我们进入setUiState方法看下。

public final void setUiState(@NotNull Student var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      MutableState $this$setValue$iv = this.uiState$delegate;
      KProperty property$iv = null;
      int $i$f$setValue = false;
      $this$setValue$iv.setValue(var1);//1这行代码
   }

最终转换为了调用$this$setValue$iv的setValue方法。那么第一个对象是什么呢?其实就是我们之前声明的MutableState:

private final MutableState uiState$delegate = SnapshotStateKt.mutableStateOf$default(new Student((String)null, 0, 3, (DefaultConstructorMarker)null), (SnapshotMutationPolicy)null, 2, (Object)null);
  

所以1这行代码,最终其实会调用到MutableState的setValue方法。

MutableState其实是一个接口,最终实现类其实是生成的对Student的一个观测对象。所以调用setValue的时候,判断值发生了变化,则会通知外层去重新刷新UI。

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2022-08-06 10:55:38  更:2022-08-06 10:56:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 4:17:13-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码