for循环和foreach循环
fun main(args: Array<String>) {
val s = "Kotlin"
//普通的for循环遍历
for (a in s) {
println("a = $a")
}
//for循环遍历加上角标
for ((b, index) in (s.withIndex())) {
println("index = $index , b = $b")
}
//foreach循环遍历
s.forEach { println(it) }
//foreach循环遍历加上角标
s.forEachIndexed { index, c -> println("index = $index , c = $c") }
}
break和continue
break和continue的用法同java。在kotlin中,foreach不可被打断,所以不能用break或continue
fun main(args: Array<String>) {
val str = "abcde"
//遍历元素:找到c就停止.
for (c in str) {
println(c)
if (c.equals('c'/*字符串的遍历方法还是变为字符进行打印*/))
break
}
//遍历元素:除了c其它都打印
for (c in str) {
if (c.equals('c')){
continue
}
println(c)
}
}
在特定位置返回
在for循环中可以嵌套循环,方法同java,但是当想要在特定位置返回时,需要在返回的循环前加上返回名+@,并且break也需要加返回名+@。返回名可以任意命名,但不可是关键字
fun main(args: Array<String>) {
val str1 = "abc"
val str2 = "ABC"
//找到str1和str2每一个元素组合
for (c1 in str1) {
for (c2 in str2) {
println("${c1}$c2")
}
}
//找到bB之后就不再打印了
jump@ for (c1 in str1) {
for (c2 in str2) {
println("${c1}$c2")
if (c1 == 'b' && c2 == 'B') {
break@jump
}
}
}
}
区间
kotlin有三种区间,分别是IntRange、CharRange、LongRange。区间有三种定义的方法
三种区间的定义方法
fun main(args: Array<String>) {
//定义整形区间
val range1 = IntRange(1, 100)
val range2 = 1.rangeTo(100)
val range3 = 1..100
//定义字符区间
val charrange1 = CharRange('a', 'b')
val charrange2 = 'a'.rangeTo('b')
val charrange3 = 'a'..'b'
//定义长整型区间
val longrange1 = LongRange(1L, 100L)
val longrange2 = 1L.rangeTo(100L)
val longrange3 = 1L..100L
}
区间的遍历
区间遍历同样可以使用for和foreach
fun main(args: Array<String>) {
val range = 1..10
//用for循环遍历
for (i in range) {
println(i)
}
//加上步长的遍历
for (i in range.step(2)) {
println(i)
}
//用for循环遍历,加上索引
for ((index, i) in range.withIndex()) {
println("index = $index ,i = $i")
}
//用foreach循环遍历
range.forEach {
println(it)
}
//用foreach循环遍历,加上索引
range.forEachIndexed { index, i -> println("index = $index , i = $i") }
}
反向区间和区间的反转
方向区间可使用down to()实现,区间反转同java中的 reversed()方法
//定义反转区间
val range = 10 downTo 1
for (i in range) {
println(i)
}
//区间反转
val range2 = range.reversed()
range2.forEach { println(it) }
数组
数组的创建
kotlin中8种基本类型都有对应的数组类型。建立String类型数组可通过Array<String>来创建。
fun main(args: Array<String>) {
//数组的定义
//如果数组中元素已知
val arr1 = arrayOf("张三","李四","王五")
val arr2 = arrayOf(1,2,3)
val arr3 = arrayOf("张三",20,"男")//可以设置不同类型元素的数组
//如果数组中元素未知
val arr4 = Array<Int>(10){0}//创建一个长度为10的int类型数组,数组内所有元素初始化值为0
val arr5 = IntArray(10)//创建一个长度为10的int类型数组
}
数组的遍历
数组遍历同样可以用for、for index、foreach、foreach index来做
fun main(args: Array<String>) {
val arr = arrayOf(1,2,3,4,5,6)
//for
for (s in arr) {
println(s)
}
//for index
for ((i,index) in arr.withIndex()) {
println("i = $i ,index = $index")
}
//foreach
arr.forEach { println(it) }
//foreach index
arr.forEachIndexed { index, i -> println("index = $index , i = $i") }
}
数组元素访问以及修改
fun main(args: Array<String>) {
val arr = arrayOf(1, 2, 3, 4, 5, 6)
//数组元素访问以及修改
//访问某一个元素
val e = arr[1]//访问第二个元素
println(e)
//数组元素的修改
//第三个元素修改为10
arr[2] = 10
println(arr[2])
arr.set(3, 11)
}
查找数组元素角标
fun main(args: Array<String>) {
val arr = arrayOf("张三","李四","张五","王五","张四","张三")
//查找第一个"张三"角标
val indexOf = arr.indexOf("张三")
println(indexOf)
//查找最后一个"张三"角标
val lastIndexOf = arr.lastIndexOf("张三")//方法一
println(lastIndexOf)
val array = arr.indexOfLast { it.equals("张三") }
println(array)
//查找第一个姓"张"的人的角标
val arr1 = arr.indexOfFirst { it.startsWith("张") }
println(arr1)
//查找最后一个姓"张"的人的角标
val arr2 = arr.indexOfLast { it.startsWith("张") }
println(arr2)
}
When表达式
when表示式是多分支的条件控制语句,在kotlin中,如果when表达式分支只有一行,可以省略{}。支持的数据格式比java要多,除了java支持的6中数据格式之外可以支持区间,表达式,判断语句。
如果是简单的when表达式 支持的6中数据格式,when表达式最终都会翻译成switch语句。如果是加强的when表达式,最终会翻译成if else形式。
fun todo(age: Int): String {
return when (age) {
in 1..6 -> "没有上学"
6 -> "开始上小学"
in 7..11 -> "正在上小学"
12 -> "开始上中学"
13, 14 -> "正在上中学"
15 -> "开始上高中"
16, 17 -> "正在上高中"
18 -> "开始上大学"
in 19..22 -> "正在上大学"
else -> "上社会大学"
}
}
when表达式有返回值,如果返回when表达式必须要有else操作。对于when表达式返回值是{}最后一行。
fun todo(age: Int): Any {
return when {
age == 7 -> {
"haha"
println("hello")//如果方法没有返回值,返回默认是Unit
10
"开始上小学"
}
}
输出的结果为 hello,开始上小学
函数表达式
函数表达式:如果函数只有一行处理代码,就可以省略掉{,以= 连接,去掉return以及返回值类型。
函数表达式只能适用于一行代码的函数
fun main(args: Array<String>) {
val a = 10
val b = 20
//求a加b的和
}
//标准格式
fun add(a:Int,b:Int):Int{
return a+b
}
//由于处理代码只有一行,可以省略掉{},以= 连接,去掉return
fun add2(a:Int,b:Int):Int= a+b
//智能类型推断
fun add3(a:Int,b:Int) = a+b
函数变量和函数引用
在Kotlin中,对象和函数的地位相同。对象可以定义对象变量,函数也可以定函数变量。
创建函数变量
fun main(args: Array<String>) {
val a = 10
val b = 20
var sum = 0
//函数变量
//求a+b
//定义一个函数变量
val padd: ((Int, Int) -> Int)? = { m, n -> m + n }//定义函数
sum = padd?.invoke(a,b)!!//调用函数
println(sum)
}
函数应用
fun main(args: Array<String>) {
val a = 10
val b = 20
var sum = 0
val padd=::add//获取函数引用
//调用函数
println(padd(a, b))
println(padd.invoke(a, b))
}
fun add(a: Int,b: Int):Int{
return a+b
}
可变参数
如果接收的参数个数不确定,可以用可变参数表示。
求多个int类型的和
fun add(vararg a:Int):Int{
var count = 0
for (item in a) {
count+=item
}
return count
}
异常处理
kotlin只有运行时异常,没有受检异常。即无论方法有没有抛出异常,编译器都不会提示处理这个异常(什么鬼设计= =)
fun main(args: Array<String>) {
val a = 10
val b = 0
var c: Int =0
try {
c = a/b
} catch (e: Exception) {
println("出现了异常")
}finally {
}
println(c)
}
递归
递归的思路同java
计算斐波那契数列第n项的值
fun main(args: Array<String>) {
println(fbnq(5))
}
fun fbnq(n: Int):Int{
if (n == 1 || n == 2) {
return 1
}
else{
return fbnq(n-1)+ fbnq(n-2)
}
}
递归和迭代的比较
kotlin函数参数都是不可变的。一般情况下,递归能解决的问题,用普通迭代也能解决。有一些问题我们用递归写起来会比较简单,因为递归更符合思维逻辑。而迭代需要抽象出数学模型才能解决。但是,如果递归的层级比较深的话就会出现栈内存溢出的异常。递归写起来简单,但是容易内存溢出;迭代写起来麻烦,但是不容易溢出内存。那么,能不能有一种方式既能够写起来简单,又能够不会内存溢出呢?于是,kotlin中有了尾递归优化。
尾递归优化
尾递归优化需要一个前提:这个递归必须是尾递归。调用当前递归的方法之后,没有做任何其它操作就是尾递归。
尾递归优化原理:帮我们把递归转换为迭代,按照递归的方式写代码,按照迭代的方式执行。
tailrec fun add(n: Int,result:Int = 0):Int{
if (n == 1){
return result+1
}
else{
return add(n-1,result+n)
}
}
tailrec就是尾递归优化的代码,添加在方法前面