Kotlin语言基础入门:Kotlin简介

在2019年Google I/O大会上,Google 宣布今后将优先采用 Kotlin 进行 Android 开发。

一,简介

Kotlin 是一种富有表现力且简洁的编程语言,不仅可以减少常见代码错误,还可以轻松集成到现有应用中。

Google 列举的 Kotlin 的优势:

  • 富有表现力且简洁:可以使用更少的代码实现更多的功能。表达自己的想法,少编写样板代码。
  • 更安全的代码:Kotlin 有许多语言功能,可帮助你避免null指针异常等常见编程错误。
  • 可互操作:可以在 Kotlin 代码中调用 Java 代码,或者在 Java 代码中调用 Kotlin 代码。Kotlin 可完全与 Java 编程语言互操作。
  • 结构化并发:Kotlin 协程让异步代码像阻塞代码一样易于使用。协程可大幅简化后台任务管理。

更重要的是,Jetpack Compose 仅支持 Kotlin,而不再支持 Java。 Google 提到多平台项目可使用 Kotlin 来开发。

二,概念介绍

1. 包的定义和使用

包的定义应该写在文件的顶部。

arduino 复制代码
package com.rustfisher.ktpractice.intro
​
import kotlin.text.*
​
// ...

Kotlin并不要求包名和文件存放位置严格对应。

2. 程序入口

Kotlin应用的程序入口是main方法。

kotlin 复制代码
fun main() {
    // ..
}

另一个main方法带有参数

kotlin 复制代码
fun main(args: Array<String>) {
    // ...
}

3. 标准输出

print方法将传入的变量打印到标准输入流。

bash 复制代码
print("Rust ")
print("Fisher")

println打印传入变量并且在最后添加一个换行。

scss 复制代码
println("an.rustfisher.com")
println(42)

4. 方法

下面是一个接受2个Int参数,返回Int的方法。

kotlin 复制代码
fun sum(a: Int, b: Int): Int {
    return a + b
}

方法主体可以是一个表达式。它的返回值可以被推断出来。

kotlin 复制代码
fun sum(a: Int, b: Int) = a + b

方法可以没有返回值,或者说是返回一个无意义的值(Unit)。

kotlin 复制代码
fun printSum(a: Int, b: Int): Unit {
    println("sum of $a and $b is ${a + b}")
}

Unit可以忽略不写。

kotlin 复制代码
fun printSum(a: Int, b: Int) {
    println("sum of $a and $b is ${a + b}")
}

5. 变量

只能读的变量需要用关键字val。它们只能被赋值1次。

kotlin 复制代码
val a: Int = 1 // 直接赋值
val b = 2      // 自动推测出是Int类型
val c: Int     // 当没有赋初始值时,需要声明类型Type required when no initializer is provided
c = 3          // 这里是推迟赋值

可以多次赋值的变量用关键字var

csharp 复制代码
var x = 5 // 自动推测出是Int型
x += 1

可以在顶部声明变量

kotlin 复制代码
val PI = 3.14
var x = 0

fun incrementX() { 
    x += 1
}

6. 创建类与实例

定义一个类,使用class关键字

kotlin 复制代码
class Fisher

类的属性可以放在定义中或者类里。比如下面这个类Rectangle,描述长方形。

kotlin 复制代码
class Rectangle(var height: Double, var length: Double) {
    var perimeter = (height + length) * 2
}

默认构造器中的变量可以直接使用。这里可以直接使用面积变量perimeter

scss 复制代码
val rectangle = Rectangle(5.0, 2.0)
println("The perimeter is ${rectangle.perimeter}")

对于kotlin v.1.4.30,类的继承用冒号:来表示。类默认都是final的,不可继承。为了继承,用open让这个类能被继承。

kotlin 复制代码
open class Shape
class Rectangle(var height: Double, var length: Double): Shape {
    var perimeter = (height + length) * 2
}

7. 注释

和其他现代编程语言类似,用//或者/**/来注释

arduino 复制代码
// 这里是注释 RustFisher
/**
 * 这是注释
 */
/* 这也是注释 */

8. 字符串模版(String tempplates)

直接使用变量,用$

要使用表达式,需要用大括号${}

ini 复制代码
var a = 1
// 简单使用$
val s1 = "a is $a" 

a = 2
// 使用方法
val s2 = "${s1.replace("is", "was")}, but now is $a"

9. 条件表达式

kotlin 复制代码
fun getMax(a: Int, b: Int): Int {
    if (a > b) {
        return a
    } else {
        return b
    }
}

Kotlin中,if也可以写成一个表达式

kotlin 复制代码
fun getMax(a: Int, b: Int) = if (a > b) a else b

10. for循环

使用in

ini 复制代码
val items = listOf("apple", "banana", "kiwifruit")
for (item in items) {
    println(item)
}

使用下标

perl 复制代码
val items = listOf("apple", "banana", "kiwifruit")
for (index in items.indices) {
    println("item at $index is ${items[index]}")
}
scss 复制代码
for (i in 1..3) {
    println(i)
}
for (i in 6 downTo 0 step 2) {
    println(i)
}

11. while循环

perl 复制代码
val items = listOf("apple", "banana", "kiwifruit")
var index = 0
while (index < items.size) {
    println("item at $index is ${items[index]}")
    index++
}

12. when表达式

when看起来有点像Java里的witch,但这两个是不同的东西。

kotlin 复制代码
fun describe(obj: Any): String =
    when (obj) {
        1          -> "One" 
        "Hello"    -> "Greeting"
        is Long    -> "Long"
        !is String -> "Not a string"
        else       -> "Unknown"
    }

13. 范围

检查一个数字是否在范围中,使用in操作

ini 复制代码
val x = 10
val y = 9
if (x in 1..y+1) {
    println("在范围内")
}

检查一个数是否超出了范围

erlang 复制代码
val list = listOf("a", "b", "c")

if (-1 !in 0..list.lastIndex) {
    println("-1 超出了范围")
}
if (list.size !in list.indices) {
    println("list的size也超出了下标范围")
}

遍历一个范围

scss 复制代码
for (x in 1..5) {
    print(x)
}

指定步进值

scss 复制代码
for (x in 1..10 step 2) {
    print(x)
}
println()
for (x in 9 downTo 0 step 3) {
    print(x)
}

14. 集合

遍历一个集合

scss 复制代码
for (item in items) {
    println(item)
}

检查集合中是否包含某个对象,用in操作

go 复制代码
when {
    "orange" in items -> println("juicy")
    "apple" in items -> println("apple is fine too")
}

用lambda表达式对集合进行filter和map操作

go 复制代码
val fruits = listOf("banana", "avocado", "apple", "kiwifruit")
fruits
    .filter { it.startsWith("a") }
    .sortedBy { it }
    .map { it.toUpperCase() }
    .forEach { println(it) }

15. 可为null的值和null检查

可以为null的变量,后面需要一个问号?

下面这个方法返回Int或者null

kotlin 复制代码
fun parseInt(str: String): Int? {
    // ...
}

使用可能返回null的方法(上面那个方法)

kotlin 复制代码
fun printProduct(arg1: String, arg2: String) {
    val x = parseInt(arg1)
    val y = parseInt(arg2)

    // Using `x * y` yields error because they may hold nulls.
    if (x != null && y != null) {
        // x 和 y 经过null检测后自动变成了非null值
        println(x * y)
    }
    else {
        println("'$arg1' or '$arg2' 不是数字")
    }    
}

或者写做

kotlin 复制代码
// ...
if (x == null) {
    println("参数错误 arg1: '$arg1'")
    return
}
if (y == null) {
    println("参数错误 arg2: '$arg2'")
    return
}

// x 和 y 经过null检测后自动变成了非null值
println(x * y)

16. 类型检查和自动转换

is来检查某个对象是不是某个类型。 如果确定某个不可变的变量的类型,那后面使用它的时候不用再显式转换

kotlin 复制代码
fun getStringLength(obj: Any): Int? {
    if (obj is String) {
        // obj已经自动转换为String类型
        return obj.length
    }

    // 这里的obj仍然是Any类型
    return null
}

或者用!is

kotlin 复制代码
fun getStringLength(obj: Any): Int? {
    if (obj !is String) return null

    // 这里的obj已经自动转换为String类型
    return obj.length
}

再换个写法

kotlin 复制代码
fun getStringLength(obj: Any): Int? {
    // 这里面的obj已经自动转换为String类型
    if (obj is String && obj.length > 0) {
        return obj.length
    }

    return null
}

更多Android进阶指南 可以详细Vx关注公众号:Android老皮 解锁 《Android十大板块文档》

1.Android车载应用开发系统学习指南(附项目实战)

2.Android Framework学习指南,助力成为系统级开发高手

3.2023最新Android中高级面试题汇总+解析,告别零offer

4.企业级Android音视频开发学习路线+项目实战(附源码)

5.Android Jetpack从入门到精通,构建高质量UI界面

6.Flutter技术解析与实战,跨平台首要之选

7.Kotlin从入门到实战,全方面提升架构基础

8.高级Android插件化与组件化(含实战教程和源码)

9.Android 性能优化实战+360°全方面性能调优

10.Android零基础入门到精通,高手进阶之路

敲代码不易,关注一下吧。ღ( ´・ᴗ・` ) 🤔

相关推荐
百万蹄蹄向前冲2 小时前
2024不一样的VUE3期末考查
前端·javascript·程序员
陈哥聊测试1 天前
软件格局在变,谁能扛起国产替代的大旗?
安全·程序员·产品
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭2 天前
SpringBoot如何实现缓存预热?
java·spring boot·spring·缓存·程序员
少年姜太公2 天前
从零开始详解js中的this(下)
前端·javascript·程序员
凌虚2 天前
Kubernetes APF(API 优先级和公平调度)简介
后端·程序员·kubernetes
小华同学ai2 天前
ShowDoc:Star12.3k,福利项目,个人小团队的在线文档“简单、易用、轻量化”还专门针对API文档、技术文档做了优化
前端·程序员·github
小青鱼4 天前
AI编程-Cursor从入门到精通系列之常用概念及解释(二)
人工智能·程序员
捡田螺的小男孩5 天前
参数校验的十个建议!收藏好,别再给测试机会提bug~
java·后端·程序员
哔哩哔哩技术5 天前
B站装机系统实践:从初创到规模化的演进
前端·程序员
程序员鱼皮5 天前
没事别想不开去创业!
计算机·面试·程序员·项目