scala-筑基篇-01-List操作

[TOC]

List简介

特性

  • 不可变
  • 递归结构
  • 同构(同质)的:元素的类型必须一致
  • 协变的:如果S是T的子类型,那么List[S]是List[T]的子类型,这点不同于java的泛型
  • 空列表为List[Nothing],Nil

创建列表

  • 所有的List都是由空列表Nil和操作符::构造出来的,::表示从前端扩张列表.
  • 实际上,用形如List(e1,e2,...)的方式创建列表在底层也是使用Nil::构造出来的
scala> List(1,2,3)
res0: List[Int] = List(1, 2, 3)

scala> 1::Nil
res1: List[Int] = List(1)

scala> 2::res1
res2: List[Int] = List(2, 1)

scala> 3::res2
res3: List[Int] = List(3, 2, 1)

操作

list的基本操作

method DESC
head List的第一个元素
tail 除了head之外的其他元素组成的List
isEmpty List是否为空
last List的最后一个元素
init 除了last之外的其他元素组成的List
scala> val l1=List(1,2,3,4,5)
l1: List[Int] = List(1, 2, 3, 4, 5)

scala> l1.tail
res4: List[Int] = List(2, 3, 4, 5)

scala> l1.head
res5: Int = 1

scala> l1.isEmpty
res6: Boolean = false

scala> l1.init
res7: List[Int] = List(1, 2, 3, 4)

scala> l1.last
res8: Int = 5

list类的一阶方法

连接

列表的链接操作符:::和扩展元素操作符::一样都是右结合的,即xs:::ys:::zs等价于xs:::(ys:::zs),不过两个操作数都是List

scala> List(1,2,3):::List(4,5,6)
res9: List[Int] = List(1, 2, 3, 4, 5, 6)

长度

内部定义:

  def length: Int = {
    var these = self
    var len = 0
    while (!these.isEmpty) {
      len += 1
      these = these.tail
    }
    len
  }

所以,length方法是比较费时的

reverse

反转list:reverse ,该方法并不是在原地修改list,因为list是不可变的,所以会返回一个新的list

  • drop和take可以理解为更为广义的tail和init操作
  • take(n)返回列表的前n个元素,if(n>list.length) return list
  • drop(n)返回除了take(n)之外的所有元素,if(n>list.length) return Nil
  • splitAt(n)在位置n处拆分列表,返回位元组。等价于(list.take(n),list.drop(n))
scala> val l1=Range(1,11).toList
l1: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> l1.take(3)
res10: List[Int] = List(1, 2, 3)

scala> l1.drop(3)
res11: List[Int] = List(4, 5, 6, 7, 8, 9, 10)

scala> l1.splitAt(3)
res12: (List[Int], List[Int]) = (List(1, 2, 3),List(4, 5, 6, 7, 8, 9, 10))

apply | indices

scala> val l1=Range(1,11).toList
l1: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> l1.apply(2)
res13: Int = 3

scala> l1(2)
res14: Int = 3

scala> l1.indices
res15: scala.collection.immutable.Range = Range(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)

zip

不匹配的元素将被遗弃

scala> val l1=Range(1,4).toList
l1: List[Int] = List(1, 2, 3)

scala> val l2=List("a","b","c","d")
l2: List[String] = List(a, b, c, d)

scala> l1.zip(l2)
res16: List[(Int, String)] = List((1,a), (2,b), (3,c))

mkString

mkString([start,]seperator[,end])

scala> val l=Range(1,5).toList
l: List[Int] = List(1, 2, 3, 4)

scala> l.mkString("start","|","end")
res17: String = start1|2|3|4end

scala> l.mkString("|")
res18: String = 1|2|3|4

list类的高阶方法

foreach

遍历list并将传入的lambda作用于每个元素

内部实现:

  @inline final override def foreach[U](f: A => U) {
    var these = this
    while (!these.isEmpty) {
      f(these.head)
      these = these.tail
    }
  }

遍历

val l1 = Range(1, 11).toList
l1.foreach(e => { print(e + " ") })

l1.foreach(print(_))

等价的java8操作

List<Integer> list = Stream.iterate(1, i -> i + 1).limit(10)
        .collect(Collectors.toList());
list.forEach(e -> {
    System.out.print(e + " ");
});

list.forEach(System.out::print);

求和

val l1 = Range(1, 11).toList
var sum = 0
l1.foreach(sum += _)
println(sum)

map

参数f:T=>R,将f作用于每个元素,并返回类型为R的新列表

构造新list,元素为原list的元素的2倍

l1.map(e => e * 2)
println(l1)

等价的java8代码

list.stream().map(e->e*2).collect(Collectors.toList());

flatMap

和map类型,但请注意返回类型

scala> val l=List("tom","cat","apache")
l: List[String] = List(tom, cat, apache)

scala> l.map(_.toList)
res19: List[List[Char]] = List(List(t, o, m), List(c, a, t), List(a, p, a, c, h, e))

scala> l.flatMap(_.toList)
res20: List[Char] = List(t, o, m, c, a, t, a, p, a, c, h, e)

filter

scala> val l=Range(1,11).toList
l: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> l.filter(e=>(e%2)==0)
res22: List[Int] = List(2, 4, 6, 8, 10)

scala> l.filter(e=>(e&1)==0)
res24: List[Int] = List(2, 4, 6, 8, 10)

partition

返回二元组(符合条件的部分,不符合条件的部分)

scala> val l=Range(1,11).toList
l: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> l.partition(e=>(e%2)==0)
res26: (List[Int], List[Int]) = (List(2, 4, 6, 8, 10),List(1, 3, 5, 7, 9))

find

返回第一个满足条件的

scala> l.find(e=>(e%2)==0)
res27: Option[Int] = Some(2)

takeWhile | dropWhile | span

  • takeWhile(p):返回能够满足p的最长前缀组成的list
  • dropWhile(p):返回list中除了takeWhile(n)的部分组成的列表
  • span:类似于slitAt结合了take和drop,span结合了takeWhile和dropWhile的结果
scala> val l=List(1,2,3,-1,2,3)
l: List[Int] = List(1, 2, 3, -1, 2, 3)

scala> l.dropWhile(_>0)
res28: List[Int] = List(-1, 2, 3)

scala> l.takeWhile(_>0)
res30: List[Int] = List(1, 2, 3)

scala> l.span(_>0)
res31: (List[Int], List[Int]) = (List(1, 2, 3),List(-1, 2, 3))

论断(forall,exists)

scala> val l=List(1,2,3,-1,2,3)
l: List[Int] = List(1, 2, 3, -1, 2, 3)

scala> l.forall(_>0)
res33: Boolean = false

scala> l.exists(_>0)
res35: Boolean = true

折叠

  • 1.foldLeft(/:)

内部实现:

//注意op第一个参数类型为seed的类型
  override /*TraversableLike*/
  def foldLeft[B](z: B)(@deprecatedName('f) op: (B, A) => B): B = {
    var acc = z//z表示初始值,seed
    var these = this
    while (!these.isEmpty) {
      //每次将op作用于当前seed和当前元素并将结果重新赋值于seed
      acc = op(acc, these.head)
      these = these.tail
    }
    //最终返回"累加"的结果
    acc
  }
  • 2.foldRight(:\)

内部实现:

//反转后调用foldLeft
//注意op第二个参数类型为seed的类型
override def foldRight[B](z: B)(op: (A, B) => B): B =
  reverse.foldLeft(z)((right, left) => op(left, right))
  • 3.fold

内部实现:

//注意op的两个参数类型相同
def fold[A1 >: A](z: A1)(op: (A1, A1) => A1): A1 = foldLeft(z)(op)

求和

val l1 = Range(1, 11).toList

var sum = l1.foldLeft(0)((acc, e) => acc + e)
println(sum) //55

sum = l1./:(0)((acc, e) => acc + e)
println(sum) //55

sum = l1.foldRight(0)((e, acc) => acc + e)
println(sum) //55

sum = l1.fold(0)((acc, e) => acc + e)
println(sum) //55

sum = l1.foldLeft(0)(_ + _)
println(sum) //55

sum = l1./:(0)(_ + _)
println(sum) //55

sum = (0 /: l1)(_ + _) //相当于 l1./:(0)
//sum = (l1 /: 0)(_ + _) //错误,相当于0./:(l1)
println(sum) //55

乘积

var pro = 1
val l1 = Range(1, 11).toList
pro = l1.foldLeft(1)((acc, e) => acc * e)
println(pro) //3628800

pro = (1 /: l1)(_ * _)
println(pro) //3628800

排序

  • 1.sortWith(p: (Int, Int) => Boolean)
val l1 = Range(1, 11).toList

var l2 = l1.sortWith((l, r) => l > r)
println(l2) //List(10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

l2 = l1.sortWith(_ > _)
println(l2) //List(10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

list对象的方法

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

推荐阅读更多精彩内容