Kotlin-第3节、函数与Lambda闭包

目录:
1、函数的特性语法
2、嵌套函数
3、扩展函数
4、Lambda闭包语法
5、高阶函数
6、内联函数

1、函数的特性语法

  • 函数的参数可以声明默认值
  • 如果函数体只有一个语句的话,可以将该语句直接赋值给该函数。
fun echo(name:String){
    println("$name")
}
fun echo(name:String = "Zhangtao"){
    println("$name")
}
fun echo(name:String) = println("$name")

2、函数嵌套

用途:在某些情况下出发递归的函数,或不希望被外部函数访问到的函数。

fun function(){
    val str = "hello world"
    fun say(count:Int = 10){
        println(str)
        if(count>0){
            say(count - 1)
        }
    }
    say()
}

3、扩展函数

kotlin特性:静态给一个类扩展成员方法和成员变量。
用类名点方法名的方式

//给File类扩展一个readText方法
fun File.readText(charset:Charset = Charsets.UTF-8):String=readBytes().toString(charset)

//调用
fun main(args:Array<String>){
    val file = File("Project2.iml")
    println(file.readText())
}

java如何使用kotlin的扩展函数呢?

public class Main{
    public static void main(String[] args){
        File file = new File("project2.iml");
        String content = FilesKt.readText(file,Charsets.UTF-8);
        System.out.println(content);
    }
}

注:该扩展函数并不是File本身的函数,它编译的时候会被编译到那个类对应的class里面去,在java中我们调用的是FileKt,第一个参数是需要扩展这个类的对象,该参数不可省略。

  • 为什么说这是静态的扩展呢?
//open值得是不final,与java的final相反的意思
open class Animal
class Dog:Animal()

//分别给其添加扩展方法
fun Animal.name() = "animal"
fun Dog.name() = "dog"

fun Animal.printName(anim:Animal){
    println(anim.name())
}

fun main(args:Array<String>){
    Dog().printName(Dog())//这里输出animal
}
  • kotlin的扩展函数它实际上是静态的给一个类添加的,它是不具备运行时的一个多态效应的。

编译完之后:

public static final String name(Animal receiver){
    return "animal";
}
public static final String name(Dog receiver){
    return "dog";
}
public static final void printName(Animal r,Animal a){
    String str = name(a)
    System.out.println(str);
}
public static final void main(String [] args){
    //强转成了animal
    printName((Animal)new Dog(), (Animal)new Dog());
}

4、kotlin的lambda闭包

public static void main(String[] args){
    Thread thread = new Thread(new Runnable(){
        public void run(){
            // ...
        }
    });
    thread.start();
}

Java8的Lambda语法:

  • 将Runnable对象省略成()-> 的形式替代。
public static void main(String[] args){
    Thread thread = new Thread(() -> {
        //...
    });
    thread.start();
}

kotlin的Lambda语法:

fun main(args:Array<String>){
    val thread = Thread({  -> Unit })
    thread.start()
}
  • 如果Lambda没有参数,可以省略箭头符号->
fun main(args:Array<String>){
    val thread = Thread({})
    thread.start()
}
  • 如果Lambda是函数的最后一个参数,可以将大括号放在小括号外面
fun main(args:Array<String>){
    val thread = Thread(){}
    thread.start()
}
  • 最简单形态:如果函数只有一个参数且这个参数是Lambda,则可以省略小括号
fun main(args:Array<String>){
    val thread = Thread{ }
    thread.start()
}
  • 如果Lambda只有一个参数的话,可以不写,然后用it去替代它。
val letResult = user.let(user:User->"let::${user.javaClass}")
|
|
val letResult = user.let{"let::${it.javaClass}"}

Lambda闭包声明

//name:String:参数
//->:闭包体
val echo = { name:String ->
    println(name)
}
fun main(args: Array<String>){
    echo.invoke("hello")
    echo("hello ")
}
  • Lambda参数有上限22个。
  • 如果传23个,会抛出NoClassDefFoundError,因为编译成class的时候,会编译成Funcation23,但是内置最多Funcation22,所以才会找不到。
fun onlyif(isDebug:Boolean,block:()->Unit){
    if (isDebug)block()
}
fun main(args:Array<String>):Unit{
    onlyif(true){
        println("打印日志")
    }
}
  • 注意:main函数中onlyif(true)用大括号括起来的部分是Lambda函数(onlyif第二参),当Lambda作为函数的最后一个参数时,可以写在函数的小括号之外。

5、高阶函数

重点:函数是“一等公民”

  • 对象.函数,这种是函数的调用。
  • 如果要把函数当做参数传递,需要传递一个函数的声明。
  • 如果传递的是“执行函数”的话,传递的实际上是函数的返回值。
val runnable = Runnable{
    println("Runnable::run")
}
//声明一个参数为空,无返回值的函数对象
val funcation:() -> Unit
//将runnable的run方法赋值给声明好的函数
function = runnable::run
//将该函数作为参数传递给onlyif
onlyif(true,function)

6、用内联优化代码

  • Kotlin的Lambda是一个匿名对象
  • 可以使用inline修饰方法,这样当方法在编译时就会拆解方法的调用为语句调用,进而减少创建不必要的对象。

创建方式:在高阶函数之前增加inline关键字。

inline fun onlyif(isDebug:Boolean,block:()->Unit){
    if (isDebug)block()
}
fun main(args:Array<String>):Unit{
    onlyif(true){
        println("打印日志")
    }
}

编译后的class:

public static final void onlyif(boolean isDebug,Function0<Unit>block){
    if(isDebug){
        block.invoke();
    }
}
public static final void main(String[] args){
    //拆解成了语句调用
    boolean isDebug = true;
    if(isDebug){
        String str = "打印日志";
        System.out.println(str);
    }
}

过分使用inline关键字会造成:

  • 增加编译器编译的负担。
  • 使得代码块变得庞大。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容