美文网首页
Kotlin学习:集合<2>.List操作符

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

作者: 半生黑豆 | 来源:发表于2020-01-16 15:09 被阅读0次

    在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 元素。

    相关文章

      网友评论

          本文标题:Kotlin学习:集合<2>.List操作符

          本文链接:https://www.haomeiwen.com/subject/ngwoactx.html