Kotlin语法(其他)

简介:

过给对象插入多个成员函数做区别是很方便的:

val (name, age) = person

多重声明一次创建了多个变量。我们声明了俩个新变量:name age 并且可以独立使用:

println(name)
println(age)

也可以在 for 循环中用:

for ((a, b) in collection) { ... }

map:

for ((key, value) in map) { ... }

2. Ranges

函数操作符是:

if (i in 1..10){ ... }
if (x !in 1.0..3.0) println(x)
if (str in "island".."isle") println(str)
for (i in 4 downTo 1) print(i) // prints '4321'
for (i in 1..4 step 2) print(i) // prints "13"
for (i in 4 downTo 1 step 2) print(i) // prints "42"
for (i in 1.0..2.0 step 0.3) print("$i ") // prints "1.0 1.3 1.6 1.9 "
for (i in (1..4).reversed()) print(i) // prints "4321"
for (i in (1..4).reversed() step 2) print(i) // prints "42"

3. 类型检查和转换

is !is 表达式: 类似Java的instanceof

“不安全”的转换符,如下的x不能为空,否则抛异常:

val x: String = y as String

"安全"转换符,如下不管 as? 右边的是不是一个非空 String 结果都会转换为可空的:

val x: String ?= y as? String

4. This 表达式

如果 this 没有应用者,则指向的是最内层的闭合范围。为了在其它范围中返回 this ,需要使用标签:this@lable

5. 等式

在 kotlin 中有两种相等:

  • 参照相等: ===,参照相等是通过 === 操作符判断的(不等是!== ) a===b 只有 a b 指向同一个对象是判别才成立。另外,你可以使用内联函数identityEquals() 判断参照相等。

  • 结构相等: ==,结构相等是通过 == 判断的。像a == b将会翻译成:a?.equals(b) ?: b === null,如果 a 不是 null 则调用 equals(Any?) 函数,否则检查 b 是否参照等于 null。

6. 运算符重载

http://kotlinlang.org/docs/reference/operator-overloading.html

7. 空安全

var a: String ="abc"
a = null // 编译错误
val l = a.length() // 因为a不能为null,所以不会报错
var b: String? = "abc"
b = null
val l = b.length() // 错误:b为空
val l = if (b != null) b.length() else -1 // 不会报错
b?.length() // 安全调用不会报错

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

bob?.department?.head?.name // 

Elvis 操作符:

val l: Int = if (b != null) b.length() else -1

也可以使用 Elvis 操作符?::

val l = b.length()?: -1

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

!!操作符:
第三个选择是 NPE-lovers。我们可以用 b!! ,这会返回一个非空的 b 或者抛出一个 b 为空的 NPE:

val l = b !!.length()

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

val aInt: Int? = a as? Int

8.异常

所有的异常类都是 Exception 的子类。使用方式与Java类似。

try {
  // some code
}
catch (e: SomeException) {
  // handler
}
finally {
  // optional finally block
}

9.注解

注解是一种将元数据附加到代码中的方法。声明注解需要在类前面使用 annotation 关键字:

annotation class fancy

用法:在多数情形中 @ 标识是可选的。只有在注解表达式或本地声明中才必须

@fancy class Foo {
    @fancy fun baz(@fancy foo: Int): Int {
        return (@fancy 1)
    }
}
// 可省略@
fancy class Foo {
    fancy fun baz(fancy foo: Int): Int {
        @fancy fun bar() { ... }
        return (@fancy 1)
    }
}

如果要给构造函数注解,就需要在构造函数声明时添加 constructor 关键字,并且需要在前面添加注解:

class Foo @inject constructor (dependency: MyDependency)

注解可以有带参数的构造函数:

annotation class special(val why: String)
special("example") class Foo {}

注解也可以用在 Lambda 中。这将会应用到 lambda 生成的 invoke() 方法:

annotation class Suspendable
val f = @Suspendable { Fiber.sleep(10) }

java 注解在 kotlin 中是完全兼容的。
如果java 中的 value 参数有数组类型,则在 kotlin 中变成 vararg 参数:

// Java
public @interface AnnWithArrayValue {
    String[] value();
}
// Kotlin
AnnWithArrayValue("abc", "foo", "bar") class C

注解实例的值在 kotlin 代码中是暴露属性。

10.反射

得到运行时的类引用:

val c = MyClass::class

引用是一种 KClass类型的值。你可以使用KClass.properties 和KClass.extensionProperties获取类和父类的所有属性引用的列表。

函数引用
使用::操作符:

fun isOdd(x: Int) =x % 2 !=0
val numbers = listOf(1, 2, 3)
println(numbers.filter( ::isOdd) ) //prints [1, 3]

这里::isOdd是是一个函数类型的值(Int) -> Boolean 

下面是返回一个由两个传递进去的函数的组合。现在你可以把它用在可调用的引用上了:

fun compose<A, B, C>(f: (B) -> C, g: (A) -> B): (A) -> C {
    return {x -> f(g(x))}
}

fun length(s: String) = s.size
val oddLength = compose(::isOdd, ::length)
val strings = listOf("a", "ab", "abc")
println(strings.filter(oddLength)) // Prints "[a, abc]"

属性引用
访问顶级类的属性,我们也可以使用::操作符:

var x = 1
fun main(args: Array<String>) {
    println(::x.get())
    ::x.set(2)
    println(x)
}

::x表达式评估为KProperty<Int>类型的属性,它允许我们使用get()读它的值或者使用名字取回它的属性。

class A(val p: Int)

fun main(args: Array<String>) {
    val prop = A::p
    println(prop.get(A(1))) // prints "1"
}

对于扩展属性:

val String.lastChar: Char
  get() = this[size - 1]

fun main(args: Array<String>) {
  println(String::lastChar.get("abc")) // prints "c"
}

与 java 反射调用
想找到一个备用字段或者 java getter 方法:

import kotlin.reflect.jvm.*

class A(val p: Int)

fun main(args: Array<String>) {
    println(A::p.javaGetter) // prints "public final int A.getP()"
    println(A::p.javaField)  // prints "private final int A.p"
}

构造函数引用
只需要使用::操作符并加上类名。下面的函数是一个没有参数并且返回类型是Foo:

calss Foo
fun function(factory : () -> Foo) {
    val x: Foo = factory()
}
function(:: Foo)

11. 动态类型

为了方便使用,dynamic应而生:

val dyn: dynamic = ...

dynamic类型关闭了 kotlin 的类型检查:

这样的类型可以分配任意变量或者在任意的地方作为参数传递 任何值都可以分配为dynamic 类型,或者作为参数传递给任何接受 dynamic 类型参数的函数 这样的类型不做 null 检查

dynamic最奇特的特性就是可以在dynamic变量上调用任何属性或任何方法。

dyn.whatever(1, "foo", dyn) // 'whatever' is not defined anywhere
dyn.whatever(*array(1, 2, 3))

动态调用可以返回dynamic作为结果,因此我们可以轻松实现链式调用:

dyn.foo().bar.bat()

当给动态调用传递一个 lambda 表达式时,所有的参数默认都是dynamic

dyn.foo {
  x -> x.bar() // x is dynamic
}

参考:

  1. http://kotlinlang.org/docs/reference/basic-syntax.html
  2. http://huanglizhuo.gitbooks.io/kotlin-in-chinesE

过给对象插入多个成员函数做区别是很方便的:

val (name, age) = person

多重声明一次创建了多个变量。我们声明了俩个新变量:name age 并且可以独立使用:

println(name)
println(age)

也可以在 for 循环中用:

for ((a, b) in collection) { ... }

map:

for ((key, value) in map) { ... }

2. Ranges

函数操作符是:

if (i in 1..10){ ... }
if (x !in 1.0..3.0) println(x)
if (str in "island".."isle") println(str)
for (i in 4 downTo 1) print(i) // prints '4321'
for (i in 1..4 step 2) print(i) // prints "13"
for (i in 4 downTo 1 step 2) print(i) // prints "42"
for (i in 1.0..2.0 step 0.3) print("$i ") // prints "1.0 1.3 1.6 1.9 "
for (i in (1..4).reversed()) print(i) // prints "4321"
for (i in (1..4).reversed() step 2) print(i) // prints "42"

3. 类型检查和转换

is !is 表达式: 类似Java的instanceof

“不安全”的转换符,如下的x不能为空,否则抛异常:

val x: String = y as String

"安全"转换符,如下不管 as? 右边的是不是一个非空 String 结果都会转换为可空的:

val x: String ?= y as? String

4. This 表达式

如果 this 没有应用者,则指向的是最内层的闭合范围。为了在其它范围中返回 this ,需要使用标签:this@lable

5. 等式

在 kotlin 中有两种相等:

  • 参照相等: ===,参照相等是通过 === 操作符判断的(不等是!== ) a===b 只有 a b 指向同一个对象是判别才成立。另外,你可以使用内联函数identityEquals() 判断参照相等。

  • 结构相等: ==,结构相等是通过 == 判断的。像a == b将会翻译成:a?.equals(b) ?: b === null,如果 a 不是 null 则调用 equals(Any?) 函数,否则检查 b 是否参照等于 null。

6. 运算符重载

http://kotlinlang.org/docs/reference/operator-overloading.html

7. 空安全

var a: String ="abc"
a = null // 编译错误
val l = a.length() // 因为a不能为null,所以不会报错
var b: String? = "abc"
b = null
val l = b.length() // 错误:b为空
val l = if (b != null) b.length() else -1 // 不会报错
b?.length() // 安全调用不会报错

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

bob?.department?.head?.name // 

Elvis 操作符:

val l: Int = if (b != null) b.length() else -1

也可以使用 Elvis 操作符?::

val l = b.length()?: -1

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

!!操作符:
第三个选择是 NPE-lovers。我们可以用 b!! ,这会返回一个非空的 b 或者抛出一个 b 为空的 NPE:

val l = b !!.length()

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

val aInt: Int? = a as? Int

8.异常

所有的异常类都是 Exception 的子类。使用方式与Java类似。

try {
  // some code
}
catch (e: SomeException) {
  // handler
}
finally {
  // optional finally block
}

9.注解

注解是一种将元数据附加到代码中的方法。声明注解需要在类前面使用 annotation 关键字:

annotation class fancy

用法:在多数情形中 @ 标识是可选的。只有在注解表达式或本地声明中才必须

@fancy class Foo {
    @fancy fun baz(@fancy foo: Int): Int {
        return (@fancy 1)
    }
}
// 可省略@
fancy class Foo {
    fancy fun baz(fancy foo: Int): Int {
        @fancy fun bar() { ... }
        return (@fancy 1)
    }
}

如果要给构造函数注解,就需要在构造函数声明时添加 constructor 关键字,并且需要在前面添加注解:

class Foo @inject constructor (dependency: MyDependency)

注解可以有带参数的构造函数:

annotation class special(val why: String)
special("example") class Foo {}

注解也可以用在 Lambda 中。这将会应用到 lambda 生成的 invoke() 方法:

annotation class Suspendable
val f = @Suspendable { Fiber.sleep(10) }

java 注解在 kotlin 中是完全兼容的。
如果java 中的 value 参数有数组类型,则在 kotlin 中变成 vararg 参数:

// Java
public @interface AnnWithArrayValue {
    String[] value();
}
// Kotlin
AnnWithArrayValue("abc", "foo", "bar") class C

注解实例的值在 kotlin 代码中是暴露属性。

10.反射

得到运行时的类引用:

val c = MyClass::class

引用是一种 KClass类型的值。你可以使用KClass.properties 和KClass.extensionProperties获取类和父类的所有属性引用的列表。

函数引用
使用::操作符:

fun isOdd(x: Int) =x % 2 !=0
val numbers = listOf(1, 2, 3)
println(numbers.filter( ::isOdd) ) //prints [1, 3]

这里::isOdd是是一个函数类型的值(Int) -> Boolean 

下面是返回一个由两个传递进去的函数的组合。现在你可以把它用在可调用的引用上了:

fun compose<A, B, C>(f: (B) -> C, g: (A) -> B): (A) -> C {
    return {x -> f(g(x))}
}

fun length(s: String) = s.size
val oddLength = compose(::isOdd, ::length)
val strings = listOf("a", "ab", "abc")
println(strings.filter(oddLength)) // Prints "[a, abc]"

属性引用
访问顶级类的属性,我们也可以使用::操作符:

var x = 1
fun main(args: Array<String>) {
    println(::x.get())
    ::x.set(2)
    println(x)
}

::x表达式评估为KProperty<Int>类型的属性,它允许我们使用get()读它的值或者使用名字取回它的属性。

class A(val p: Int)

fun main(args: Array<String>) {
    val prop = A::p
    println(prop.get(A(1))) // prints "1"
}

对于扩展属性:

val String.lastChar: Char
  get() = this[size - 1]

fun main(args: Array<String>) {
  println(String::lastChar.get("abc")) // prints "c"
}

与 java 反射调用
想找到一个备用字段或者 java getter 方法:

import kotlin.reflect.jvm.*

class A(val p: Int)

fun main(args: Array<String>) {
    println(A::p.javaGetter) // prints "public final int A.getP()"
    println(A::p.javaField)  // prints "private final int A.p"
}

构造函数引用
只需要使用::操作符并加上类名。下面的函数是一个没有参数并且返回类型是Foo:

calss Foo
fun function(factory : () -> Foo) {
    val x: Foo = factory()
}
function(:: Foo)

11. 动态类型

为了方便使用,dynamic应而生:

val dyn: dynamic = ...

dynamic类型关闭了 kotlin 的类型检查:

这样的类型可以分配任意变量或者在任意的地方作为参数传递 任何值都可以分配为dynamic 类型,或者作为参数传递给任何接受 dynamic 类型参数的函数 这样的类型不做 null 检查

dynamic最奇特的特性就是可以在dynamic变量上调用任何属性或任何方法。

dyn.whatever(1, "foo", dyn) // 'whatever' is not defined anywhere
dyn.whatever(*array(1, 2, 3))

动态调用可以返回dynamic作为结果,因此我们可以轻松实现链式调用:

dyn.foo().bar.bat()

当给动态调用传递一个 lambda 表达式时,所有的参数默认都是dynamic

dyn.foo {
  x -> x.bar() // x is dynamic
}

参考:

  1. http://kotlinlang.org/docs/reference/basic-syntax.html
  2. http://huanglizhuo.gitbooks.io/kotlin-in-chines

本文转自天天_byconan博客园博客,原文链接:http://www.cnblogs.com/tiantianbyconan/p/4805196.html ,如需转载请自行联系原作者
相关文章
|
16天前
|
Java 开发工具 Android开发
Kotlin语法笔记(26) -Kotlin 与 Java 共存(1)
本系列教程笔记详细讲解了Kotlin语法,适合需要深入了解Kotlin的开发者。若需快速学习Kotlin,建议查看“简洁”系列教程。本期重点介绍了Kotlin与Java的共存方式,包括属性、单例对象、默认参数方法、包方法、扩展方法以及内部类和成员的互操作性。通过这些内容,帮助你在项目中更好地结合使用这两种语言。
31 1
|
17天前
|
Java 开发工具 Android开发
Kotlin语法笔记(26) -Kotlin 与 Java 共存(1)
Kotlin语法笔记(26) -Kotlin 与 Java 共存(1)
29 2
|
17天前
|
IDE 开发工具 数据安全/隐私保护
Kotlin语法 - 类成员
本教程详细讲解Kotlin语法,适合深入学习。若需快速掌握,推荐“简洁”系列教程。内容涵盖类成员、函数与方法、属性访问控制及初始化等。类成员包括方法和属性,函数独立于类,属性访问通过getter/setter实现,支持延迟初始化。
33 3
|
18天前
|
数据安全/隐私保护 Kotlin
Kotlin语法 - 类成员
Kotlin语法 - 类成员
|
16天前
|
Java 编译器 Android开发
Kotlin语法笔记(28) -Kotlin 与 Java 混编
本系列教程详细讲解了Kotlin语法,适合需要深入了解Kotlin的开发者。对于希望快速学习Kotlin的用户,推荐查看“简洁”系列教程。本文档重点介绍了Kotlin与Java混编的技巧,包括代码转换、类调用、ProGuard问题、Android library开发建议以及在Kotlin和Java之间互相调用的方法。
16 1
|
16天前
|
安全 Java 编译器
Kotlin语法笔记(27) -Kotlin 与 Java 共存(二)
本教程详细讲解Kotlin语法,适合希望深入了解Kotlin的开发者。若需快速入门,建议查阅“简洁”系列教程。本文重点探讨Kotlin与Java共存的高级话题,包括属性访问、空安全、泛型处理、同步机制及SAM转换等,助你在项目中逐步引入Kotlin。
16 1
|
17天前
|
Java 编译器 Android开发
Kotlin语法笔记(28) -Kotlin 与 Java 混编
Kotlin语法笔记(28) -Kotlin 与 Java 混编
21 2
|
17天前
|
IDE 开发工具 开发者
Kotlin语法 - 函数与Lambda表达式
本教程详细讲解了Kotlin中的函数与Lambda表达式,包括函数的基本定义、默认返回值类型、匿名函数、Lambda表达式的定义及简化、Lambda与函数引用的结合使用,以及如何在Lambda中实现循环控制。适合希望深入了解Kotlin语法的开发者。
34 1
|
18天前
|
人工智能 Scala Kotlin
Kotlin语法 - 运算符与中缀表达式
Kotlin语法 - 运算符与中缀表达式
|
18天前
|
JavaScript Java Kotlin
Kotlin语法 - 常量与变量
Kotlin语法 - 常量与变量