当前位置: 首页 > news >正文

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" }
}

printprintln因为参数类型为Any,可能为函数类型。可以看到 JVM 平台它们的实现(actual)其实是 Java 的System.out.printSystem.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,在addthis + 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 表达式匿名函数内联函数扩展函数中缀函数递归函数 & 尾递归函数 函数的定义 函数可以理解成一个小小的加工厂&#xff0c;给入特定的原材料&#xff0c;它就会给出特定的产品。 fun [接…...

动态路由协议实验——RIP

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

数据结构 | 二叉树(基本概念、性质、遍历、C代码实现)

1.树的基本概念 树是一种 非线性 的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一个具有层次关系的集合。 把它叫做树是因 为它看起来像一棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的。 有一个特殊的结点&#xff0c;称为根…...

很多Oracle中的SQL语句在EF中写不出来

很多复杂的Oracle SQL语句在Entity Framework&#xff08;EF&#xff09;中很难直接表达出来。虽然EF提供了一种方便的方式来使用C#代码查询和操作数据库&#xff0c;但它在处理某些复杂的SQL特性和优化时可能会有局限性。 以下是一些在EF中可能难以直接实现的Oracle SQL功能和…...

浏览器打开PHP文件弹出下载而不是运行代码

说明 使用phpstudy&#xff0c;极少会出现这种情况。 这里主要是帮助大家理解&#xff0c;为什么上传的木马不运行。 问题原因 首先需要理解&#xff0c;访问PHP文件弹出下载&#xff0c;说明服务端的容器&#xff08;比如Apache或者Nginx&#xff09;把文件当成了一个普通二…...

安卓自定义UI组件开发流程

安卓自定义ui组件开发流程 开发安卓自定义UI组件的流程大致可以分为以下几个步骤&#xff1a; 确定需求和设计&#xff1a; 确定需要自定义的UI组件的功能和外观。设计组件的交互逻辑和视觉效果。 创建自定义组件类&#xff1a; 创建一个新的Java类&#xff0c;继承自View、V…...

【LINUX】LINUX基础(目录结构、基本权限、基本命令)

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

Aigtek功率放大器的主要性能要求有哪些

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

2024.5.29晚训参考代码

因为本套题没有BFS例题&#xff0c;所以我先把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.网络&#xff0c;互连网&#xff0c;互联网&#xff08;因特网&#xff09;1.网络2.互连网3.互联网&#xff08;因特网&#xff09; 2.互联网简介1.互联网发展的三个阶段2.互联网服务提供者&#xff08;ISP&#xff09;3.互联网的组…...

C语言多个源程序编译的CMakeList文件编写/源程序生成动态库

1.编译多个源程序时CMakeLists文件编写 1.若源程序目录结构如下&#xff1a; main.cpp中include“LCD_2inch4.h”头文件&#xff0c;而LCD_2inch4.h中include其它源程序&#xff0c;则CmakeLists.txt文件可为如下&#xff1a; # 设置项目名称 cmake_minimum_required(VERSI…...

C# list集合

一、list集合基本使用 1.添加元素 ① 单个元素添加 List<int> list new List<int>();for (int i 0; i < 3; i){list.Add(i);}//输出&#xff1a;0,1,2 ②初始化时添加元素 List<int> list2 new List<int> { 1, 2, 3 };//输出&#xff1a;0,1…...

****三次握手和四次挥手

一、三次握手 1.简要描述TCP三次握手的过程 第一次握手&#xff0c;客户端发送SYN包到服务器&#xff1b; 第二次握手&#xff0c;服务器收到SYN包&#xff0c;回复一个SYNACK包&#xff1b; 第三次握手&#xff0c;客户端收到服务器的SYNACK包后&#xff0c;回复一个ACK包…...

开发语言Java+前端框架Vue+后端框架SpringBoot开发的ADR药物不良反应监测系统源码 系统有哪些优势?

开发语言Java前端框架Vue后端框架SpringBoot开发的ADR药物不良反应监测系统源码 系统有哪些优势&#xff1f; ADR药物不良反应监测系统具有多个显著的优势&#xff0c;这些优势主要体现在以下几个方面&#xff1a; 一、提高监测效率与准确性&#xff1a; 通过自动化的数据收集…...

问题排查|记录一次基于mymuduo库开发的服务器错误排查(段错误--Segmentation fault (core dumped))

问题记录&#xff1a; 在刚完成mymuduo库之后&#xff0c;写了一个简单的测试服务器&#xff0c; 但是在服务器运行后直接报错&#xff1a; cherryhcss-ecs-4995:~/mymuduo/example$ ./testserver Segmentation fault (core dumped)出现多错误这通常意味着程序试图访问其内存空…...

Mysql常用操作DQL数据库、表操作:

DQL是指MySQL数据库中的数据查询语言&#xff08;Data Query Language&#xff09;。它是用来从数据库中检索所需数据的语言。DQL允许用户通过指定查询条件和筛选条件来检索数据库中的数据&#xff0c;并以所需的方式来显示结果。DQL语句可以用于从单个表中查询数据&#xff0c…...

标题:Go语言中的YAML魔法:轻松配置你的环境

摘要&#xff1a; 本文将介绍如何在Go语言项目中使用YAML文件来管理配置&#xff0c;包括如何读取YAML文件以及如何在代码中解析和使用这些配置。 正文&#xff1a; 在编程世界中&#xff0c;配置管理是每个项目都必须面对的问题。对于Go语言项目来说&#xff0c;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 运行结果 源代码下载地址&#xf…...

使用 Vue 3 和 qrcode.js 开发二维码显示组件

二维码在现代应用中广泛使用&#xff0c;例如支付、身份验证、链接分享等。本文将介绍如何使用 Vue 3 和 qrcode.js 库来创建一个灵活的二维码显示组件&#xff0c;并展示如何在应用中使用它。 1. 安装必要的依赖 首先&#xff0c;我们需要安装 Vue 3 和 qrcode.js。如果你还…...

LabVIEW异步编程概述

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

在rocky linux 9.5上在线安装 docker

前面是指南&#xff0c;后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...

深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法

深入浅出&#xff1a;JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中&#xff0c;随机数的生成看似简单&#xff0c;却隐藏着许多玄机。无论是生成密码、加密密钥&#xff0c;还是创建安全令牌&#xff0c;随机数的质量直接关系到系统的安全性。Jav…...

QT: `long long` 类型转换为 `QString` 2025.6.5

在 Qt 中&#xff0c;将 long long 类型转换为 QString 可以通过以下两种常用方法实现&#xff1a; 方法 1&#xff1a;使用 QString::number() 直接调用 QString 的静态方法 number()&#xff0c;将数值转换为字符串&#xff1a; long long value 1234567890123456789LL; …...

根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:

根据万维钢精英日课6的内容&#xff0c;使用AI&#xff08;2025&#xff09;可以参考以下方法&#xff1a; 四个洞见 模型已经比人聪明&#xff1a;以ChatGPT o3为代表的AI非常强大&#xff0c;能运用高级理论解释道理、引用最新学术论文&#xff0c;生成对顶尖科学家都有用的…...

【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习

禁止商业或二改转载&#xff0c;仅供自学使用&#xff0c;侵权必究&#xff0c;如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...

JAVA后端开发——多租户

数据隔离是多租户系统中的核心概念&#xff0c;确保一个租户&#xff08;在这个系统中可能是一个公司或一个独立的客户&#xff09;的数据对其他租户是不可见的。在 RuoYi 框架&#xff08;您当前项目所使用的基础框架&#xff09;中&#xff0c;这通常是通过在数据表中增加一个…...

人工智能(大型语言模型 LLMs)对不同学科的影响以及由此产生的新学习方式

今天是关于AI如何在教学中增强学生的学习体验&#xff0c;我把重要信息标红了。人文学科的价值被低估了 ⬇️ 转型与必要性 人工智能正在深刻地改变教育&#xff0c;这并非炒作&#xff0c;而是已经发生的巨大变革。教育机构和教育者不能忽视它&#xff0c;试图简单地禁止学生使…...

Kafka入门-生产者

生产者 生产者发送流程&#xff1a; 延迟时间为0ms时&#xff0c;也就意味着每当有数据就会直接发送 异步发送API 异步发送和同步发送的不同在于&#xff1a;异步发送不需要等待结果&#xff0c;同步发送必须等待结果才能进行下一步发送。 普通异步发送 首先导入所需的k…...

比较数据迁移后MySQL数据库和OceanBase数据仓库中的表

设计一个MySQL数据库和OceanBase数据仓库的表数据比较的详细程序流程,两张表是相同的结构,都有整型主键id字段,需要每次从数据库分批取得2000条数据,用于比较,比较操作的同时可以再取2000条数据,等上一次比较完成之后,开始比较,直到比较完所有的数据。比较操作需要比较…...

从0开始学习R语言--Day17--Cox回归

Cox回归 在用医疗数据作分析时&#xff0c;最常见的是去预测某类病的患者的死亡率或预测他们的结局。但是我们得到的病人数据&#xff0c;往往会有很多的协变量&#xff0c;即使我们通过计算来减少指标对结果的影响&#xff0c;我们的数据中依然会有很多的协变量&#xff0c;且…...