Kotlin 扩展函数与内联函数
Kotlin扩展函数
Kotlin 的扩展函数是 Kotlin 中非常强大且实用的功能。它允许你为现有的类添加新的方法,而不需要修改其源代码。这意味着你可以在已有的类上“扩展”新的功能,使用起来就像是原本就存在这些方法一样。
扩展函数的基本语法
fun 类名.方法名(参数列表): 返回类型 {// 函数体
}
我们去看看Kotlin官方为我们实现的扩展函数例子,就拿官方对Float的扩展方法pow()举例,Float类本身并没有实现乘方功能的方法,在没有这个方法之前,我们只能使用Math类的静态方法进行乘方运算:
Math.pow(2.0, 2.0)
当我们在Kotlin中这样写,编译器会提示我们:

这里告诉我们我们可以用Kotlin的函数如下:
2.0.pow(2.0)//pow函数的实现
public actual inline fun Double.pow(x: Double): Double = nativeMath.pow(this, x)
我们看到这个pow函数是Kotlin对Double类的扩展。我们也可以自己多现有的类进行扩展,例如我们需要打印字符串的最后一个字符,我们可以对String类进行扩展:
fun String.printLastChar() {println(this[length - 1])
}fun main() {//对扩展函数的调用"Hello".printLastChar()//取函数的引用使用invoke进行调用String::printLastChar.invoke("Hello")//直接调用函数引用(String::printLastChar)("Hello")
}
扩展函数的作用域
扩展函数可以定义在Kotlin文件中,是其不属于任何一个类,其他地方都可以使用,类似于上面的String.printLastChar()方法;另外一种是可以声明在类中:
class ExtensionsSample{fun normalFun(){"Java".printLastChar()"Kotlin".printFirstChar()}private fun String.printFirstChar() {println(this[0])}
}
如果扩展函数定义在某个类中,而定义在类中,printFirstChar()方法即是ExtensionsSample类的成员函数,同时又是String类的扩展函数,String类是这个扩展函数的Receiver,定义在类中的扩展函数的作用范围是类及其内部类。在kotlin中也无法通过对象调用定义在其内部的扩展方法,即是限定符是默认的public。
扩展函数的原理
我们对扩展函数所在的文件进行反编译,看其生成的字节码文件反编译的结果如下:
public final void printFirstChar(@NotNull String $this$printFirstChar) {Intrinsics.checkNotNullParameter($this$printFirstChar, "$this$printFirstChar");char var2 = $this$printFirstChar.charAt(0);System.out.println(var2);
}
扩展函数并不是修改原有类的实际代码,它是在编译时通过生成额外的代码来实现的。虽然你能通过扩展函数像访问原有方法一样调用新方法,但它们实际上并没有改变原有类的行为。扩展函数只是通过静态解析(编译时确定类型)来实现的。
因此,扩展函数无法真正覆盖或修改类中的已有方法,也不能访问类的私有成员。
Kotlin扩展属性
Kotlin 还支持扩展属性,它可以为类添加属性,但只能是可读属性(即没有 setter 方法),因为扩展属性不能存储实际的数据。
val 类名.属性名: 类型get() = // 计算属性的值
val String.lastChar: Charget() = this[this.length - 1]fun main() {println("HelloWorld".lastChar)
}
在某些情况下,我们不需要扩展方法,只需要扩展属性即可。
inline
Kotlin官方文档在介绍inline关键字中写道使用不当是会有一定的性能问题的,官方的原话是这样介绍:
Using higher-order functions imposes certain runtime penalties: each function is an object, and it captures a closure. A closure is a scope of variables that can be accessed in the body of the function. Memory allocations (both for function objects and classes) and virtual calls introduce runtime overhead.
使用高阶函数会带来一些运行时的效率损失:每一个函数都是一个对象,并且会捕获一个闭包。 闭包那些在函数体内会访问到的变量的作用域。 内存分配(对于函数对象和类)和虚拟调用会引入运行时间开销。但是在许多情况下通过内联化 lambda 表达式可以消除这类的开销。
- 使用 inline 修饰的函数,会在调用点处将函数的代码“展开”(也叫内联),而不是生成一个实际的函数调用。这意味着函数体的代码会直接插入到调用它的地方,这避免了函数调用所带来的栈帧创建和返回的性能开销。
- 高阶函数指的是接受另一个函数作为参数的函数。在没有 inline 时,传入的函数会在堆栈上创建一个对象,并通过引用进行调用,这会导致额外的内存分配和垃圾回收。使用 inline 后,传入的函数会直接嵌入到调用点,避免了创建和传递函数对象的开销。
- 由于内联函数的代码直接插入到调用的地方,Kotlin 会避免为这些高阶函数(例如 lambda 表达式)分配对象,从而减少了内存的消耗。
如果一个高阶函数在循环中被调用,或者在Android中的onDraw方法中调用,高阶函数在执行的过程中就会创建临时对象,短时间创建大量对象,并且这些对象的生命周期都很短,就会造成内存抖动
fun main() {highOrder(5) {println(it)it}
}fun highOrder(number: Int, callBack: (Int) -> Int) {callBack(number + 1)
}
我们对上面的文件进行反编译可以发现确实是在执行的时候是通过Function1
这个类来实现我们的高阶函数传入的函数
public static final void main() {highOrder(5, (Function1)null.INSTANCE);
}// $FF: synthetic method
public static void main(String[] var0) {main();
}public static final void highOrder(int number, @NotNull Function1 callBack) {Intrinsics.checkNotNullParameter(callBack, "callBack");callBack.invoke(number + 1);
}
而我们给高阶函数添加highOrder前面添加关键字inline之后,让他变成内联函数在经过反编译可以看到:
public static final void main() {int number$iv = 5;int $i$f$highOrder = false;int it = number$iv + 1;int var3 = false;System.out.println(it);
}
而这里就是直接把我们传入的lambda表达式具体的执行逻辑直接放到了这里,由此可见inline关键字的主要作用就是减少调用栈的深度,并能够优化高阶函数在执行时的性能问题。
noinline
如果不希望内联所有传给内联函数的 lambda 表达式参数都内联,那么可以用 noinline 修饰符标记不希望内联的函数参数:
inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) { …… }
假设有一种场景,你的高阶函数传入了两个函数参数,你希望其中的一个函数参数以函数引用的方式作为返回值,就可以使用到noinline关键字:
inline fun foo(callBack: (Int) -> Int, noinline result: () -> Unit): () -> Unit {callBack(2)return result
}
noinline的作用就是局部的、有指向性的关闭Kotlin的内联优化。
crossinline
我们先来看一段代码:
fun main() {println("This is before of heightFunc")heightFunc {return@heightFunc}println("This is after of heightFunc")
}fun heightFunc(call: () -> Unit) {call()println("This is inner of heightFunc")
}
这样写所有的打印的地方都会执行,而使用inline关键字标记heightFunc函数之后,并且去掉return的标签,那么结果就发生了改变
fun main() {println("This is before of heightFunc")heightFunc {return}println("This is after of heightFunc")
}fun heightFunc(call: () -> Unit) {call()println("This is inner of heightFunc")
}
最终的执行结果为只打印了This is before of heightFunc,我们通过内联函数的优化可以想到其中的原因。
当我们需要在内联函数中间接的调用传入的lambda表达式时,编译器会给我们如下提示:

而我们在外面调用内联函数,在lambda表达式中使用return的时候,是不被允许的,要使用必须加标签显示的指定退出那个函数

所以crossinline的作用就是如果你希望禁止 lambda 中的 return 语句影响外部函数的执行,可以使用 crossinline。这块lambda中的return和内联函数结合起来使用具体得多写,编译器会给出提示。
相关文章:
Kotlin 扩展函数与内联函数
Kotlin扩展函数 Kotlin 的扩展函数是 Kotlin 中非常强大且实用的功能。它允许你为现有的类添加新的方法,而不需要修改其源代码。这意味着你可以在已有的类上“扩展”新的功能,使用起来就像是原本就存在这些方法一样。 扩展函数的基本语法 fun 类名.方…...
uniapp中对于文件和文件夹的处理,内存的查询
目录 移动文件到指定文件夹 新增本地文件夹 设定本地文件过期时间,清除超时文件,释放内存 操作本地文件之----删除 uniapp获取设备剩余存储空间的方法 读取本地文件夹下的文件 移动文件到指定文件夹 function moveTempFile(tempFilePath, targetFo…...
【Android开发】安卓手机APP使用机器学习进行QR二维码识别(完整工程资料源码)
前言:本项目是一个 Android 平台的二维码扫描应用,具备二维码扫描和信息展示功能。借助 AndroidX CameraX 库实现相机的预览、图像捕获与分析,使用 Google ML Kit 进行二维码识别。为方便大家了解项目全貌,以下将介绍项目核心代码文件 MainActivity.java 和 AndroidManifes…...
企业文件防泄密软件哪个好?
在企业文件防泄密软件领域,天锐绿盾和中科数安都是备受认可的品牌,它们各自具有独特的特点和优势。 以下是对这两款软件的详细比较: 天锐绿盾 功能特点 集成性强:集成了文件加密、数据泄露防护DLP、终端安全管理、行为审计等数据安…...
mysql 参数max_connect_errors研究
1.在server端设置max_connect_errors3,超过3次连接错误就block mysql> set global max_connect_errors3; Query OK, 0 rows affected (0.00 sec) mysql> show variables like max_connect_errors; --------------------------- | Variable_name | Value…...
linux 下连接mysql(下)
case 表达式 表t1中的数据如下。 select * from t1; ---------------------------- | id | student_no | name | age | ---------------------------- | 3 | 202501 | ll | 10 | | 4 | 202502 | tt | 15 | ----------------------------如果学号是202501,…...
【Qt 常用控件】多元素控件(QListWidget、QTableWidgt、QTreeWidget)
**View和**Widget的区别? **View的实现更底层,**Widget是基于**View封装实现的更易用的类型。 **View使用MVC结构 MVC是软件开发中 经典的 软件结构 组织形式,软件设计模式。 M(model)模型。管理应用程序的核心数据和…...
Linux 远程文件复制传输-----scp/rsync/sftp
scp(Secure Copy Protocol)是基于 SSH 的安全文件传输工具,可用于在本地和远程计算机之间复制文件或目录。 1. scp(基于 SSH 复制文件) a. 复制文件到远程 从本地复制到远程 scp localfile.txt userremote_host:/remo…...
VS2022中.Net Api + Vue 从创建到发布到IIS
VS2022中.Net Api Vue 从创建到发布到IIS 前言一、先决条件二、创建项目三、运行项目四、增加API五、发布到IIS六、设置Vue的发布 前言 最近从VS2019 升级到了VS2022,终于可以使用官方的.Net Vue 组合了,但是使用过程中还是有很多问题,这里记录一下. 一、先决条件 Visual …...
Windows 11 搭建私有知识库(docker、dify、deepseek、ollama)
一、操作系统信息 版本 Windows 11 家庭中文版 版本号 23H2 安装日期 2023/8/21 操作系统版本 22631.4460二、搭建思路 ollama拉取deepseek、bge-m3模型docker拉取dify的镜像dify链接ollama使用模型,并上传文件搭建知识库,创建应用 三、搭建步骤…...
安装OpenJDK21(linux、macos)
文章目录 安装OpenJDK21java21linux下安装配置mac下安装 安装OpenJDK21 java21 封神!Java 21正式发布了,迎来了史诗级新特性,堪称版本最强!!! 视频链接:https://www.bilibili.com/video/BV1E8…...
变分边界详解
起因 当时看VAE论文时有这么一段,但是看完直接一头雾水,这都那跟哪,第一个公式咋做的变换就变出那么一堆。网上搜了很多博客都语焉不详,只好自己来写一篇,希望能解答后来人的疑惑。 公式1 参考文章:证据…...
Softhsm储存安全数据性能整理
目标:存储百万条数据对象 测试方案一:总大小2GB,每个数据对象大小约512KB,总条数4096条; 测试方案一:总大小2GB,每个数据对象大小约256B,总条数8388608条; 测试环境&am…...
KaiOS 4.0 APN List 界面加载debug
问题背景 在列表选中APN进入编辑后,退出返回列表界面时无法焦点选中编辑的APN。 代码分析 路径:gaia/apps/settings/js/panels/apn_list/panel.js 分析SettingsPanel界面加载的步骤逻辑 onBeforeShow -> onShow -> onBeforeHide -> onHide return SettingsPanel(…...
Next.js 15【实用教程】2025最新版
官网 https://nextjs.org/docs/app/getting-started Next.js 简介 Next.js 由 Vercel 开发和维护,旨在解决单页应用(SPA)和多页应用(MPA)在性能和 SEO 上的不足。 核心特性 服务端渲染(SSR)--…...
2025-02-13 学习记录--C/C++-PTA 7-17 爬动的蠕虫
一、题目描述 ⭐️ 二、代码(C语言)⭐️ #include <stdio.h>int main() {int N, U, D; // N: 井的总高度,U: 每分钟向上爬的高度,D: 每分钟滑下的高度int height 0; // 蠕虫当前的高度int minute 0; // 蠕虫爬行的时间sc…...
Elasticsearch+Logstash+Kibana可视化集群部署
文章目录 1.组件介绍简述2.集群规划3.Es组件部署4.Logstash组件部署5.Kibana组件部署6.Kibana的基础使用 1.组件介绍简述 Elasticsearch:开源实时分布式搜索和分析引擎,支持大规模数据存储和高吞吐量,提供丰富的搜索功能和可扩展性。 Logsta…...
React VS Vue
React 和 Vue 是目前最流行的两个前端框架,它们在设计理念、生态系统和开发体验上各有特点。以下是对 React 和 Vue 的全方位对比: 1. 核心设计理念 React 库而非框架:React 是一个用于构建 UI 的库,专注于视图层,其…...
DeepSeek+Excel 效率翻倍
2025年初,DeepSeek以惊人的效率突破技术壁垒,用极低的成本实现了与行业顶尖AI相媲美的性能,瞬间成为全球科技领域的热门话题。 那么AI工具的普及将如何改变我们的工作方式?Excel会被取代吗? 今天,珠珠带你…...
将Sqlite3数据库挂在内存上处理
创作灵感:最近把小学生的口算题从2位数改到3位数,100以内四则运算练习(千纬数学)再次更新,选取难题-CSDN博客要不断刷题目,以前100以内的加减乘除也是这样刷出来的,代码如下: impor…...
Vue3.5 企业级管理系统实战(六):Vue3中defineProps用法
上一节封装图标组件 SvgIcon 时,用到了 defineProps,因为它在开发中的重要性,这里简单看一下它的用法,已熟知用法的此节可跳过。 在 Vue3 的组件化开发体系里,组件间通信是构建高效、可维护应用程序的核心环节。defin…...
HTTP/2 由来及特性
HTTP/2 的由来 HTTP/1.x 的局限性 性能瓶颈 队头阻塞问题:在HTTP/1.x中,一个TCP连接在同一时间只能处理一个请求,后续请求必须等待前面的请求处理完成并收到响应后才能被处理。例如,当一个页面有多个资源(如图片、脚…...
electron.vite 项目创建以及better-sqlite3数据库使用
1.安装electron.vite npm create quick-start/electronlatest中文官网:https://cn.electron-vite.org/ 2. 安装项目依赖 npm i3.修改 electron-builder 配置文件 appId: com.electron.app productName: text33 directories:buildResources: build files:- !**/.v…...
蓝桥杯 Java B 组之枚举算法(暴力破解)
Day 3:枚举算法(暴力破解) 枚举算法(Brute Force)是一种 暴力搜索 方法,它通过 遍历所有可能的情况 来找到正确答案。虽然它的 时间复杂度较高,但在 数据范围较小 时,它是一种简单且…...
AI 控制web浏览器基础知识准备,名词解释Xvfb,x11vnc,novnc,playwright,gradio
在探索如何让AI控制Web浏览器实现自动化任务时,了解底层技术栈是关键。本文将解析五个核心组件:Xvfb、x11vnc、novnc、playwright和gradio,这些工具共同构成了AI驱动浏览器的基础架构。 1. Xvfb(X Virtual Framebuffer࿰…...
C++,STL容器适配器,stack:栈深入解析
文章目录 一、容器概览与核心特性核心特性速览二、底层实现原理1. 容器适配器设计2. 默认容器对比三、核心操作详解1. 容器初始化2. 元素操作接口3. 自定义栈实现四、实战应用场景1. 括号匹配校验2. 浏览器历史记录管理五、性能优化策略1. 底层容器选择基准2. 内存预分配技巧六…...
Vue笔记(十)
一、AI的基本认知 二、ChatGPT的基本使用 三、AI插件--Copilot入门 1.Copilot是由OpenAI和GitHub合作开发的AI编程辅助插件,基于大量代码训练,能根据上下文自动生成代码建议。 2.安装与配置:在常用代码编辑器(如Visual Studio Cod…...
Ubuntu下载安装Docker-Desktop
下载 Ubuntu | Docker Docs 预备工作 Ubuntu增加docker apt库-CSDN博客 安装 sudo apt-get updatesudo apt install gnome-terminal# sudo apt install -y docker-composesudo apt-get install ./docker-desktop-amd64.deb 测试 sudo docker run hello-worldHello from D…...
DeepSeek 突然来袭,AI 大模型变革的危机与转机藏在哪?
随着人工智能技术的飞速发展,大模型领域不断涌现出具有创新性的成果。DeepSeek 的横空出世,为 AI 大模型领域带来了新的变革浪潮。本文将深入探讨 DeepSeek 出现后 AI 大模型面临的危机与转机。 冲冲冲!!! 目录 一、…...
C#运动控制——轴IO映射
1、IO映射的作用 该功能允许用户对专用 IO 信号的硬件输入接口进行任意配置,比如轴的急停信号,通过映射以后,可以将所有轴的急停信号映射到某一个IO输入口上,这样,我们只要让一个IO信号有效就可以触发所有轴的急停。 进…...
