Programming In Scala笔记-第十七章、Scala中的集合类型
本章主要介绍Scala中的集合类型,主要包括:Array
, ListBuffer
, Arraybuffer
, Set
, Map
和Tuple
。
一、序列
序列类型的对象中包含多个按顺序排列好的元素,可以访问其中特定位置的元素。序列类型主要包括List
,Array
, ListBuffer
, ArrayBuffer
。
1、List
List
在前一章已经介绍过,略。
2、Array
数组在很多编程语言中都会用到。下面代码中包括了如何定义一个Array
类型变量,如何在定义Array
变量时赋初始值,以及如何访问和更新特定位置的元素。
val fiveInts = new Array[Int](5) // 生成一个长度为5的数组
val fiveToOne = Array(5, 4, 3, 2, 1) // 生成一个长度为5的数组并初始化
fiveInts(0) = fiveToOne(4) // 访问指定位置上的元素并赋值
fiveInts
运行结果如下,
Scala中的数组和Java中的数组是相同的。
3、ListBuffer
ListBuffer
类型的完整路径名为scala.collection.mutable.ListBuffer
。
在ListBuffer
对象的前面和后面增加新元素都非常的方便,使用+=
方法在该对象后新增元素,使用+=:
在该对象前新增元素。当元素添加完毕后,调
用toList
方法将ListBuffer
对象转化成List
,使得构建List
对象更加高效。
import scala.collection.mutable.ListBuffer
val buf = new ListBuffer[Int]
buf += 1
buf += 2
buf
3 +=: buf
buf.toList
结果如下,
4、ArrayBuffer
ArrayBuffer
的完整路径是scala.collection.mutable.ArrayBuffer
。ArrayBuffer
和Array
类似,支持所有Array
上的操作。
使用ArrayBuffer
的好处是,定义时可以不指定长度。并且在ArrayBuffer
的头部和尾部添加或移除元素更加便捷。
import scala.collection.mutable.ArrayBuffer
val buf = new ArrayBuffer[Int]()
buf += 12
buf += 15
buf
buf.length
buf(0)
运行结果如下,
5、String
String
对象本质上是字符序列。可以直接在String
对象上进行遍历,遍历的内容就是该对象的每一个字符元素。
val a = "abcdefg"
a.foreach(println)
def hasUpperCase(s: String) = s.exists(_.isUpper)
hasUpperCase("Robert Frost")
hasUpperCase("e e cummings")
运行结果如下,
二、Set
和Map
Scala中同时提供了mutable
和immutable
类型的Set
和Map
类型。有关Set
和Map
类型的类继承关系,如下图所示,
Map
类继承关系:
Set
类继承关系:
从上面两图可以看到,分别有三个Map
和Set
命名的trait,只不过这三个trait位于不同的package中。直接使用Map
或者Set
,在默认情况下,使用的是imutable
这个。如果想要实现mutable
类型的Set
或Map
需要显式的import。
下面分别介绍Set
和Map
类型。
1、Set
Set
对象中不会包含重复元素。下面将一个String
类型变量切割成单词,然后生成一个Set
对象。
val text = "See Spot run. Run, Spot. Run!"
val wordsArray = text.split("[ !,.]+")
val words = mutable.Set.empty[String]
for (word <- wordsArray)
words += word.toLowerCase
words
运行结果如下,最终的Set
对象中只包含了三个不重复的元素。
Set
对象上的常用操作列表如下,注意区分哪些是mutable
类型的Set
,哪些是imutable
类型的Set
。
操作符 | 作用及结果 |
---|---|
val nums = Set(1, 2, 3) |
生成一个新的Set 对象,并初始化 |
nums + 5 |
新增一个元素,得到一个新的Set 对象,结果为Set(1, 2, 3, 5) |
nums - 3 |
删除Set 中的一个元素,结果为Set(1, 2, 5) |
nums ++ List(5, 6) |
将List 中的元素全部增加到Set 中,结果为Set(1, 2, 3, 5, 6) |
nums -- List(1, 2) |
去除Set 中的包含在List 中的元素,结果为Set(3) |
nums $ Set(1, 3, 5, 7) |
返回两个Set 对象中共有的元素,结果为Set(1, 3) |
nums.size |
获取当前Set 中的元素个数,结果为3 |
nums.contains(3) |
判断nums 中是否包含指定元素,结果为true |
import scala.collection.mutable |
引入mutable类型的Set 对象 |
val words = mutable.Set.empty[String] |
生成一个空的mutable.Set 对象 |
words += "the" |
往words 中增加元素,words 的内容随之发生变化,结果为Set(the) |
words -= "the" |
从words 中移除元素,此时words 为空,结果为Set() |
words ++= List("do", "re", "mi") |
将List 中的元素全部添加到words 中,结果为Set(do, re, mi) |
words --= List("do", "re") |
将List 中包含的元素从words 中移除,结果为Set(mi) |
words.clear |
清空words ,移除全部元素,结果为Set() |
2、Map
直接看示例,展示了如何新建Map
对象,如何往Map
对象中添加键值对,以及如何从Map
对象中获取指定key
的value
值。
val map = mutable.Map.empty[String, Int]
map("hello") = 1
map("there") = 2
map
map("hello")
运行结果为,
Map
上的常见操作如下,注意区分哪些是mutable
类型的Map
对象,哪些是imutable
类型的Map
对象。
操作符 | 作用及结果 |
---|---|
val nums = Map("i" -> 1, "ii" -> 2) |
生成一个新的Map 对象,并初始化 |
nums + ("vi" -> 6) |
新增一个元素,得到一个新的Map 对象,结果为Map(i -> 1, ii -> 2, vi -> 6) |
nums - "ii" |
删除Map 中的一个元素,结果为Map(i -> 1) |
nums ++ List("iii" -> 3, "v" -> 5) |
将List 中的元素全部增加到Map 中,结果为Map(i -> 1, ii -> 2, iii -> 3, v -> 5) |
nums -- List("i", "ii") |
去除Map 中的key 为包含在List 中的元素,结果为Map() |
nums.size |
获取当前Map 中的键值对个数,结果为2 |
nums.contains("ii") |
判断nums 中是否包含指定key 为ii 的键值对,结果为true |
nums("ii") |
获取指定key 的value 值,结果为2 |
nums.keys |
获取当前nums 对象中所有的key 值 |
nums.keySet |
获取当前nums 对象中所有的key 值 ,以Set 形式返回。结果为Set(i, ii) |
nums.values |
获取当前nums 对象中所有的value 值 |
nums.isEmpty |
判断nums 是否为空,结果为false |
import scala.collection.mutable |
引入mutable类型的Map 对象 |
val words = mutable.Map.empty[String, Int] |
生成一个空的mutable.Map 对象 |
words += ("one" -> 1) |
往words 中增加元素,words 的内容随之发生变化,结果为Map(one -> 1) |
words -= "one" |
从words 中移除元素,此时words 为空,结果为Map() |
words ++= List("one" -> 1, "two" -> 2, "three" -> 3) |
将List 中的键值对全部添加到words 中,结果为Map(one -> 1, two -> 2, three -> 3) |
words --= List("one", "two") |
将List 中包含的key 从words 中移除,结果为Map(three -> 3) |
3、TreeMap
和TreeSet
的其他子类
Map
和Set
还有其他形式的子类,比如TreeSet
,TreeMap
。
前面展示的Map
和Set
中的元素都是无序的,需要有序的Map
和Set
可以使用TreeMap
和TreeSet
类型。TreeSet
是对其中的元素进行排序,TreeMap
是对其中键值对的Key
进行排序,底层使用的是红黑树。
import scala.collection.immutable.TreeSet
val ts = TreeSet(9, 3, 1, 8, 0, 2, 7, 4, 6, 5)
val cs = TreeSet('f', 'u', 'n')
import scala.collection.immutable.TreeMap
var tm = TreeMap(3 -> 'x', 1 -> 'x', 4 -> 'x')
tm += (2 -> 'x')
tm
运行结果如下,
三、初始化集合
在前面的示例代码中可以看到,一般是调用该类型伴生对象的工厂方法,并且传入初始参数值来初始化一个集合的对象。比如
List(1, 2, 3)
Set('a', 'b', 'c')
import scala.collection.mutable
mutable.Map("hi" -> 2, "three" -> 5)
Array(1.0, 2.0, 3.0)
运行结果如下,从结果可以看到,Scala编译器会根据传入初始参数的类型,推断出集合中的元素类型。
但是,对于mutable
类型的集合对象来说,由于需要动态的增加新元素,最好在初始化集合对象时手动声明其类型,否则会出现以下报错情况。
val stuff = mutable.Set(42)
stuff += "abracadabra"
报错如下,由于初始化时传入的是Int
类型元素,Scala编译器会认为该Set
对象中的元素必须是Int
类型,当新增一个String
类型元素时,就会报出类型不匹配的错。
正确的做法如下,
val stuff: mutable.Set[Any] = mutable.Set(42)
stuff += "abracadabra"
结果如下,
四、元组
Scala中提供一种特殊的类型,Tuple
。Tuple
中可以存储不同类型的元素,元组元素用圆括号包围。如下所示
val t = (1, "hello", Console)
访问Tuple
中的元素,使用_1
表示第一个元素,_2
表示第二个元素,以此类推。比如
t._1
t._2
t._3
运行结果如下,
也可以在生成Tuple
对象时直接指定各元素对应的变量名,相当于同时声明并初始化多个元素。需要注意的是变量个数与元组中元素个数需要保持一致。
val (t1, t2, t3) = (1, "hello", Console)
结果如下,
Programming In Scala笔记-第十七章、Scala中的集合类型的更多相关文章
- Programming In Scala笔记-第十一章、Scala中的类继承关系
本章主要从整体层面了解Scala中的类层级关系. 一.Scala的类层级 在Java中Object类是所有类的最终父类,其他所有类都直接或间接的继承了Object类.在Scala中所有类的最终父类为A ...
- Programming In Scala笔记-第五章、Scala中的变量类型和操作
这一章的一些基础性的东西,主要包括Scala中的基本变量类型,以及相关的一些操作符. 一.简单类型 下表中列出Scala语言中的基本类型,以及其字节长度,其中Byte, Short, Int, Lon ...
- Programming In Scala笔记-第七章、Scala中的控制结构
所谓的内建控制结构是指编程语言中可以使用的一些代码控制语法,如Scala中的if, while, for, try, match, 以及函数调用等.需要注意的是,Scala几乎所有的内建控制结构都会返 ...
- [汇编学习笔记][第十七章使用BIOS进行键盘输入和磁盘读写
第十七章 使用BIOS进行键盘输入和磁盘读写 17.1 int 9 中断例程对键盘输入的处理 17.2 int 16 读取键盘缓存区 mov ah,0 int 16h 结果:(ah)=扫描码,(al) ...
- C++ primer plus读书笔记——第14章 C++中的代码重用
第14章 C++中的代码重用 1. 使用公有继承时,类可以继承接口,可能还有实现(基类的纯虚函数提供接口,但不提供实现).获得接口是is-a关系的组成部分.而使用组合,类可以获得实现,但不能获得接口. ...
- Programming In Scala笔记-第六章、函数式对象
这一章主要是以定义和完善一个有理数类Rational为线索,分析和介绍有关类定义,构造函数,方法重写,变量定义和私有化,以及对操作符的定义等. 一.Rational类定义和构造函数 1.定义一个空类 ...
- Programming In Scala笔记-第四章、类和对象
类似于Java,Scala中也有类和对象的概念. 一.类.属性和方法 1.类 类是对一类事物的抽象,当一个类被定义后,就可以以该定义为模板,定义该类的一系列对象.比如说有以下一个模板 人类: 有姓名: ...
- 《linux内核设计与实现》读书笔记第十七章
第17章 设备与模块 四种内核成分 设备类型:在所有 Unix 系统中为了统一普通设备的操作所采用的分类. 模块: Linux 内核中用于按需加载和卸载目标码的机制. 内核对象:内核数据结构中支持面向 ...
- The C++ Programming Language 学习笔记 第6章 表达式和语句
1.关于strcpy函数. 书中说c风格的字符串尽量少用,strcpy这样的函数应该也要少用.这里讲这个函数主要是要通过本章课后练习第十题来讲一下前面提及的要点.巩固一下前几章的知识.写了一段,本来感 ...
随机推荐
- Spring-cloud(四)服务发现与消费:ribbon的使用
说明: ribbon是spring-cloud中作为服务消费者的一种角色,客户端可以通过它来对服务提供者的服务进行消费, 比如本例中是服务提供者注册到注册中心,服务提供者提供了一个服务接口,返回一个h ...
- .Net Core 通过依赖注入和动态加载程序集实现宿程序和接口实现类库完全解构
网上很多.Net Core依赖注入的例子代码,例如再宿主程序中要这样写: services.AddTransient<Interface1, Class1>(); 其中Interface1 ...
- 连接mysql数据库报错java.sql.SQLException: The server time zone value '�й���ʱ��' is unrecognized...解决方法
今天连接mysql数据库报错如下: java.sql.SQLException: The server time zone value '�й���ʱ��' is unrecognized or r ...
- C++因继承引发的隐藏与重写
在区分隐藏和重写之前,先来理一理关于继承的东西... [继承] 继承是面向对象复用的重要手段.通过继承定义一个类,继承是类型之间的关系建模,共享公有的东西,实现各自本质不同的东西.简单的说,继承就是指 ...
- Java学习图形界面+网络编程案例---------网络简易通讯
主要思想: 主类继承JPanel,在构造方法中将JFrame设成空布局:在其中适当位置添加组件:实现事件监听处理 DATE:2015-10-31 服务器端代码: /** * @author Oyc * ...
- [HNOI 2009]最小圈
Description 考虑带权的有向图$G=(V,E)$以及$w:E\rightarrow R$,每条边$e=(i,j)(i\neq j,i\in V,j\in V)$的权值定义为$w_{i,j}$ ...
- hdu 1133 Buy the Ticket(Catalan)
Buy the Ticket Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) T ...
- 使用python+requests+unittest实现接口自动化测试
这两天一直在找直接用python做接口自动化的方法,在网上也搜了一些博客参考,今天自己动手试了一下. 一.整体结构 上图是项目的目录结构,下面主要介绍下每个目录的作用. Common:公共方法:主要放 ...
- 【译】基于主机的卡仿真(Host-based Card Emulation)
基于主机的卡仿真(Host-based Card Emulation) 能提供NFC功能很多Android手机已经支持NFC卡模拟.在大多数情况下,该卡是由设备中的单独的芯片仿真,所谓的安全元件.由无 ...
- python webdriver环境搭建
一.准备安装包 1.下载python 2.下载setuptools 3.下载pip 二.windows环境安装 1.安装python,建议选择python2.7.5版本. 2.安装setuptools ...