Kotlin也没那么难(一)

do not speak,show my code...

基本概念

函数

kotlin:
fun functionName(parameter1: Int, parameter2: Int): Int {
    //该函数返回两者最大值
    return if (parameter1 > parameter2) parameter1 else parameter2
}
java:
Integer functionName(Integer parameter1, Integer parameter2) {
    //该函数返回两者最大值
    return parameter1 > parameter2 ? parameter1 : parameter2;
}

要点:

  1. kotlin中没有基本数据类型(int,float等),都是以对象形式存在(Int,Float),kotlin也没有数组,以Array类存在
  2. kotlin以 fun 关键字开头,而java以 返回类型 开头,kotlin的 返回类型:返回类型 的形式衔接在函数声明括号后
  3. kotlin的所有变量声明都是 变量名: 变量类型 ,java则是 变量类型 变量名
  4. kotlin也有三目运算符,不过写法是 if (表达式) 表达式正确时的值 else 表达式错误时的值,java则是 表达式 ? 表达式正确时的值 : 表达式错误时的值
  5. kotlin语句末尾可以省略分号
表达式函数体

kotlin中if是表达式,而不是语句。语句是没有自己的值,而表达式是有值的。表达式可以作为其他表达式的一部分也可以作为函数的返回值。在kotlin中除了循环(for、do、while)以外大多数控制结构都是表达式。
所以上面的函数就可以简化成以下代码:

fun functionStruct(parameter1: Int, parameter2: Int) = 
  if (parameter1 > parameter2) parameter1 else parameter2

细心的朋友可能发现了上面的函数没有声明返回类型,因为对于返回一个表达式的函数来说,编译器会自动分析返回类型,术语叫 类型推导


变量

kotlin不会以变量类型开头,因为有些的变量声明的类型是可以省略的

var number0 = 100
val number1 = 100
val number2 : Int = 100
val number3 //编译不通过
val number4 : Int //编译不通过

var 表示该变量是可变变量,可以重新赋值
val 表示该变量是不可变变量,不可以重新赋值


字符串模板 "...${变量}..."

eg:var age = "my age is ${number0}",这种情况下大括号可以省略
如果是java会写成 String age = "my age is " + number0;
eg:"...${list.get(0)}..."这个时候大括号不能省


类和属性

kotlin中默认public修饰符
kotlin定义一个类最简单的方法就是class People(var name: String, val age: Int)
以上代码定义了一个People类,该类有两个属性name和age
由于name被var修饰,所以kotlin会为name创建一个getter和一个setter方法
由于age被val修饰,所以kotlin会为name创建一个getter方法
使用起来很简单

val my = People("name", 20)
my.name = "new Name"
println("my name is ${my.name} and my age is ${my.age}")

要点:

  1. kotlin创建对象是 Xxx() ,java是** new Xxx();**
  2. kotlin调用属性的getter方法直接 xxx.property
  3. kotlin调用属性的setter方法直接 xxx.property = value

注意:
如果写成class People(name: String, age: Int),即省略了var和val,则这两个变量都只是构造函数用的临时变量,也就是说kotlin不会再自动给People类创建name和age变量

自定义getter和setter方法
class People(var name: String, val age: Int) {
    var desc: String = "这是我的个人描述"
        get() {
            print("调用了getter方法")
            return "my name is $name and my age is $age"
        }
        set(value) {
            print("调用了setter方法")
            field = value
        }
}

需要注意的是这里有个 field 变量,这个变量术语叫做 幕后变量 ,这个后面我们会深入了解,这里只需要知道把要更新的值赋予给该变量就OK啦


目录和包

kotlin的包声明形式和java一样:package com.example.ice.kotlindemo,不过有意思的是kotlin不强制包声明和文件路径一致,也就是说该kotlindemo.kt文件不必强制放在com/example/ice/目录下。
kotlin文件可以同时声明多个类,甚至还可以声明函数和变量!java中只能在类中声明函数,并且一个java文件只能有一个public class的而且该class类名还必须和文件名一致


枚举和When

enum
enum class Color {
    BLUE, RED, GREEN
}

kotlin中enum是软关键字,只有出现在class面前才有特殊意义,其他情况你甚至可以当一个变量使用 val enum: Color = Color.BLUE

when

when有点类似java中的switch,但是when是表达式

kotlin:
fun testWhen(enum: Color) =
        when (enum) {
            Color.BLUE -> "blue"
            Color.RED -> "red"
            Color.GREEN -> "green"
            else -> "default"
        }
java:
public String testWhen(Color color) {
    String returnValue;
    switch (color) {
        case RED:
            returnValue = "red";
            break;
        case BLUE:
            returnValue = "blue";
            break;
        case GREEN:
            returnValue = "green";
            break;
        default:
            returnValue = "default";
            break;
    }
    return returnValue;
}

要点:

  1. when不用java中的 break
  2. when用 -> 替代 java中的 case
  3. when用 else 替代java中的 default
  4. when是表达式不是语句块
在when分支上合并多个选项
when (enum) {
    Color.BLUE, Color.RED -> "blue or red"
    Color.GREEN -> "green"
    else -> "default"
}
when结构中可以使用任意对象
var people1: People = People("name1", 20)
var people2: People = People("name2", 20)
fun testWhen2(people: People) =
        when (people) {
            people1 -> "is people1"
            people2 -> {
                "is people2"
            }
            else -> "no match"
        }

要点:

  1. when表达式会把最后一行代码的结果当为返回值(见people2的情况,没有写return,但是会返回"is people2")
  2. when表达式某种情况下只有返回语句可以省略大括号(见people1的情况)
不带参数的when
fun testWhen3() =
        when {
            1 > 2 -> print("1 > 2")
            2 > 1 -> print("2 > 1")
            else -> throw ArithmeticException("error")
        }

类型转换

kotlin使用 is 代替java的 instanceof 进行类型判断
kotlin使用 as 进行显示转换 var num = 100 as Float
注意:as也可以用于导入语句 import xxx.xxx.People as P 这样P就代表了People


循环

kotlin的循环比较于java来说没有很大的改变,但是多了一些关键字需要我们注意

var i = 0
for (i in 0..100) { //打印 01234..100
//for (i in 0 until 100) { //打印 01234..99
//for (i in 100 downTo 0 step 2) { //打印 1009896..0
    print(i)
}
in 常常用来迭代list或map
val list = listOf(1, 2, 3)
for (i in list) {
    print(i)
}

kotlin中使用 listOf 方法可以创建一个list

如果你想迭代list的同时获取当前的index,可以使用如下写法
for ((index, value) in list.withIndex()) {
//(index,value)的写法专业术语叫做解构声明,后面我们会深入了解
    print("index = $index and value = $value")
}
in 也可以用来判断对象是否在集合中
if (2 in list) {
    print("2 exist in list")
}
!in 关键字

Emm,这个看前面一个 ! 就知道是和 in 相反的结果了


try catch finally

  1. kotlin没有 throws 关键字(想起java被throws支配的恐惧了么)
  2. try也是一个表达式
val myAge = try {
    1 / 0
} catch (e: Exception) {
    print(e)
} finally {
    100
}



函数

集合创建
listOf(1, 2, 3)
mutableListOf(1, 2, 3)

setOf(1, 2, 3)
mutableSetOf(1, 2, 3)

mapOf(1 to "1", 2 to "2", 3 to "3")
mutableMapOf(1 to "1", 2 to "2", 3 to "3")

有mutable前缀的方法创建的集合,元素是可以增添、移除、修改的

默认参数
fun defaultParaFun(para1: Int = 0, para2: Int, para3: Int) = para1 + para2 + para3
defaultParaFun(para2 = 2, para3 = 4)

我们给参数para1设置了默认值0
在调用该函数的时候我们对传入的值进行了显示的声明参数,比如2就是参数para2的值,3就是参数para3的值,此时函数返回值为5(0+2+3)
有意思的是如果我们给所有参数都加默认值:

fun defaultParaFun(para1: Int = 0, para2: Int = 1, para3: Int = 2) = para1 + para2 + para3
defaultParaFun() // 0+1+2
defaultParaFun(para3 = 5) // 0+1+5
defaultParaFun(para2 = 5) // 0+5+2
defaultParaFun(para1 = 5, para2 = 5) // 5+5+2

同时java是不支持默认参数的,所以我们可以给方法加上@JvmOverloads注解。这样的话kotlin编译器会将该方法生成一系列重载方法


result.png

由于我这个例子中 defaultParaFun 是直接写在 KotlinDemo.kt 文件中的(术语叫顶级函数),所以java调用要使用 KotlinDemoKt.defaultParaFun 形式,顶级函数是不是很像java中的静态函数(kotlin是没有static关键字的!)?

扩展函数
fun Int.add2(): Int {
    return this + 2
}
print(0.add2())

要点:

  1. 我们定义了一个add2函数用于将一个 数+2 并返回
  2. add2函数面前使用 Int. 表示该函数是对Int类的扩展(就像是Int类本身的方法一样)
  3. add2函数体内的this其实就是调用该方法的对象(本例中this就是0)
  4. this可以省略,就变成了 return +2
  5. java中调用就得写成 KotlinDemoKt.add2(0) 其实就是被编译成了静态函数
  6. 如果Int类也有add2则优先调用Int类的add2函数,即成员方法优先调用
  7. 如果子类和基类有同名的扩展函数,调用哪个扩展函数由调用对象的声明类型决定
  8. 除了有扩展函数也有扩展属性,不过扩展属性必须要有getter函数,同时也不能初始化
可变参数
fun varargFun(vararg list: Int) {
    print(list[0])
}
fun varargFun2(array: Array<Int>) {
    listOf(*array)
}
varargFun(1, 2, 3)

vararg 关键字表示该参数是可变参数
* 关键字是展开运算符,在该例中把array展开为多个对象并用于构造list

中缀调用

细心的朋友可能看到了之前我们定义map时用的是mapOf(1 to "1", 2 to "2", 3 to "3")
这里的to其实就是中缀调用,实际上就是一个to函数,中缀调用就是把方法名称放在两个参数中间,记得空格
我们可以看源码 infix fun <A, B> A.to(that: B): Pair<A, B> = Pair(this, that)
其中尖括号里的是泛型,Pair是一个有两个成员变量的类(一个叫first一个叫second)
我们先把infix关键字盖住,其实这就是一个扩展函数对不对!
我们在加上infix关键字,这个扩展函数就可以中缀调用了
需要注意的是,infix只能用于只有一个参数的函数上

解构声明

val (para1, para2) = 1 to "1"
在这个例子中,para1就是1,para2就是"1"
其实我们在之前循环的章节就有用到过解构声明

for ((index, value) in list.withIndex()) {
    print("index = $index and value = $value")
}

解构声明不仅可以用在Pair和List也可以用在Map,我们会在后面的文章中深入了解解构规则

字符串
var string1 = "\\"
var string2 = """\"""

其实string1和string2都是表示
只不过普通的字符串包含转义,\ 就是个转义字符,所以 \\ 才能代表
而在多重引号字符串中就不包含转义,所以可以直接用 \ 来表示
注:多重引号字符串用来写正则特别方便

局部函数

这是我最喜欢的语法糖之一:函数里可以再定义函数

fun outer(){
    print("outer invoke")
    fun inner(){
        print("inner invoke")
    }
    inner()
}

局部函数中可以访问外部函数中的所有参数和变量(记得定义在局部函数之前哦)!

结语

通过本篇文章的学习我们已经算 入门kotlin 了,下篇文章我会写 类和接口,lambda编程以及可空性 相关。基本上这两篇文章内容熟练掌握就可以应对开发需求,之后会有篇幅讲Kotlin的高阶使用:反射、泛型等。
最后祝大家国庆Happy~

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

推荐阅读更多精彩内容