kotlin 委托
一、类委托
interface DB{fun insert()
}
class SqliteDB : DB {override fun insert() {println(" SqliteDB insert")}
}class MySql : DB{override fun insert() {println(" MySql insert")}
}class OracleDB : DB{override fun insert() {println(" OracleDB insert")}
}class CreateDB(db:DB) : DB by dbfun main() {val db = CreateDB(MySql())db.insert()
}
运行main()方法:

CreateDB类需要实现的接口(DB)方法委托给了具体实现类(MySql、OracleDB、SqliteDB)来实现,我认为这和设计模式中的策略模式如出一辙。
通过AndroidStudio的自带工具
Tools -> kotlin -> show kotlin Bytecode ,然后点Decompile,将字节码进行反编译得到java代码
也能佐证我们的想法:

二、委托属性
private var floatValue = 789.12
private var num by ::floatValuefun main() {println("num:$num")num = 567.88println("floatValue:$floatValue")
}
运行结果:

这个示例中将属性num委托给了 属性floatValue,即对num进行get和set操作其实都作用在了floatValue上,因此会有上面的运行结果。
同样我们可以看下反编译后的代码:

让我们稍感惊讶的是,经过转换后甚至都没有定义num这个属性,只是增加了getNum和setNum两个方法来获取和改变floatValue的值。
三、自定义委托
方式1:
class Test3{var text:String by StringDelegate()
}class StringDelegate{operator fun getValue(owner:Any,property:KProperty<*>):String{return "delegate value"}operator fun setValue(owner:Any,property:KProperty<*>,value:String){println("value:$value")}
}fun main() {val t = Test3()println("text:${t.text}")t.text = "haha"
}
要点:
- 想要成为可接受委托的类需要实现运算符重载方法,若委托类实现了getValue和setValue(即可读可写),则被委托属性text可用var修饰;若委托类只实现了getValue(即只读),则被委托属性text可用val修饰。
- getValue的返回值,和setValue的value参数值类型要与被委托属性类型一致。
- getValue和setValue中的owner参数必须是被委托属性text所属的类或其父类。
方式2:
class Test3{var text:String by StringDelegate()var text2:String by StringDelegate2()
}class StringDelegate{operator fun getValue(owner:Any,property:KProperty<*>):String{return "delegate value"}operator fun setValue(owner:Any,property:KProperty<*>,value:String){println("value:$value")}
}class StringDelegate2:ReadWriteProperty<Any,String>{override fun getValue(thisRef: Any, property: KProperty<*>): String {return "delegate value2"}override fun setValue(thisRef: Any, property: KProperty<*>, value: String) {println("value2:$value")}}fun main() {val t = Test3()println("text:${t.text}")t.text = "haha"println("text2:${t.text2}")t.text2 = "haha"
}
要点:
- 这种方式的原理就是方式一,只不过是系统Api给我们提供了一种相对方便的使用方式。
- 此方式需要出入两个泛型,第一个必须是被委托属性text所属的类或其父类,第二个是被委托属性的类型
- 若委托类支持可读可写则继承ReadWriteProperty,若只支持可读则继承ReadOnlyProperty
此方式原理:

方式3:
在上面的基础之上还有另一种方式,可以根据属性的一些特征来返回一个合适的委托实现类,有点类似于抽象工厂模式,根据特定的需求返回合适的实现,如下:
class Test3{var text:String by StringDelegate()var text2:String by StringDelegate2()var text3:String by StringDelegateProvide()
}class StringDelegate{operator fun getValue(owner:Any,property:KProperty<*>):String{return "delegate value"}operator fun setValue(owner:Any,property:KProperty<*>,value:String){println("value:$value")}
}class StringDelegate2:ReadWriteProperty<Any,String>{override fun getValue(thisRef: Any, property: KProperty<*>): String {return "delegate value2"}override fun setValue(thisRef: Any, property: KProperty<*>, value: String) {println("value2:$value")}}class StringDelegate3:ReadWriteProperty<Any,String>{override fun getValue(thisRef: Any, property: KProperty<*>): String {return "delegate value3"}override fun setValue(thisRef: Any, property: KProperty<*>, value: String) {println("value3:$value")}}class StringDelegateProvide{operator fun provideDelegate(thisRef: Any,property: KProperty<*>):ReadWriteProperty<Any,String>{return if (property.name.contains("text3")){StringDelegate3()}else{StringDelegate2()}}
}fun main() {val t = Test3()println("text:${t.text}")t.text = "haha"println("text2:${t.text2}")t.text2 = "haha"println("text3:${t.text3}")
}
四、使用案例
通过委托封装Mmkv存取值的过程
import com.blankj.utilcode.util.GsonUtils
import com.tencent.mmkv.MMKV
import kotlin.properties.ReadWriteProperty
import kotlin.reflect.KPropertyopen class MMKVDelegate<T>(private val mmkv: MMKV,private val cls: Class<T>,private val key: String,private val defaultValue: T
) : ReadWriteProperty<Any?, T> {@Suppress("UNCHECKED_CAST")override fun getValue(thisRef: Any?, property: KProperty<*>): T {return when {cls.isEnum -> {val enumValues = cls.enumConstantsval enumValue = mmkv.decodeString(key, null)enumValues?.firstOrNull { it.toString() == enumValue } ?: defaultValue}cls.isData || MMKVDelegatedClass::class.java.isAssignableFrom(cls) -> runCatching {GsonUtils.fromJson(mmkv.decodeString(key), cls)}.getOrNull() ?: defaultValuecls.isAssignableFrom(String::class.java) -> mmkv.decodeString(key, defaultValue as String) as Tcls.isAssignableFrom(Int::class.java) -> mmkv.decodeInt(key, defaultValue as Int) as Tcls.isAssignableFrom(Boolean::class.java) -> mmkv.decodeBool(key, defaultValue as Boolean) as Tcls.isAssignableFrom(Float::class.java) -> mmkv.decodeFloat(key, defaultValue as Float) as Tcls.isAssignableFrom(Long::class.java) -> mmkv.decodeLong(key, defaultValue as Long) as Tcls.isAssignableFrom(Double::class.java) -> mmkv.decodeDouble(key, defaultValue as Double) as Tcls.isAssignableFrom(ByteArray::class.java) -> mmkv.decodeBytes(key, defaultValue as ByteArray) as Telse -> throw IllegalArgumentException("Unsupported type.")}}override fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {val ok = when {cls.isEnum -> mmkv.encode(key, value?.toString())cls.isData || MMKVDelegatedClass::class.java.isAssignableFrom(cls) -> mmkv.encode(key,GsonUtils.toJson(value))cls.isAssignableFrom(String::class.java) -> mmkv.encode(key, value as String)cls.isAssignableFrom(Int::class.java) -> mmkv.encode(key, value as Int)cls.isAssignableFrom(Boolean::class.java) -> mmkv.encode(key, value as Boolean)cls.isAssignableFrom(Float::class.java) -> mmkv.encode(key, value as Float)cls.isAssignableFrom(Long::class.java) -> mmkv.encode(key, value as Long)cls.isAssignableFrom(Double::class.java) -> mmkv.encode(key, value as Double)cls.isAssignableFrom(ByteArray::class.java) -> mmkv.encode(key, value as ByteArray)else -> throw IllegalArgumentException("Unsupported type.")}if (!ok) {throw MMKVDelegateException.EncodeException()}}
}
import kotlin.jvm.internal.Reflectionval <T> Class<T>.isData: Boolean get() = Reflection.getOrCreateKotlinClass(this).isData
假设我们在项目中需要在本地记录一些App行为信息
object AppBehavior {private const val TAG = "AppBehavior"private const val MMKV_ID = "AppBehavior"//记录App启动次数var appStartTime: Int by SettingValue(Int::class.java,Key.APP_START_TIME,0)/*** APP行为的属性委托* 值存储成功时,自动发送UserLocalSettingChangedEvent事件* @param cls 属性的类型* @param key 存储到MMKV的key* @param defaultValue 从MMKV获取失败时,返回的默认值*/private class SettingValue<T>(private val cls: Class<T>,private val key: Key,private val defaultValue: T) : MMKVDelegate<T>(MMKV.mmkvWithID(MMKV_ID), cls, key.name, defaultValue) {override fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {runCatching {super.setValue(thisRef, property, value)}.onFailure {Log.e(TAG, it.toString())if (it !is MMKVDelegateException) {throw it}}}}
}
调用:
AppBehavior.appStartTime++
相关文章:
kotlin 委托
一、类委托 interface DB{fun insert() } class SqliteDB : DB {override fun insert() {println(" SqliteDB insert")} }class MySql : DB{override fun insert() {println(" MySql insert")} }class OracleDB : DB{override fun insert() {println(&quo…...
Stream流的中间方法
一.Stream流的中间方法 注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程 注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据 二.filter filter的主要用法是…...
【车载开发系列】ParaSoft单元测试环境配置(四)
【车载开发系列】ParaSoft单元测试环境配置(四) 【车载开发系列】ParaSoft单元测试环境配置(四) 【车载开发系列】ParaSoft单元测试环境配置(四)一. 如何设置过滤二. 如何设置静态扫描的规则三. 如何设置单…...
IDEA 设置自动定位文件
一、场景分析 IDEA 在使用的过程中,发现有时候,打开一个类,它并不能自动帮我们在左侧 Project 树中定位出文件,需要自己手动点击 瞄准 图标。很不方便。 二、解决方法 1、点击 瞄准 图标旁边的 竖三点 2、将 Alwasy Select Opene…...
Nature Machine Intelligence 基于强化学习的扑翼无人机机翼应变飞行控制
尽管无人机技术发展迅速,但复制生物飞行的动态控制和风力感应能力,仍然遥不可及。生物学研究表明,昆虫翅膀上有机械感受器,即钟形感受器campaniform sensilla,探测飞行敏捷性至关重要的复杂气动载荷。 近日࿰…...
[Web安全 网络安全]-XXE 外部实体注入攻击XML
文章目录: 一:前言 1.定义 1.1 XXE 1.2 XML可扩展标记语言 2.DDT文档类型定义 2.1 分类 2.2 元素element DTD元素 DTD属性 2.3 实体entity DTD实体类别 DTD实体声明引用 声明:内部 外部 参数实体 公共实体 引用:…...
8--苍穹外卖-SpringBoot项目中套餐管理 详解(二)
目录 删除套餐 需求分析和设计 代码开发 根据id查询套餐 mapper层 Service层 ServiceImpl层 Mapper层 批量删除套餐 mapper层 Service层 ServiceImpl层 Mapper层 SetmealMapper.xml 修改套餐 需求分析和设计 代码开发 起售停售套餐 需求分析和设计 代码开发…...
测试面试题:pytest断言时,数据是符点类型,如何断言?
在使用 Pytest 进行断言时,如果数据是浮点类型,可以使用以下方法进行断言: 一、使用pytest.approx pytest.approx可以用来比较两个浮点数是否近似相等。例如: import pytestdef test_float_assertion():result 3.14159expecte…...
Python与MongoDB交互
一、基本概念 MongoDB: 一个面向文档的数据库系统,使用BSON(Binary JSON)作为存储格式。集合(Collection): 类似于关系型数据库中的表,是文档的集合。文档(Document): MongoDB中的基…...
安卓AI虚拟女友项目开发的Android开发环境搭建
第五章:Android开发环境搭建与基础入门 5-1 项目讲解思路说明 本文是安卓AI数字虚拟人项目实战的第五章,开发安卓AI安卓版数字虚拟人的Android基础部分。 在本章中,我们将详细介绍如何搭建Android开发环境,包括Android Studio的…...
基于SpringBoot+Vue+MySQL的智能垃圾分类系统
系统展示 用户前台界面 管理员后台界面 系统背景 随着城市化进程的加速,垃圾问题日益凸显,不仅对环境造成污染,也给城市管理带来了巨大挑战。传统的垃圾分类方式不仅费时费力,而且手工操作容易出现错误,导致垃圾分类效…...
你的个人文件管理助手:AI驱动的本地文件整理工具
🌐 引言 在数字化时代,我们经常面临文件管理的挑战。电脑中的文件杂乱无章,寻找特定文件变得既费时又费力。幸运的是,现在有了一款名为本地文件整理器的神器,它利用AI技术帮助你快速、智能地整理文件,同时…...
【PyTorch】环境配置
框架介绍 Pytorch简介 2017年1月,FAIR(Facebook AI Research)发布了PyTorch。PyTorch是在Torch基础上用python语言重新打造的一款深度学习框架。Torch是采用Lua语言作为接口的机器学习框架,但因为Lua语言较为小众,导…...
枫叶MTS格式转换器- 强大、操作简单的MTS、M2TS视频转换工具供大家学习研究参考
一款功能强大、操作简单的MTS、M2TS视频转换工具,欢迎下载使用。 使用本MTS格式转换器可以帮助您将索尼和松下等摄像机录制的MTS、M2TS格式高清视频转换为其他流行的视频格式,如MP4、3GP、AVI、MPEG、WMV、ASF、MOV、RM、VCD、SVCD、DVD、MKV、FLV、SWF、MPG、MP3、WAV、WMA…...
Vscode把全部‘def‘都收起来的快捷键
在 VSCode 中,你可以使用以下快捷键来收起所有函数定义 (def): Windows/Linux: Ctrl K, Ctrl 0macOS: Cmd K, Cmd 0 这个快捷键组合会折叠当前文件中所有的代码块(包括所有函数和类定义)。你可以通过相同的快捷键再次展开这…...
Web和UE5像素流送、通信教程
一、web端配置 首先打开Github地址:https://github.com/EpicGamesExt/PixelStreamingInfrastructure 找到自己虚幻引擎对应版本的项目并下载下来,我这里用的是5.3。 打开项目找到PixelStreamingInfrastructure-master > Frontend > implementat…...
【YOLO目标检测电梯间电动车与人数据集】共4321张、已标注txt格式、有训练好的yolov5的模型
目录 说明图片示例 说明 数据集格式:YOLO格式 图片数量:4321 标注数量(txt文件个数):4321 标注类别数:2 标注类别名称:person、electricBicycle 数据集下载:电梯间电动车与人数据集 图片示例 数据…...
【网络安全】公钥基础设施
1. PKI 定义 1.1 公钥基础设施的概念 公钥基础设施(Public Key Infrastructure,简称PKI)是一种基于公钥密码学的系统,它提供了一套完整的解决方案,用于管理和保护通过互联网传输的信息。PKI的核心功能包括密钥管理、…...
云原生(四十一)| 阿里云ECS服务器介绍
文章目录 阿里云ECS服务器介绍 一、云计算概述 二、什么是公有云 三、公有云优缺点 1、优点 2、缺点 四、公有云品牌 五、市场占有率 六、阿里云ECS概述 七、阿里云ECS特点 阿里云ECS服务器介绍 一、云计算概述 云计算是一种按使用量付费的模式,这种模式…...
计算机网络:计算机网络体系结构 —— OSI 模型 与 TCP/IP 模型
文章目录 计算机网络体系结构OSI 参考模型TCP/IP 参考模型分层的必要性物理层的主要问题数据链路层的主要问题网络层的主要问题运输层的主要问题应用层的主要问题 分层思想的处理方法发送请求路由器转发接受请求发送响应接收响应 计算机网络体系结构 计算机网络体系结构是指将…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...
Zustand 状态管理库:极简而强大的解决方案
Zustand 是一个轻量级、快速和可扩展的状态管理库,特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...
Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...
2024年赣州旅游投资集团社会招聘笔试真
2024年赣州旅游投资集团社会招聘笔试真 题 ( 满 分 1 0 0 分 时 间 1 2 0 分 钟 ) 一、单选题(每题只有一个正确答案,答错、不答或多答均不得分) 1.纪要的特点不包括()。 A.概括重点 B.指导传达 C. 客观纪实 D.有言必录 【答案】: D 2.1864年,()预言了电磁波的存在,并指出…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...
【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
MFC 抛体运动模拟:常见问题解决与界面美化
在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...
为什么要创建 Vue 实例
核心原因:Vue 需要一个「控制中心」来驱动整个应用 你可以把 Vue 实例想象成你应用的**「大脑」或「引擎」。它负责协调模板、数据、逻辑和行为,将它们变成一个活的、可交互的应用**。没有这个实例,你的代码只是一堆静态的 HTML、JavaScript 变量和函数,无法「活」起来。 …...
Elastic 获得 AWS 教育 ISV 合作伙伴资质,进一步增强教育解决方案产品组合
作者:来自 Elastic Udayasimha Theepireddy (Uday), Brian Bergholm, Marianna Jonsdottir 通过搜索 AI 和云创新推动教育领域的数字化转型。 我们非常高兴地宣布,Elastic 已获得 AWS 教育 ISV 合作伙伴资质。这一重要认证表明,Elastic 作为 …...
