Kotlin 学习笔记(1)基本语法

Kotlin 学习笔记(1)基本语法


关于 Kotlin 的安装,Android studio 3.0及以上自带,其他的可以通过搜索插件 Kotlin 来进行安装。下面来说说有关 Kotlin 的语法。

以下是本人的学习笔记,入有任何不妥之处,随时欢迎拍砖指正。
谢谢 ^_^

1. Kotlin Android 扩展

Kotlin 插件自带有安卓扩展,因此不需要再单独安装额外插件。
我们直接 new Kotlin Activity 后出现提示让配置 Kotlin

通过点击来自动导入 Kotlin 所需要的配置

实际上在项目的 Gradle上多了一行

项目 module 的 build.gradle 中多了

apply 多了

以上都是自动生成的

然后我们需要在项目 module 的 build.gradle 文件中添加以下代码

apply plugin: 'kotlin-android-extensions'

这样我们就可以用简写了,比如在 XML 中定义一个 TextView

<TextView
        android:id="@+id/hello"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"/>

我这里 ID 设置为『hello』那么我们在 Activity 中的代码就变成了

class KotlinActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_kotlin)
        hello.text = "你是不是傻"
    }
}

这样直接通过 hello.text 来设置内容,而不是原来长长的

 findView(R.id.textView) as TextView

该插件相当于把 XML 中的控件『id』作为 Activity 的额外属性来使用。

语句的结尾不用加;

简洁了很多~~

2. 定义函数

Kotlin 中所有的函数直接通过 fun 来声明。

fun sum(a: Int , b: Int) : Int{
    return a + b
}

fun sum(a: Int, b: Int) = a + b

上面两种方式都可以通过以下代码来调用。

println(sum(3, 5))
// 输出如下:
I/System.out: 8

3. 定义局部变量

var 表示变量,val 表示常量(相当于java中被final修饰)
val a: Int = 1  // 立即初始化
val b = 2   // 推导出Int型
val c: Int  // 当没有初始化值时必须声明类型
c = 3       // 赋值
println("a = $a, b = $b, c = $c")
var x = 5 // 推导出Int类型
x += 1
println(" x = $x")
// 输出如下:
I/System.out: x = 6

4. 注释

与 Java 和 JavaScript 一样,Kotlin 支持单行注释和块注释。

5. 使用字符串模板

var a = 1
// 使用变量名作为模板:
val s1 = "a is $a"
println(s1)

a = 2
// 使用表达式作为模板:
val s2 = "${s1.replace("is", "was")}, but now is $a"
println(s2)
println(s1)
// 输出如下:
I/System.out: a is 1
I/System.out: a was 1, but now is 2                   
I/System.out: a is 1

我发现在 val s2 中通过 replace 修改了 s1 的值,但是在最后的输出 s1 时,s1 的值并没有改变。
通过进入 replac 方法得知使用该方法后的字符串会重新生成新的字符串,并不改变原值~O(∩_∩)O哈哈~ 一看就是我 Java 基础没学好~
附图:

6. 使用条件表达式

fun maxOf(a: Int, b: Int): Int {
    if (a > b) {
        return a
    } else {
        return b
    }
}
fun maxOf(a: Int, b: Int) = if (a > b) a else b
println("我今年${maxOf(0, 18)}岁了.")
hello.text = "我今年${maxOf(0, 18)}岁了."
// 输出如下:
I/System.out: 我今年18岁了.

以上两种方式都可以表达。

7. 使用可空变量以及空值检查

开发中最常见的错误应该就是NullPointerException了,那么 Kotlin 是怎么解决这个问题的呢?

7.1 可空类型和非空类型

Kotlin 中用 ? 来表示该变量是否可以为空。下面看实例


当我们没有明确 String 可以为空的情况下,当 String 赋予null 时,会报错,导致无法运行。

前者我们可以直接调用

val l2 = a2.length
但是后者
val l3 = a3.length

我们不可以直接调用,因为 a3 可能为空,这里我们就要用到条件表达式了

7.2 在条件中检查 null
val l3 = if (a3 != null) a3.length else -1

// 输出如下:
I/System.out: 3

或者

if (a3 != null && a3.isNotEmpty())
    println("Stirng of length ${a3.length}")
else
    println("Empty string")   
// 输出如下:
I/System.out: Stirng of length 3
7.3 安全调用

使用安全操作符,?.

a3?.length

如果 a3 不为空则返回长度,否则返回空。
这个表达式的的类型是Int?

安全调用在链式调用是是很有用的。比如,如果 Bob 是一个雇员可能分配部门(也可能不分配),如果我们想获取 Bob 的部门名作为名字的前缀,就可以这样做:

bob?.department?.head?.name

这样的调用链在任何一个属性为空都会返回空。

7.4 Elvis 操作符

使用 Elvis 操作符,?:

val l = b.length?: -1

如果 ?: 左边表达式不为空则返回,否则返回右边的表达式。注意右边的表达式只有在左边表达式为空才会返回。

7.5 !! 操作符

用 b!! ,这会返回一个非空的 b 或者抛出一个 b 为空的 NPE

val l = b !!.length

7.6 安全转换

普通的转换可能产生 ClassCastException 异常。另一个选择就是使用安全转换,如果不成功就返回空:

var b = "aaa"
val aInt: Int? = b as? Int
println(aInt)
输出如下:
I/System.out: null

这样如果 b 是 int 的类型的就输出,不是就返回 null
而不会像以前一样程序 crash 后报 ClassCastException 异常

8. 使用值检查并自动转换

使用 is 操作符检查一个表达式是否是某个类型的实例。如果对不可变的局部变量或属性进行过了类型检查,就没有必要明确转换:

fun printLength(obj: Any) {
    println("'$obj' string length is ${getStringLength(obj) ?: "... err, not a string"} ")
}
fun getStringLength(obj: Any): Int? {
  if (obj is String) {
    // obj 将会在这个分支中自动转换为 String 类型
    return obj.length
  }

  // obj 在种类检查外仍然是 Any 类型
  return null
}

通过调用:

printLength("Incomprehensibilities")
printLength(1000)
printLength(listOf(Any()))
输出如下:
I/System.out: 'Incomprehensibilities' string length is 21 
I/System.out: '1000' string length is ... err, not a string 
I/System.out: '[java.lang.Object@72fd5e4]' string length is ... err, not a string 

当然我们也可以换一种方式来写

fun getStringLength(obj: Any): Int? {
  if (obj !is String) return null

  // obj 将会在这个分支中自动转换为 String 类型
  return obj.length
}

甚至可以这样

fun getStringLength(obj: Any): Int? {
    // obj 将会在&&右边自动转换为 String 类型
  if (obj is String && obj.length > 0) {
    return obj.length
  }

  return null
}

以上代码的输出结果是不变的。

9. 使用循环

val items = listOf("Google", "Apple", "Amazon")
for (index in items.indices) {
    println("item at $index is ${items[index]}")
}
// 输出如下: 
I/System.out: item at 0 is Google
I/System.out: item at 1 is Apple
I/System.out: item at 2 is Amazon

10. 使用 while 循环

val items = listOf("Google", "Apple", "Amazon")
var index = 0
while (index < items.size) {
    println("item at $index is ${items[index]}")
    index++
}

输出结果和 for 循环一样

11. 使用 when 表达式

Kotlin 中的 when 表达式就是类似于 Java 中的 switch 语句

when (obj) {
  1          -> "One"
  "Hello"    -> "Greeting"
  is Long    -> "Long"
  !is String -> "Not a string"
  else       -> "Unknown"
}
println(describe(1))
println(describe("Hello"))
println(describe(1000L))
println(describe(2))
println(describe("other"))

参数分别使用了『1』,『Hello』,『1000L』,『2』,『other』
分别对应下列的输出。和 switch 语法一模一样,只不过把一些过程给简化了。

// 输出如下:
I/System.out: item at 2 is kiwi
I/System.out: One
I/System.out: Greeting
I/System.out: Long
I/System.out: Not a string
I/System.out: Unknown

可能对于第四个有疑惑

println(describe(2))

!is String -> "Not a string"

以上意思是说 2 不是 String 类型的实例,结果为 true 所以输出了 『Not a string』

12. 使用ranges

使用 in 操作符检查数值是否在某个范围内:

val x = 10
val y = 9
if (x in 1..y+1) {
    println("fits in range")
}
// 输出如下:
I/System.out: fits in range

表达式的意思是说 x 是否在 1 ~ y + 1 的范围内
从定义的 x 和 y 的值来说,显而易见为 true

检查数值是否在范围外:

val list = listOf("a", "b", "c")

if (-1 !in 0..list.lastIndex) {
    println("list.lastIndex: ${list.lastIndex}")
    println("-1 is out of range")
}

if (list.size !in list.indices) {
    println("list.indices: ${list.indices}")
    println("list size is out of valid list indices range too")
}
// 输出如下:
I/System.out: list.lastIndex: 2
I/System.out: -1 is out of range
I/System.out: list.indices: 0..2
I/System.out: list size is out of valid list indices range too

!in 即不属于该范围 -1 不属于0~2,3也不属于0~2
所以他们的println语句都输出了。

在范围内迭代或者使用步进:

for (x in 1..5) {
  println(x)
}

println("----------------")

for (x in 1..10 step 2) {
  println(x)
}

println("----------------")

for (x in 9 downTo 0 step 3) {
  println(x)
}
// 输出如下:
I/System.out: 1
I/System.out: 2
I/System.out: 3
I/System.out: 4
I/System.out: 5
I/System.out: ----------------
I/System.out: 1
I/System.out: 3
I/System.out: 5
I/System.out: 7
I/System.out: 9
I/System.out: ----------------
I/System.out: 9
I/System.out: 6
I/System.out: 3
I/System.out: 0

step 跳跃
downTo 从大到小
看了 Log 日志,聪明的你应该懂了吧~~

13. 使用集合

对一个集合进行迭代:

val items = listOf("Google", "Apple", "Amazon")
for (item in items) {
    println(item)
}

println("----------------")

when {
       "tutu_oo" in items -> println("Change The World")
       "Google" in items -> println("My Dream")
}        

println("----------------")

val fruits = listOf("banana", "avocado", "apple", "kiwi")
   fruits
           .filter { it.startsWith("a") }
           .sortedBy { it }
           .map { it.toUpperCase() }
           .forEach { println(it) }
I/System.out: Google
I/System.out: Apple
I/System.out: Amazon
I/System.out: ----------------
I/System.out: My Dream
I/System.out: ----------------
I/System.out: APPLE
I/System.out: AVOCADO
I/System.out: ----------------

第一个是最普通的循环打印
接下来是 when 表达式,它只会执行一次
如果表达式是这样的

when {
       "Amazon" in items -> println("Change The World")
       "Google" in items -> println("My Dream")
} 
I/System.out: Google
I/System.out: Apple
I/System.out: Amazon
I/System.out: ----------------
I/System.out: Change The World
I/System.out: ----------------
I/System.out: APPLE
I/System.out: AVOCADO
I/System.out: ----------------

后面的 Google 即使也在 items 集合里面但是也不会输出『My Dream』

最后就是链式调用了,让我想到了 RxJava ,首先通过 filter 过滤只留下『a』 开头的,然后应该是自然排序,接着通过 map 转换成大写,最后将每一个都输出。

关于我

一只成长中的图图。

感谢

参考自:
https://www.kotlincn.net/docs/tutorials/android-plugin.html

https://huanglizhuo.gitbooks.io/kotlin-in-chinese/content/GettingStarted/Basic-Syntax.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值