【Scala】使用Option、Either和Try处理数据交互

原创 2015年07月09日 22:08:47

Scala数据交互

Scala使用一种函数式的方式来处理数据交互,包括入参及返回值。

  • Option: 解决null(空指针)问题
  • Either: 解决返回值不确定(返回两个值的其中一个)问题
  • Try: 解决函数可能会抛出异常问题

Option/Some/None的使用

Option实际上有3个类型:Option、Some和None,Some和None都是Option的子类型,Some和None。Option表示可选的值,它的返回类型是scala.Somescala.None。Some代表返回有效数据,None代表返回空值。

返回Option对象的函数

该函数以String对象作为输入,如果String对象被正确转换为Int对象,返回Sone[Int];否则返回None:

def toInt(s: String): Option[Int] = {
  try {
    Some(Integer.parseInt(s.trim))
  } catch {
    case e : Exception => None
  }
}

在Scala的集合类中使用Option

假设有一个字符串列表,我们希望得到该列表中所有的整数,通过将toInt方法传入List对象的map方法中,将列表元素转换成Some或None值:

scala> val bag = List("1", "2", "foo", "4", "bar")
bag: List[String] = List(1, 2, foo, 4, bar)

//通过flatten将原来的Option对象列表转换为整数列表
//由于Option是一个含有一个元素或0个元素(None)的集合,故能做出该转换
scala> bag.map(toInt).flatten
res1: List[Int] = List(1, 2, 4)

//通过flatMap实现相同的转换
scala> bag.flatMap(toInt)
res2: List[Int] = List(1, 2, 4)

//通过collect方法实现同样的功能
scala> bag.map(toInt).collect{case Some(i) => i}
res12: List[Int] = List(1, 2, 4)

Option的高阶函数

下面的函数是将一个Option[String]对象中字符串的长度打印出来,其中用到了Option的map方法和foreach方法:

def printContentLength(x: Option[String]): Unit = {
  x.map("length: " + _.length).foreach(println)
}

val value1 = Some("value1")
val value2 = None

printContentLength(value1) //length: 6
printContentLength(value2) //无打印

下面是将Option[String]对象中的字符串进行修剪并转换为大写字母:

def trimUpper(x: Option[String]): Option[String] = {
  x map (_.trim) filter (!_.isEmpty) map (_.toUpperCase)
}

val name1 = Some("  name  ")
val name2 = None
println(trimUpper(name1) ) //Some(NAME)
println(trimUpper(name2) ) //None

Try/Success/Failue的使用

在readfile类似的方法里,我们会使用了try catch语法。Scala2.10提供了Try来更优雅的实现这一功能。对于有可能抛出异常的操作。我们可以使用Try来包裹它,得到Try的子类Success或者Failure,如果计算成功,返回Success的实例,如果抛出异常,返回Failure并携带相关信息。

import scala.util.{Try, Success, Failure}

def divideBy(x: Int, y: Int): Try[Int] = {
  Try(x / y)
}

println(divideBy(1, 1).getOrElse(0)) // 1
println(divideBy(1, 0).getOrElse(0)) //0
divideBy(1, 1).foreach(println) // 1
divideBy(1, 0).foreach(println) // no print

divideBy(1, 0) match {
  case Success(i) => println(s"Success, value is: $i")
  case Failure(s) => println(s"Failed, message is: $s")
} 
//Failed, message is: java.lang.ArithmeticException: / by zero

readTextFile例子

如果该方法返回成功,将打印/etc/passwd文件的内容;如果出现异常,将打印错误信息,java.io.FileNotFoundException: Foo.bar (No such file or directory)

def readTextFile(filename: String): Try[List[String]] = {
  Try(Source.fromFile(filename).getLines.toList)
}

val filename = "/etc/passwd"
readTextFile(filename) match {
  case Success(lines) => lines.foreach(println)
  case Failure(f) => println(f)
  }
}

另外

  • Try有类似集合的操作 filter, flatMap, flatten, foreach, map
  • get, getOrElse, orElse方法
  • toOption可以转化为Option
  • recover,recoverWith,transform可以让你优雅地处理Success和Failure的结果

Either/Left/Right的使用

程序设计中经常会有这样的需求,一个函数(或方法)在传入不同参数时会返回不同的值。返回值是两个不相关的类型,分别为: Left 和 Right 。惯例中我们一般认为 Left 包含错误或无效值, Right包含正确或有效值。
在Scala 2.10之前,Either/Right/Left类和Try/Success/Failure类是相似的效果。

def divideBy2(x: Int, y: Int): Either[String, Int] = {
  if(y == 0) Left("Dude, can't divide by 0")
  else Right(x / y)
}

divideBy2(1, 0) match {
  case Left(s) => println("Answer: " + s)
  case Right(i) => println("Answer: " + i)
}
//print "Answer: Dude, can't divide by 0"

除了使用match case方式来获取数据,我们还可以分别使用 .right.get 和 .left.get 方法,当然你需要使用 .isRight 或 .isLeft 先判断一下。Left或Right类型也有 filter, flatMap, foreach, get, getOrElse, map 方法,它们还有toOption, toSeq 方法,分别返回一个Option或Seq 。

转载请注明作者Jason Ding及其出处
GitCafe博客主页(http://jasonding1354.gitcafe.io/)
Github博客主页(http://jasonding1354.github.io/)
CSDN博客(http://blog.csdn.net/jasonding1354)
简书主页(http://www.jianshu.com/users/2bd9b48f6ea8/latest_articles)
Google搜索jasonding1354进入我的博客主页

版权声明:本文为博主原创文章,未经博主允许不得转载。

67.scala编程思想笔记——使用Either

67.scala编程思想笔记——使用Either 欢迎转载,转载请标明出处: 源码下载连接请见第一篇笔记。          如果将每个错误都当做异常处理会使代码变得过于凌乱,scala引入了不相交并...
  • notbaron
  • notbaron
  • 2016年01月04日 22:03
  • 30157

学习Scala:使用try-catch表达式处理异常

本文节选自Martin Odersky,Lex Spoon和Bill Venners所著,Regular翻译的《Programming in Scala》的第七章。Scala是一种针对 JVM 将函数...

You can either disable the module by using --without-http-cache option, or install the OpenSSL libra

系统只安装了基本系统 先创建用于Nginx运行的用户和组: groupadd www useradd -g www www cd 到nginx源码根目录下,运行 ./configure \ ...
  • zoutuo
  • zoutuo
  • 2012年01月21日 00:02
  • 2824

【Scala】使用Option、Some、None,避免使用null

【Scala】使用Option、Some、None,避免使用null 字数1522 阅读5207 评论1 喜欢8 避免null使用 大多数语言都有一个特殊的关键字或者对象来表示一...

<点点滴滴>有关Scala的Option类的使用

//快学scala的第四章课后习题第一题 val prices=Map("xiaomi"->5,"meizu"->10,"huawei"->20) // 短短一行代码就可以把价格打九折返回给一...
  • AnQ17
  • AnQ17
  • 2016年03月23日 15:34
  • 234

【Scala】使用Option、Some、None,避免使用null

避免null使用大多数语言都有一个特殊的关键字或者对象来表示一个对象引用的是“无”,在Java,它是null。在Java 里,null 是一个关键字,不是一个对象,所以对它调用任何方法都是非法的。但是...

【Scala】使用Option、Some、None,避免使用null

避免null使用 大多数语言都有一个特殊的关键字或者对象来表示一个对象引用的是“无”,在Java,它是null。在Java 里,null 是一个关键字,不是一个对象,所以对它调用任何方法都是非法的。...

69.scala编程思想笔记——使用Try转换异常

69.scala编程思想笔记——使用Try转换异常 欢迎转载,转载请标明出处: 源码下载连接请见第一篇笔记。 为了进一步降低对异常进行管理的代码量,Try会铺货所有异常,并且产生一个作为结果的对象。...
  • notbaron
  • notbaron
  • 2016年01月04日 22:05
  • 30631

用scala,slick做数据交互层,play做mvc开发restful接口

介绍 scala为我们带来了函数式编程和面向对象编程,且scala代码被编译成class字节码,运行在JVM平台上。更为神奇的是scala可以无缝调用现有的众多java类库。scala比java有...

使用try-catch-finally处理异常

  • 2013年12月06日 10:46
  • 4KB
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:【Scala】使用Option、Either和Try处理数据交互
举报原因:
原因补充:

(最多只允许输入30个字)