Scala学习(三)----数组相关操做

数组相关操做前端

摘要:java

本篇主要学习如何在Scala中操做数组。JavaC++程序员一般会选用数组近似的结构(好比数组列表向量)来收集一组元素。在Scala中,咱们的选择更多,不过如今咱们先假定不关心其余选择,而只是想立刻开始用数组。本篇的要点包括:程序员

1. 若长度固定则使用Array,若长度可能有变化则使用ArrayBuffer算法

2. 提供初始值时不要使用new编程

3. 用()来访问元素数组

4. 用for (elem<-arr)来遍历元素数据结构

5. 用for (elem<-arr if…)…yield…来将原数组转型为新数组函数式编程

6. Scala数组和java数组能够互操做;用AnayBuffer,使用scalacollection.JavaConversions中的转换函数函数

定长数组学习

若是你须要一个长度不变的数组,能够用Scala中的Array。例如:

val nums=new Array[Int] (10) //长度为10的整数数组,全部元素初始化为0

val a=new Array [String] (10) //长度为10的字符串数组,全部元素初始化为null

val s= Array("Hello", "World") //长度为2的Array[String]类型是推断出来的,已提供初始值就不须要new

S (0) ="Goodbye" //Array("Goodby ","World"),使用()而不是[]来访问元素

在JVM中,Scala的Array以Java数组方式实现。示例中的数组在JVM中的类型为java.lang.String[]。Int、Double或其余与Java中基本类型对应的数组都是基本类型数组。

举例来讲,Array(2,3,5,7,11)在JVM中就是一个int[]

变长数组:缓冲

尾端操做缓冲数组

对于那种长度按须要变化的数组Java有ArrayList,C++有vector。Scala中的等效数据结构为ArrayBuffer

import scala.collection.mutable.ArrayBuffer

val b=ArrayBuffer[lnt]() // 或者new ArrayBuffer [int],一个空的数组缓冲,准备存放整数

b+=1 // ArrayBuffer (1),用+=尾端添加元素

b+=(1,2,3,5) // ArrayBuffer(1,1,2,3,5),在尾端添加多个元素,以括号包起来

b++= Array(8, 13, 21) // ArrayBuffer(1, 1, 2, 3, 5, 8,13, 21) //用++=操做符追加任何集合

b.trimEnd(5) // ArrayBuffer(1, 1, 2),移除最后5个元素

在数组缓冲的尾端添加或移除元素是一个高效的操做

任意位置操做缓冲数组

你也能够在任意位置插入或移除元素,但这样的操做并不那么高效。全部在那个位置以后的元素,都必须被平移。举例以下:

b.insert (2,6) //ArrayBuffer(1, 1, 6, 2),在下标2以前插入

b.insert (2,7,8,9) // ArrayBuffer(1, 1,7,8,9, 6,2),你能够插入任意多的元素

b.remove(2) // ArrayBuffer(1,1,8,9,6,2)

b.remove (2,3) //ArrayBuffer(1,1, 2),第2个参数的含义是要移除多少个元素

有时你须要构建一个Array,但不知道最终须要装多少元素。在这种状况下,先构建一个数组缓冲,而后调用:

b.toArray //Array(1, 1,2)

反过来,调用乱toBuffer能够将一个数组a转换成一个数组缓冲

遍历数组和数组缓冲

全遍历

在Java和C++中,数组数组列表向量有一些语法上不一样,Scala则更加统一。大多数时候,你能够用相同的代码处理这两种数据结构。如下是for循环遍历数组或数组缓冲的语法:

for (i <- 0 until a.length) //变量i的取值从0到a length -1

println(i+":"+a(i))

utiI是Richlnt类的方法,返回全部小于但不包括上限的数字。例如:

0 until 10 // Range(0,1,2,3,4,5,6,7,8, 9)

须要注意的是,0 until 10其实是一个方法调用:0.until(10)

条件遍历

以下结构:

for(I <- 区间)

会让变量i遍历该区间的全部值。拿本例来讲,循环变量i前后取值0、1,等等,直到但不包含a.length。若是想要每两个元素一跳,可让i这样来进行遍历:

0 until (a.length,2) //Range(0,2,4,…)

若是要从数组的尾端开始,遍历的写法为:

(0 until a.length) .reverse //Range(...,2,1,0)

若是在循环体中不须要用到数组下标,咱们也能够直接访问数组元素,就像这样:

for (elem <- a)

println (elem)

这和Java中的"加强版"for循环,或者C++中的"基于区间的"for循环很类似。变量elem前后被设为a(0),而后a(1),依此类推

数组转换

for中的推导式和守卫

在前面,你看到了如何像Java或C++那样操做数组。不过在Scala中,你能够走得更远。从一个数组或数组缓冲出发,以某种方式对它进行转换是很简单的。这些转换动做不会修改原始数组,而是产生一个全新的数组。像这样使用for推导式

val a=Array(2, 3, 5, 7, 11)

val result=for (elem <- a) yield 2*elem //result是Array(4,6,10, 14, 22)

for(…)yield循环建立了一个类型与原始集合相同的新集合。若是你从数组出发,那么你获得的是另外一个数组。若是你从数组缓冲出发,那么你在for(…)yield以后获得的也

是数组缓冲

结果包含yield以后的表达式的值,每次迭代对应一个。一般,当你遍历一个集合时,你只想处理那些知足特定条件的元素。这个需求能够经过守卫for中的if来实现。在这里咱们对每一个偶数元素翻倍,并丢掉奇数元素:

for (elem <- a if elem%==0) yield 2*elem

请留意结果是个新的集合,原始集合并无受到影响

一种等价方法

除上述以外,还有另外一种作法是

a.filter (_%2==0).map(2*_)

甚至

a.filter { _%2 == 0 } map {2*_ }

某些有着函数式编程经验的程序员倾向于使用filtermap而不是守卫yield,这不过是一种风格罢了与for循环所作的事彻底相同。你能够根据喜爱任意选择。

高效数组操做

考虑以下示例:给定一个整数的数组缓冲,咱们想要移除除第一个负数以外的全部负数传统的依次执行的解决方案会在遇到第一个负数时置一个标记,而后移除后续出现的负数元素

var first=true

var n=a.length

var i=0

while ( i<n ) {

if (a(i) >= 0)

i+=1

else{

if (first) {

first=false

i+=1

} else {

a.remove (i)

n-=1

}

}

}

但这个方案其实并不那么好:从数组缓冲中移除元素并不高效,把非负数值拷贝到前端要好得多。

首先收集须要保留的下标:

var first= true

val indexes=for (i <- 0 until a.length if first || a(i)>=0) yield {

if (a(i)<0)

first=false;

i

}

而后将元素移动到该去的位置,并截断尾端:

for(j <- 0 until indexes.length)

a(j)= a(indexes(j))

a.trimEnd (a.length -indexes.length)

这里的关键点是,拿到全部下标好过逐个处理

经常使用算法

求和与排序

有一种说法,很大比例的业务运算不过是在求和与排序。还好Scala有内建的函数来处理这些任务

Array(1,7,2, 9).sum // 19,对ArrayBuffer一样适用

要使用sum方法,元素类型必须是数值类型:要么是整型,要么是浮点数或者Biglnteger/BigDecimal。

同理,min和max输出数组或数组缓冲中最小和最大的元素。

ArraryBuffer("Mary", "had","a","little", "lamb").max // "little"

sorted方法将数组或数组缓冲排序并返回通过排序的数组或数组缓冲,这个过程并不会修改原始版本:

val b=ArrayBuffer(1,7,2, 9)

val bSorted=b.sorted(_ < _) // b没有被改变,bSorted是ArrayBuffer(1,2,7,9)

还能够提供一个比较函数,不过你须要用sortWith方法:

val bDescending=b.sorted(_ > _) // ArrayBuffer(9,7,2, 1)

能够直接对一个数组排序,但不能对数组缓冲排序

val a=Array(1,7,2,9)

scala.util. Sorting.quickSortIa(a) // a如今是Array(1,2,7,9)

关于num、max和quickSort方法,元素类型必须支持比较操做,这包括了数字、字符串以及其余带有Ordered特质的类型。

显示数组内容

最后,若是你想要显示数组数组缓冲的内容,能够用mkString方法,它容许你指定元素之间的分隔符。该方法的另外一个重载版本可让你指定前缀后缀。例如:

a.mkString("and") // "1 and 2 and 7 and 9"

a.mkString("<" , "," , ">") // "<1,2,7,9>"

和toString相比:

a.toString // " [I@85b8d",这里被调用的是Java的毫无心义的toString方法

b.toString // "ArrayBuffer(l,7,2, 9)",toString方法报告了类型,便于调试

解读Scaladoc

数组和数组缓冲有许多有用的方法,咱们能够经过浏览Scala文档来获取这些信息。对Array类的操做方法列在ArrayOps相关条目下。从技术上讲,在数组上应用这些操做以前,数组都会被转换成ArrayOps对象。

因为Scala的类型系统比java更丰富,在浏览Scala的文档时,你可能会遇到一些看上去很奇怪的语法。所幸,你并不须要理解类型系统的全部细节就能够完成不少有用

的工做。你能够把下表用作"解码指环"。

多维数组

和Java样,多维数组是经过数组的数组来实现的。举例来讲,Double的二维数组类型为:

Array[Array[Double]]

要构造这样一个数组,能够用ofDim方法

val matrix=Array.ofDim[Double](3,4) //三行,四列要访问其中的元素,使用两对圆括号:

matrix (row) (column) =42

你能够建立不规则的数组每一行的长度各不相同

val triangle=new ArraylArray [Int] (10)

for (i <- 0 until triangle.length)

triangle(i)=new Array[lnt] (i+1)

与Java互操做

因为Scala数组是用java数组实现的,你能够在Java和Scala之间来回传递。若是你调用接受返回java.utiI.List的Java方法,则固然能够在Scala代码中使用Java的ArrayList但那样作没什么意思。你彻底能够引入scala.collection.JavaConversions里的隐式转换方法。这样你就能够在代码中使用Scala缓冲,在调用Java方法时,这些对象会被自动包装成Java列表

举例来讲,java.lang.ProcessBuilder类有一个以List<String>为参数的构造器。如下是在Scala中调用它的写法:

import scala.collection.JavaConversions.bufferAsJavaList

import scala.collection.mutable.ArrayBuffer

val command = ArrayBuffer("ls", "-al", "/home/cay")

val pb = new ProcessBuilder(command) // Scala到Java的转换

Scala缓冲被包装成了一个实现了java.util.List接口的Java类的对象。反过来说,当Java方法返回java.util.List时,咱们可让它自动转换成一个Buffer

import scala.collection.JavaConversions.asScalaBuffer

import scala.collection.mutable.Buffer

val cmd: Buffer[String] = pb.command() // Java到Scala的转换

须要注意的是,不能使用ArrayBuffer——包装起来的对象仅能保证是个Buffer。若是Java方法返回一个包装过的Scala缓冲,那么隐式转换会将原始的对象解包出来。拿本例来讲,cmd == command。☆☆

若是,您认为阅读这篇博客让您有些收获,不妨点击一下右下角的【推荐】。
若是,您但愿更容易地发现个人新博客,不妨点击一下左下角的【关注我】。
若是,您对个人博客所讲述的内容有兴趣,请继续关注个人后续博客,我是【Sunddenly】。

本文版权归做者和博客园共有,欢迎转载,但未经做者赞成必须保留此段声明,且在文章页面明显位置给出原文链接,不然保留追究法律责任的权利。

相关文章
相关标签/搜索