Kotlin学习:集合<2>.List操作符

在java中,如果我们要判断List集合中的某个元素是否符合判断条件,会有如下代码

 public static void main(String[] args) {
        ArrayList<String> list = createList();
        String matchResult = null;
        for (int i = 0; i < list.size(); i++) {
            if (list.get(i).contains("a")) {
                matchResult = list.get(i);
                break;
            }
        }
        System.out.println(matchResult);
    }
    private static ArrayList<String> createList() {
        ArrayList<String> list = new ArrayList<>();
        list.add("bc");
        list.add("bdf");
        list.add("cca");
        list.add("bbc");
        list.add("efg");
        return list;
    }
>>>cca

for循环这一段模版代码,可能在不同的方法中会多次的用到,而大量的模版代码只会增加工作量。为此,kotlin里封装了很多List集合操作符,方便我们的使用

any() 如果集合至少有一个元素,返回 true ,否则返回 false 。

val emptyList = listOf<Int>()
emptyList.any()
>>>false

val list1 = listOf(1)
list1.any()
>>>true

any(predicate: (T) -> Boolean): Boolean 如果至少有一个元素符合给定条件predicate,返回true,否则返回false

val list = listOf(1, 2, 3)
// 元素2满足{it%2==0}
list.any({it%2==0}) 
>>>true

// 没有元素满足{it>4}
list.any({it>4}) 
>>>false

all(predicate: (T) -> Boolean): Boolean 如果List集合的全部元素都满足给定条件predicate,那么返回true,否则返回false

val list = listOf(0,2,4,6,8)
 list.all({it%2==0})
>>>true
 list.all({it>2})
>>>false

none() 如果集合没有任何元素,返回true,否则返回false

val list = listOf<Int>()
 list.none()
>>>true

none(predicate: (T) -> Boolean): Boolean 如果集合元素全都不满足给定条件predicate,返回true,否则返回false

val list = listOf(0,2,4,6,8)
list.none({it%2==1})
>>> true
list.none({it>0})
>>> false

count()计算集合中元素的个数

val list = listOf(0,2,4,6,8,9)
 list.count()
>>>6

如果点进去查看这个count()函数的源码,会发现,它是直接把List集合的size返回了

public inline fun <T> Collection<T>.count(): Int {
    return size
}

我看到这个函数源码的时候,觉得这个封装有点鸡肋,后来一想,这样写的好处可能是为了方便链式调用吧

count(predicate: (T) -> Boolean): Int 返回集合中满足给定条件predicate的元素的数量

val list = listOf(0,2,4,6,8,9)
 list.count({it%2==0})
>>> 5

reduce(operation: (acc: S, T) -> S): S 以集合第一个元素为初始值,从第一项到最后一项根据operation累计运算,这个操作符结合着源码看更容易理解些

public inline fun <S, T : S> Iterable<T>.reduce(operation: (acc: S, T) -> S): S {
1.    val iterator = this.iterator()
2.   if (!iterator.hasNext()) throw UnsupportedOperationException("Empty collection can't be reduced.")
3.    var accumulator: S = iterator.next()
4.     while (iterator.hasNext()) {
5.         accumulator = operation(accumulator, iterator.next())
6. }
7. return accumulator
}

代码示例

 val list= listOf(1,2,3)
 val result = list.reduce { s, t ->
        print("s=$s ")
        print("t=$t")
        println()
        s + t
    }
    println(result)
>>> s=1 t=2
>>> s=3 t=3
>>> result=6

结合着示例代码,我们在看源码就很容易了,这个函数就是把集合的第一元素赋值给变量accumulator,源码的第3行。然后开始遍历这个集合,并且把当前的accumulator值和集合的下一个值回调给operation,再把operation函数的返回值重新赋值给accumulator,如此重复,源码4、5行。最后把最终的运算结果accumulator返回

reduceIndexed(operation: (index: Int, acc: S, T) -> S): S 这个函数和reduce函数的区别就是加入了集合的下标,如果需要用到下标就用这个,不需要下标就用reduce

reduceRight(operation: (T, acc: S) -> S): S 以集合最后一个元素为初始值,从最后一项到第一项根据operation累计运算

 val list = listOf("1", "2", "3")
  val result = list.reduceRight { i, acc ->
         i + acc
    }
>>> 123

reduceRight 操作符使用的时候需要注意的是,在拼接字符串的时候i + accacc + i结果是不同的

 val list = listOf("1", "2", "3")
  val result = list.reduceRight { i, acc ->
         acc + i
    }
>>> 321

reduceRightIndexed(operation: (index: Int, T, acc: S) -> S): S 这个函数和**reduceRight **的区别也是加入了下标。

fold(initial: R, operation: (acc: R, T) -> R): R 以initial为初始值,从第一项到最后一项根据operation累计运算。这个操作符就相当于是带初始值的reduce

 val list=listOf(1,2,3,4)
 list.fold(10,{total, next -> next + total})
>>> 20

foldIndexed(initial: R, operation: (index: Int, acc: R, T) -> R): Rfold的区别也是将集合元素对应的下标一并回调给operation

foldRight(initial: R, operation: (T, acc: R) -> R): R 以initial为初始值,从最后一项到第一项根据operation累计运算。这个操作符就相当于是带初始值的reduceRight

val list = listOf("a","b","c")
list.foldRight("xyz",{s, pre -> pre + s})
>>> xyzcba

拼接字符串的时候同样需要注意顺序
reduceRightIndexed(operation: (index: Int, T, acc: S) -> S): SfoldRight的区别也是将集合元素对应的下标一并回调给operation

forEach(action: (T) -> Unit): Unit 遍历集合

val list = listOf(0, 1, 2, 3, 4)
list.forEach { value -> print(value) }
>>> 01234

forEachIndexed(action: (index: Int, T) -> Unit): Unit 带下标遍历集合

 val list = listOf("1", "2", "3")
    list.forEachIndexed { index, value ->
        print("index=$index,value=$value ")
    }
>>>index=0,value=1 index=1,value=2 index=2,value=3

max(): T? 返回集合中最大的元素。字符串也可以直接比较,字符串的大小比较是按照对应的下标的字符进行比较的。 另外,布尔值的比较是 true 大于 false :

 val list = listOf(1, 3,4 )
    println(list.max())
    >>>4
    val strList = listOf("你", "我","他" )
    println(strList.max())
    >>>我

maxBy(selector: (T) -> R): T? 获取selector运算后的最大值对应的那个元素的值,如果集合为空则直接返回null,概念比较抽象,结合者源码就很容易理解了

public inline fun <T, R : Comparable<R>> Iterable<T>.maxBy(selector: (T) -> R): T? {
    val iterator = iterator()
    if (!iterator.hasNext()) return null
    var maxElem = iterator.next()
    if (!iterator.hasNext()) return maxElem
    var maxValue = selector(maxElem)
    do {
        val e = iterator.next()
        val v = selector(e)
        if (maxValue < v) {
            maxElem = e
            maxValue = v
        }
    } while (iterator.hasNext())
    return maxElem
} 

不直接比较集合元素的大小,而是以集合元素为入参的函数 selector: (T) -> R 返回值来比较大小,最后返回此元素的值(注意,不是对应的 selector 函数的返回值)

  val list = listOf(1, 3, 4)
    println( list.maxBy {
        it > 3
    })
>>>4

这里需要注意的一点是,上面提到过,boolean值比较大小的时候true大于false

minBy(selector: (T) -> R): T? 获取传参函数selector运算后的最小值对应的那个元素的值,如果集合为空则直接返回null。和maxby类似只不过返回最小值。

sum() 返回集合中所有元素的和。
sumBy(selector: (T) -> Int): Int 返回应用于集合中每个元素的[selector]函数生成的所有值的总和。

val list = listOf(1,2,3,4)
list.sumBy({it})
>>> 10
list.sumBy({it*it})
>>> 30

sumBy 函数返回值是 Int 。它的入参 selector 是一个函数类型 (T) -> Int ,也就是说这个selector也是返回Int类型的函数。

take(n: Int): List<T> 挑出该集合前n个元素的子集合,如果n等于0,返回空集;如果n大于集合 size ,返回该集合

val list = listOf(1, 3, 4)
    println(list.take(2))
>>> [1, 3]

takeWhile(predicate: (T) -> Boolean): List<T> 挑出满足条件的元素的子集合。

public inline fun <T> Iterable<T>.takeWhile(predicate: (T) -> Boolean): List<T> {
    val list = ArrayList<T>()
    for (item in this) {
        if (!predicate(item))
            break
        list.add(item)
    }
    return list
}

从第一个元素开始,判断是否满足 predicate 为true,如果满足条件的元素就添加到返回 ArrayList 中。只要遇到任何一个元素不满足条件,就结束循环,返回list 。

takeLast(n: Int): List<T> 挑出后n个元素的子集合。从集合倒数n个元素起,取出到最后一个元素的子集合。如果传入0,返回空集。如果传入n大于集合size,返回整个集合。如果传入负数,直接抛出IllegalArgumentException。

takeLastWhile(predicate: (T) -> Boolean) 从最后开始挑出满足条件元素的子集合,遇到不满足的元素,直接终止循环,并返回子集合
drop(n: Int) 去除前n个元素返回剩下的元素的子集合
dropWhile(predicate: (T) -> Boolean) 去除满足条件的元素返回剩下的元素的子集合,当遇到一个不满足条件的元素时,中止操作,返回剩下的元素子集合。

val list = listOf(1, 3, 4)
println(list.dropWhile1 {it>3 })
>>>[1, 3, 4]

示例代码中,判断条件是大于3的元素,而集合中第一个元素是1,不满足条件,所以直接将1和剩下的元素放入集合返回
dropLast(n: Int) 从集合最后去除n个元素
dropLastWhile(predicate: (T) -> Boolean) 从最后去除满足条件的元素
slice(indices: IntRange): List<T> 返回包含指定indices范围内元素的子集合。

val list = listOf(2,4,6,8,9,11,12,16)
 list.slice(1..3)
>>>[4, 6, 8]
list.slice(2..7)
>>> [6, 8, 9, 11, 12, 16]

slice(indices: Iterable<Int>) 返回指定下标的元素子集合

val list = listOf(2,4,6,8,9,11,12,16)
list.slice(listOf(2,4,6))
>>>[6, 9, 12]

filterTo(destination: C, predicate: (T) -> Boolean)取出满足条件的元素并赋值给destination

val list = listOf(1,2,3,4,5,6,7)
val dest = mutableListOf<Int>()
list.filterTo(dest,{it>3})
>>>[4, 5, 6, 7]

filter(predicate: (T) -> Boolean) 返回满足条件元素的子集合,这个函数直接调用的filterTo,如果对返回的子集合没有特殊要求,使用这个函数更简便。另外,还有下面常用的过滤函数filterNot(predicate: (T) -> Boolean) 用来过滤所有不满足条件的元素 ; filterNotNull()过滤掉 null 元素。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容