Kotlin 函数
文章目录
- 函数的定义
- 函数的返回值
- 参数默认值 & 调用时参数指定
- 函数作用域
- Lambda 表达式
- 匿名函数
- 内联函数
- 扩展函数
- 中缀函数
- 递归函数 & 尾递归函数
函数的定义
函数可以理解成一个小小的加工厂,给入特定的原材料,它就会给出特定的产品。
fun [接收者类型.][函数名]([参数名: 参数类型], ...)[:返回值类型] [函数体]
我们只需要写一次,便可以反复使用:
fun greeting(name: String) {println("Hello $name!")
}fun main() {greeting("Today")greeting("Kotlin")
}
Hello Today
Hello Kotlin
Note:调用函数时,函数参数有多个,且未指定参数名,需要按参数顺序传入参数。
函数的返回值
函数都会有返回值,默认的返回值是Unit
(无论何时,Unit
都是唯一的一个值(单例),不可以被实例化),可以省略不写:
fun greeting(name: String): Unit {println("Hello $name!")return Unit// return
}
函数的返回值会给到调用处,也就是说下方的getResult()
就代表了默认返回值Unit
。我们可以直接打印出返回值,或者将其赋值给变量:
fun getResult() {}fun main() {val result: Unit = getResult()println(result)print(getResult())
}
kotlin.Unit
kotlin.Unit
我们也可以指定返回其他类型:
fun plus1(old: Int): Int {return old + 1
}fun main() {val new = plus1(0)print(new)
}
1
甚至可以返回一个函数(这可能有点超纲了,看不懂可以先不理,请参考下方Lambda 表达式和匿名函数):
这里的getFun
返回了一个无参数的、返回值为String
类型的函数。
fun getFun(): () -> String {return { "Hello World" }
}fun main() {val greet = getFun()val msg = greet()print(msg)
}
Hello World
Note:Kotlin 中函数类型的书写格式是(参数名和冒号
:
可省略):([参数名: 参数类型], ...) -> [返回值类型]
如
fun main() {val myFun: (Int) -> Int = {it + 1}print(myFun(0)) }
1
与之类似的写法是:
fun main() {fun myFun(int: Int): Int {return int + 1}print(myFun(0)) }
或者
fun main() {val myFun = fun(int: Int): Int {return int + 1}print(myFun(0)) }
一个函数可以作为另一个函数的参数:
fun caller(block: () -> Unit) {// 调用 block 函数block()
}fun main() {// 调用 caller 函数,传入 lambdacaller({ print("Hello World") })
}
Hello World
如果函数体可以用一句表达式表示,则可以直接用等号=
,返回值类型也可以省略:
fun getYear() = 2024fun getInt(boolean: Boolean) = if (boolean) 1 else 0fun getBoolean(int: Int) = when(int) {0 -> falseelse -> true
}fun main() {println(getYear())println(getInt(true))print(getBoolean(1))
}
2024
1
true
参数默认值 & 调用时参数指定
当某个参数为可选时,可以为参数赋默认值(我们一般将可选参数放于必要参数之后,但这不是硬性要求):
fun greet(otherParams: Int, name: String = "Kotlin") {print("Hello $name")
}fun main() {greet(0)
}
Hello Kotlin
我们也可以在调用时指定参数传值(下方的exampleParam
):
fun greet(name: String, exampleParam: Int = 0, exampleParam1: Int = 1) {println("Hello $name")print("exampleParam: $exampleParam, exampleParam1: $exampleParam1")
}fun main() {greet("Kotlin", exampleParam = 2)
}
Hello Kotlin
exampleParam: 2, exampleParam1: 1
函数作用域
在函数中定义的变量、函数,在函数外无法访问:
fun main() {fun funScope() {val name = "Kotlin"fun innerFun() {}}// print(name) 无法访问// innerFun() 无法访问
}
函数内可以访问函数外定义的变量、函数:
fun main() {val name = "Kotlin"fun outerFun() { print("Outer Fun") }fun funScope() {// 可以调用外部内容println(name)outerFun()}funScope() // 我们需要调用函数才能看到其运行结果
}
Kotlin
Outer Fun
Lambda 表达式
我们已经不止一次提到它了,它的写法是这样的:
// 匿名函数
{ [参数名: 参数类型], ... ->[函数体]
}
不指明类型的函数类型变量,它的类型会是() -> Unit
。
当 lambda 的参数只有一个时,这个参数可以省略不写,通过it
指定:
fun main() {val myLambda: (String) -> Unit = {print("Hello $it")}
}
当 lambda 作为函数的最后一个参数进行传递时,可以将花括号{}
移到调用的小括号()
外面,称为尾部 lambda(trailing lambda):
fun caller(name: String, block: () -> Unit) {}fun main() {caller("Kotlin", { print("不移出小括号") })caller("Kotlin") {print("移出小括号")}
}
若没有return
,lambda 表达式的最后一个值会作为函数的返回值:
fun caller(block: () -> String) {print(block())
}fun main() {caller { "ABC" }
}
ABC
匿名函数
匿名函数无需写函数名(lambda 表达式也是匿名函数):
fun([参数名: 参数类型], ...)[: 返回值类型] [函数体]
fun main() {val mySingleLineFun = fun() = 1val myFun = fun(name: String) {print("Hello $name")}println(mySingleLineFun())myFun("Kotlin")
}
1
Hello Kotlin
内联函数
内联函数可以将参数中 lambda 表达式的代码插入到函数调用处,提高性能。声明内联函数只需要在fun
前加inline
。内联函数会使编译后的代码更加庞大,我们必须在最合适的时候使用它(错误使用时 IDEA 会警告)。用得比较多的场景是函数有参数为函数类型。
inline fun caller(block: () -> String) {print(block())
}fun main() {caller { "ABC" }
}
print
或println
因为参数类型为Any
,可能为函数类型。可以看到 JVM 平台它们的实现(actual
)其实是 Java 的System.out.print
或System.out.println
(可以通过按住Ctrl
键,鼠标点击函数,跳转到函数声明处):
//
@kotlin.internal.InlineOnly
public actual inline fun print(message: Any?) {System.out.print(message)
}...@kotlin.internal.InlineOnly
public actual inline fun println(message: Any?) {System.out.println(message)
}
如果不希望某一函数类型的参数被内联时,可以将其标记为noinline
:
inline fun caller(noinline noinlineBlock: () -> Unit
) {}
当内联函数(下方call
)可内联的函数类型参数(下方block
)被传入的非内联函数(下方noinlineBlock
)调用时,需要标记为crossinline
:
inline fun caller(noinline noinlineBlock: () -> Unit
) {}inline fun call(crossinline block: () -> Unit) {caller{ block() }
}
扩展函数
还记得我们最开始说的函数类型声明吗?这里有一个接收者:
fun [接收者类型.][函数名]([参数名: 参数类型], ...)[:返回值类型] [函数体]
我们可以通过声明接收者,将某一函数定义为接收者所拥有的函数,称为其扩展函数。
这可能有点难以理解,因为我们还没有讲到类,我在这里做一个简单的解释:
- 我们定义了一个以
Int
作为接收者的函数add
用于对Int
类型数值加上(+
)值other
。 - 在该函数中,
this
会指代接收者Int
类型,例如这里调用int.add
,在add
中this + other
相当于0 + 3
,结果为3
,会返回到调用处。
fun Int.add(other: Int) = this + otherfun main() {val int = 0print(int.add(3))
}
3
中缀函数
我们可以通过 infix
关键字将一个函数声明为中缀函数,我们还是以上方扩展函数为例(因为中缀函数必须为扩展函数或成员方法,而且有且仅有一个参数)。
可以看到运行结果其实是一样的,只是在调用时可以将int.add(3)
写成int add 3
,函数名作为类似运算符的存在。
infix fun Int.add(other: Int) = this + otherfun main() {val int = 0println(int.add(3))print(int add 3)
}
3
3
递归函数 & 尾递归函数
递归就是一个函数自己调用自己。
fun myFun() {myFun()
}
我们细想一下就会发现这样是不可取的,myFun
调用了myFun
,而被调用的myFun
又调了myFun
······看来一时半会是停不了了。
如果我们给递归加上条件,当start == end
时才递归,它就能够停下来:
fun countTo(end: Int, start: Int = 0) {println("现在是: $start")if (start == end) returnelse countTo(end, start + 1)
}fun main() {countTo(5)
}
现在是: 0
现在是: 1
现在是: 2
现在是: 3
现在是: 4
现在是: 5
其实就有点像循环:
fun main() {var start = 0val end = 5while (start <= end) {println("现在是: $start")start ++}
}
当递归调用在末尾时,可以在fun
前加tailrec
,使函数成为尾递归函数(tail recursive functions),编译器会优化该递归,生成一个循环(参考示例)。
相关文章:
Kotlin 函数
文章目录 函数的定义函数的返回值参数默认值 & 调用时参数指定函数作用域Lambda 表达式匿名函数内联函数扩展函数中缀函数递归函数 & 尾递归函数 函数的定义 函数可以理解成一个小小的加工厂,给入特定的原材料,它就会给出特定的产品。 fun [接…...

动态路由协议实验——RIP
动态路由协议实验——RIP 什么是RIP RIP(Routing Information Protocol,路由信息协议)是一种内部网关协议(IGP),是一种动态路由选择协议,用于自治系统(AS)内的路由信息的传递。RIP协议基于…...

数据结构 | 二叉树(基本概念、性质、遍历、C代码实现)
1.树的基本概念 树是一种 非线性 的数据结构,它是由n(n>0)个有限结点组成一个具有层次关系的集合。 把它叫做树是因 为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。 有一个特殊的结点,称为根…...
很多Oracle中的SQL语句在EF中写不出来
很多复杂的Oracle SQL语句在Entity Framework(EF)中很难直接表达出来。虽然EF提供了一种方便的方式来使用C#代码查询和操作数据库,但它在处理某些复杂的SQL特性和优化时可能会有局限性。 以下是一些在EF中可能难以直接实现的Oracle SQL功能和…...
浏览器打开PHP文件弹出下载而不是运行代码
说明 使用phpstudy,极少会出现这种情况。 这里主要是帮助大家理解,为什么上传的木马不运行。 问题原因 首先需要理解,访问PHP文件弹出下载,说明服务端的容器(比如Apache或者Nginx)把文件当成了一个普通二…...
安卓自定义UI组件开发流程
安卓自定义ui组件开发流程 开发安卓自定义UI组件的流程大致可以分为以下几个步骤: 确定需求和设计: 确定需要自定义的UI组件的功能和外观。设计组件的交互逻辑和视觉效果。 创建自定义组件类: 创建一个新的Java类,继承自View、V…...

【LINUX】LINUX基础(目录结构、基本权限、基本命令)
文章目录 LINUX的目录结构LINUX的基本权限LINUX基本命令 LINUX的目录结构 /:表示根目录bin:存放二进制可执行文件(命令ls、cat、mkdir等)boot:存放系统引导文件dev:存放设备文件etc:存放系统配置文件home:…...

Aigtek功率放大器的主要性能要求有哪些
功率放大器是电子系统中的重要组件,用于将低功率信号放大到高功率水平。功率放大器的性能直接影响到信号的放大质量和系统的整体性能。下面西安安泰将介绍功率放大器的主要性能要求。 增益:功率放大器应当具有足够的增益,即将输入信号的幅度放…...

2024.5.29晚训参考代码
因为本套题没有BFS例题,所以我先把BFS模板放着 #include<bits/stdc.h> using namespace std; int n,m;//n*m的棋盘 int dis[402][402]; bool vis[402][402]; int X[]{-2,-2,-1,-1,1,1,2,2};//偏移量的表 int Y[]{-1,1,-2,2,-2,2,-1,1};//定义一个数组&…...

【计算机网络】——概述(图文并茂)
概述 一.信息时代的计算机网络二.互联网概述1.网络,互连网,互联网(因特网)1.网络2.互连网3.互联网(因特网) 2.互联网简介1.互联网发展的三个阶段2.互联网服务提供者(ISP)3.互联网的组…...

C语言多个源程序编译的CMakeList文件编写/源程序生成动态库
1.编译多个源程序时CMakeLists文件编写 1.若源程序目录结构如下: main.cpp中include“LCD_2inch4.h”头文件,而LCD_2inch4.h中include其它源程序,则CmakeLists.txt文件可为如下: # 设置项目名称 cmake_minimum_required(VERSI…...
C# list集合
一、list集合基本使用 1.添加元素 ① 单个元素添加 List<int> list new List<int>();for (int i 0; i < 3; i){list.Add(i);}//输出:0,1,2 ②初始化时添加元素 List<int> list2 new List<int> { 1, 2, 3 };//输出:0,1…...
****三次握手和四次挥手
一、三次握手 1.简要描述TCP三次握手的过程 第一次握手,客户端发送SYN包到服务器; 第二次握手,服务器收到SYN包,回复一个SYNACK包; 第三次握手,客户端收到服务器的SYNACK包后,回复一个ACK包…...

开发语言Java+前端框架Vue+后端框架SpringBoot开发的ADR药物不良反应监测系统源码 系统有哪些优势?
开发语言Java前端框架Vue后端框架SpringBoot开发的ADR药物不良反应监测系统源码 系统有哪些优势? ADR药物不良反应监测系统具有多个显著的优势,这些优势主要体现在以下几个方面: 一、提高监测效率与准确性: 通过自动化的数据收集…...
问题排查|记录一次基于mymuduo库开发的服务器错误排查(段错误--Segmentation fault (core dumped))
问题记录: 在刚完成mymuduo库之后,写了一个简单的测试服务器, 但是在服务器运行后直接报错: cherryhcss-ecs-4995:~/mymuduo/example$ ./testserver Segmentation fault (core dumped)出现多错误这通常意味着程序试图访问其内存空…...
Mysql常用操作DQL数据库、表操作:
DQL是指MySQL数据库中的数据查询语言(Data Query Language)。它是用来从数据库中检索所需数据的语言。DQL允许用户通过指定查询条件和筛选条件来检索数据库中的数据,并以所需的方式来显示结果。DQL语句可以用于从单个表中查询数据,…...
标题:Go语言中的YAML魔法:轻松配置你的环境
摘要: 本文将介绍如何在Go语言项目中使用YAML文件来管理配置,包括如何读取YAML文件以及如何在代码中解析和使用这些配置。 正文: 在编程世界中,配置管理是每个项目都必须面对的问题。对于Go语言项目来说,YAML文件是一…...

STM32高级控制定时器之输入捕获模式
目录 概述 1 输入捕获模式 1.1 原理介绍 1.2 实现步骤 1.3 发生输入捕获流程 2 使用STM32Cube配置工程 2.1 软件环境 2.2 配置参数 2.3 生成项目文件 3 功能实现 3.1 PWM调制占空比函数 3.2 应用函数库 4 测试 4.1 功能框图 4.2 运行结果 源代码下载地址…...
使用 Vue 3 和 qrcode.js 开发二维码显示组件
二维码在现代应用中广泛使用,例如支付、身份验证、链接分享等。本文将介绍如何使用 Vue 3 和 qrcode.js 库来创建一个灵活的二维码显示组件,并展示如何在应用中使用它。 1. 安装必要的依赖 首先,我们需要安装 Vue 3 和 qrcode.js。如果你还…...

LabVIEW异步编程概述
LabVIEW异步编程是一种在图形化编程环境中处理并行任务的方法。通过异步执行,可以提高程序的响应速度和资源利用效率,使得多个任务可以独立进行而不互相干扰。 原理 LabVIEW异步编程的核心在于使用异步调用节点(Asynchronous Call By Refer…...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)
0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述,后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作,其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

云原生玩法三问:构建自定义开发环境
云原生玩法三问:构建自定义开发环境 引言 临时运维一个古董项目,无文档,无环境,无交接人,俗称三无。 运行设备的环境老,本地环境版本高,ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...

AI+无人机如何守护濒危物种?YOLOv8实现95%精准识别
【导读】 野生动物监测在理解和保护生态系统中发挥着至关重要的作用。然而,传统的野生动物观察方法往往耗时耗力、成本高昂且范围有限。无人机的出现为野生动物监测提供了有前景的替代方案,能够实现大范围覆盖并远程采集数据。尽管具备这些优势…...
站群服务器的应用场景都有哪些?
站群服务器主要是为了多个网站的托管和管理所设计的,可以通过集中管理和高效资源的分配,来支持多个独立的网站同时运行,让每一个网站都可以分配到独立的IP地址,避免出现IP关联的风险,用户还可以通过控制面板进行管理功…...

协议转换利器,profinet转ethercat网关的两大派系,各有千秋
随着工业以太网的发展,其高效、便捷、协议开放、易于冗余等诸多优点,被越来越多的工业现场所采用。西门子SIMATIC S7-1200/1500系列PLC集成有Profinet接口,具有实时性、开放性,使用TCP/IP和IT标准,符合基于工业以太网的…...
鸿蒙(HarmonyOS5)实现跳一跳小游戏
下面我将介绍如何使用鸿蒙的ArkUI框架,实现一个简单的跳一跳小游戏。 1. 项目结构 src/main/ets/ ├── MainAbility │ ├── pages │ │ ├── Index.ets // 主页面 │ │ └── GamePage.ets // 游戏页面 │ └── model │ …...

FFmpeg avformat_open_input函数分析
函数内部的总体流程如下: avformat_open_input 精简后的代码如下: int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...
微服务通信安全:深入解析mTLS的原理与实践
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、引言:微服务时代的通信安全挑战 随着云原生和微服务架构的普及,服务间的通信安全成为系统设计的核心议题。传统的单体架构中&…...
WEB3全栈开发——面试专业技能点P7前端与链上集成
一、Next.js技术栈 ✅ 概念介绍 Next.js 是一个基于 React 的 服务端渲染(SSR)与静态网站生成(SSG) 框架,由 Vercel 开发。它简化了构建生产级 React 应用的过程,并内置了很多特性: ✅ 文件系…...