Kotlin-26.集合(Collection)
官方文档: http://kotlinlang.org/docs/reference/collections.html
1.可变和不可变集合(mutable and immutable collection)
和大多数语言不同,Kotlin区分可变集合和不可变集合(list,set,map等)
因为精确控制集合何时可变,有助于消除bug,设计出良好的API框架!
了解可变集合的只读视图(read-only view)和不可变集合的区别是很重要的,
它们都容易创建,但类型系统不能表现它们区别,所以需要由我们跟踪!
和Java类似,Kotlin的List<out T>类型继承自Collection<T>,进而继承自Iterable<T>,
List<out T>提供只读操作,如size,get等接口;而MutableList<T>提供可变操作,即改变list的方法,
这一模式也适用于:
Set<out T>/ MutableSet<T>,
Map<K, out V>/ MutableMap<K, V>
2.List和Set的基本用法
Kotlin没有专门语法结构创建list和set,需要用kotlin标准库(standard library)的方法,
例如 listOf(),mutableListOf(), setOf(),mutableSetOf()
1.可变集合的只读视图(read-only view)和不可变集合的区别
类型系统不能表现它们区别:
//arrayListOf也是可变的
val arrayList = arrayListOf(1,2,3)
val muList: MutableList<Int> = mutableListOf(1, 2, 3)
val readOnlyView: List<Int> = muList //只读视图(read-only view)
//readOnlyView本身不可变,但指向的muList却是可变
muList.add(4)
println(readOnlyView) //输出"[1, 2, 3, 4]"
//readOnlyView是不可变,arrayListOf和mutableListOf都是可变的
arrayList.clear()
readOnlyView.clear() //readOnlyView不可变,编译错误
//使用listOf集合才是真正的完全不可变!
val list = listOf(1, 2, 3)
//setOf不可变,hashSetOf和mutableSetOf都是可变的
val rSet = setOf("a", "b", "c", "c")
val hSet = hashSetOf("a", "b", "c", "c")
val wSet = mutableSetOf("a", "b", "c", "c")
2.List<out T>和Set<out T>类型都是协变(covariant),
如果Rectangle继承自Shape,可把List<Rectangle>类型赋值给List<Shape>类型,
但是对于可变集合(mutable collection)是不允许的,因为将导致运行时失败!
3.给调用者返回一个永远不变的集合(在某个特定时间的一个快照snapshot)
class Controller {
private val _items = mutableListOf<String>()
//items本身不可变,toList方法复制_items项,因此返回的list永远不会改变
val items: List<String> get() = _items.toList()
}
4.List和set有用的扩展方法
val items = listOf(1, 2, 3, 4)
items.first() == 1
items.last() == 4
items.filter { it % 2 == 0 } //返回 [2, 4]
val rwList = mutableListOf(1, 2, 3)
rwList.requireNoNulls() //返回 [1, 2, 3]
if (rwList.none { it > 6 }) println("No items above 6") // 输出“No items above 6”
val item = rwList.firstOrNull()
3.Map的基本用法
kotlin的Map同样遵循List和Set的模式,很容易实例化和访问!
在非性能关键代码中创建map,可用简单惯用语法(idiom):mapOf(a to b, c to d)
map使用实例如下:
//hashMapOf和mutableMapOf都是可变的
val readWriteMap = hashMapOf("foo" to 1, "bar" to 2)
val readWriteMap = mutableMapOf("foo" to 1, "bar" to 2)
println(readWriteMap["foo"]) //输出“1”
readWriteMap["foo"]=0 //改变值
//snapshot: Map<String, Int>不可变
val snapshot: Map<String, Int> = HashMap(readWriteMap)
//mapOf不可变
val map = mapOf("foo" to 1, "bar" to 2)
4.List,Set和Map的新建对象
kotlin也可以像Java一样,直接使用list/set/map的子类实例化-新建对象,
新建对象变量默认都是可变的,但如果变量类型声明为List<out T>/Set<out T>/Map<K, out V>,
那么新建对象变量就不可变!
fun main(args: Array<String>) {
val l = ArrayList<String>()
//如果声明为val l: List<String>,那么list不可变
l.add("a")
l.add("b")
l.add("b")
val s = HashSet<String>()
//如果声明为val s: Set<String>,那么set不可变
s.add("a")
s.add("b")
s.add("b")
val m = HashMap<Int,String>()
//如果声明为val m: Map<Int,String>,那么map不可变
m.put(1,"a")
m.put(2,"b")
println("$l, $s, $m")//输出:[a, b, b, b], [a, b], {1=a, 2=b}
}
简书:http://www.jianshu.com/p/e1d099c3724f CSDN博客: http://blog.csdn.net/qq_32115439/article/details/74086374
GitHub博客:http://lioil.win/2017/07/01/Kotlin-Collection.html
Coding博客:http://c.lioil.win/2017/07/01/Kotlin-Collection.html