刚接触Scala时就很奇怪, 为什么Java已经有null了,却偏偏还要弄出个None

后来依然我行我素在Scala里使用null, 结果就是经常被NullPointerException折磨得阴魂不散...于是终于领悟.

WHY?

举个例子:

假设有一个简单的对象A, A类里有2个Int属性, a和b

case class A(val a : Int, val b : Int)

  

然后有这么一个map

val map1 = Map(1 -> A(111, 1111), 2 -> A(222, 2222), 3 -> A(333, 3333))

  

然后有这么一段业务逻辑需要从这个map中取出其中2个A,并且拿他们的a属性相加

val key1 :Int = //其他地方给我的值,我也不知道具体是多少
val key2 :Int = //其他地方给我的值,我也不知道具体是多少
println(map1.getOrElse(key1, null).a + map1.getOrElse(key2, null).a) //111 + 报 NullPointerException

  

那么恭喜你, 遇到了NullPointerException. 为什么? 假设别人给你的key1=1, key2=4, 那么map1.getOrElse(key2, null).a 将会报出NullPointerException

不仅这样, 你在报错输出中得到的抛出异常的代码就是上面第3行println这句, 单从日志来看你根本无法知道别人给你有问题的是key1还是key2

所以在Java中, 一个对象的引用, 已经被默认为可能会是null了, 但有时候敲代码时会极度容易犯错, 同时就会信心满满的认为 "这个对象怎么可能是null呢?",

可是一旦真的是null, 只能庆幸这个报错不是出现在生产环境中了.

HOW?

那么我们如何在scala里避免使用null?

很简单, scala提供的Option,和None就是个非常好的设计! 有多好? 代码撸起

像scala List里, 如果我要查找一个list里面符合要求的元素, 不一定能找得到, 所以返回的是一个Option[元素类型]

    val list = List(1 ,2, 3, 4)[Int]
val found : Option[Int] = list.find(_ > 5)
found match {
case None => //由你决定找不到怎么办
case Some(value) =>//由你决定找到想要的value值怎么办
}

  

所以对于之前的需求,一段业务逻辑需要从这个map中取出其中2个A,并且拿他们的a属性相加

    //可以先用get, get 出来的东西是个 Option[A], 再做判断
var finalResult = 0 //0默认值
(map1.get(key1), map2.get(key2)) match {
case (Some(value1), Some(value2)) =>
//都找到了, 可以愉快的相加了
finalResult = value1.a + value2.a
case (None, Some(value2)) =>
//第一个找不到, 第二个找到了怎么办的代码
case (Some(value1), None) =>
//第一个找到了, 第二个找不到怎么办的代码
case _=>
//其他情况
}

  

scala的Map可以使用get方法获取对象, 获取出来的是一个 Option[对象类型], 为什么不像java那样直接取出null? 这里就是因为要考虑到取不出来东西的情况.

所以Option存在的意义, 就是为了在代码中注明, 让大家一看就知道: "这个东西可能是空的! 你们用的时候给我小心点" 这样的暗示.

有了这个暗示, 你可就不能随意取出option里面的东西了, 警醒你每次使用, 都要先判断. isEmpty 或是 nonEmpty

当然更优雅的写法就像上面那样, 使用匹配模式, 在出现多种情况之间选择做什么事情.

Option还提供了其他很骚的用法. 你甚至可以把Option当成是一个只能容纳1个元素的容器来看待.

假设又有那么一个需求, 我有一个Option[A], 我不确定这个Option是否==None, 但我想要尝试取出里面的A对象的a属性, 如果没有的话就给我个0吧, 那么代码可以这么写

    val aOpt : Option[A] = //我不确定他是None, 还是Some(A)反正别人给我的

    //就这么简单粗暴
val myVal1 :Int = if(aOpt.isEmpty) 0 else aOpt.get.a //也可以先把Option[A]map成Option[Int](这里是Int,因为是A的a属性的类型), 再使用getOrElse 取不到就给个自己定的默认值
val myVal2 :Int = aOpt.map(aObj => aObj.a).getOrElse(0)

  

还有另一个需求, 我想知道这个Option[A] 里的A对象的a属性是否等于100

val has100 :Boolean = aOpt.exists(aObj => aObj.a == 100)

  

Final

已经有不少文章吐槽null的设计是多么糟糕: 传送门请看:http://www.php230.com/weixin1446428231.html

既然scala已经考虑到并提供优雅代码的解决方式. 那以后可以少在scala里给用null了.

Scala不使用null 而使用Option,None,Some的好处的更多相关文章

  1. Scala中的"null" 和“_”来初始化对象

    Alternatives Use null as a last resort. As already mentioned, Option replaces most usages of null. I ...

  2. Scala 学习笔记之集合(7) Option

    object CollectionDemo8 { def main(args: Array[String]): Unit = { //Option集合的使用,可以用来安全的判断null或非null,放 ...

  3. 转载:【Scala】使用Option、Some、None,避免使用null

    转载自Jason DingGitCafe博客主页(http://jasonding1354.gitcafe.io/)Github博客主页(http://jasonding1354.github.io/ ...

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

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

  5. scala Option 里的 orNull orElse getOrElse 区别和使用

    Scala Option Scala Option类型用来表示一个值是可选的(有值或无值) Option[T] 是一个类型为 T 的可选值的容器: 如果值存在, Option[T] 就是一个 Some ...

  6. scala(一)Nothing、Null、Unit、None 、null 、Nil理解

    相对于java的类型系统,scala无疑要复杂的多!也正是这复杂多变的类型系统才让OOP和FP完美的融合在了一起! Nothing: 如果直接在scala-library中搜索Nothing的话是找不 ...

  7. Scala Option类型

    转载自: Scala 初学者指南, 这里有一系列很棒的文章 类型 Option 可能你已经见过它在 Map API 中的使用:在实现自己的提取器时,我们也用过它, 然而,它还需要更多的解释. 你可能会 ...

  8. scala中option、None、some对象

    转载:http://www.jianshu.com/p/95896d06a94d 1.option类型避免对象是空值,造成空指针异常. 2.None对象表示null,在没有对象返回时使用,some在有 ...

  9. Scala 技术笔记之 Option Some None

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

随机推荐

  1. Django2.1集成xadmin管理后台所遇到的错误集锦,解决填坑

    django默认是有一个admin的后台管理模块,但是丑,功能也不齐全,但是大神给我们已经集成好了xadmin后台,我们拿来用即可,但是呢,django已经升级到2.1版本了,xadmin貌似跟不上节 ...

  2. Python之路-基础数据类型之字符串

    字符串类型 字符串是不可变的数据类型 索引(下标) 我们在日常生活中会遇到很多类似的情况,例如吃饭排队叫号,在学校时会有学号,工作时会有工号,这些就是一种能保证唯一准确的手段,在计算机中也是一样,它就 ...

  3. notification 使用的基本方法

    当某个应用程序希望向用户发出一些提示信息,而应用程序又不在前台,可以借助Notification来实现.发出一条通知后,手机最上方额通知栏会显示一个图标,下来状态栏以后可以看到详细内容. 一.通知的基 ...

  4. SHIWEITI

    //Wannafly挑战赛19(牛客网) //A 队列Q #include <iostream> #include <cstdio> #include <cstring& ...

  5. hive实现根据用户分组,按用户记录求上下两条记录的时间差

    在mysql,数据如下:#查询某一用户该日抽奖时间 select draw_time from user_draw_log where user_id = 1 and draw_date='2016- ...

  6. jvm探秘之三:GC初步

    GC即垃圾收集器,虚拟机的必要组成部分. 不过这里说当然是,hotspot虚拟机(jvm的主要版本)的GC机制,前面说过了jvm的组成部分,那么想当然GC只需要负责方法区和堆就好了,虚拟机栈.本地方法 ...

  7. SJTU 1077 加分二叉树

    http://acm.sjtu.edu.cn/OnlineJudge/problem/1077 题意: 设一个n个节点的二叉树tree的中序遍历为(l,2,3,…,n),其中数字1,2,3…,n为节点 ...

  8. windows下,cmd 运行 python 脚本,选中文字就停止运行了【已解决】

    参考资料: https://jingyan.baidu.com/article/ce09321bb95dda2bff858f26.html 问题原因: cmd 里面,快速编辑模式会暂停程序 解决步骤: ...

  9. Analyze Program Runtime Stack

    Introduce: Process Explorer is an advanced process management utility that picks up where Task Manag ...

  10. linux下编译运行TIGL Viewer步骤

    linux下编译运行TIGL Viewer步骤(仅为了正确编译安装的话直接跳到步骤3) 1. linux发行版选择:由于linux发行版众多,不同版本包含的库版本可能存在差别,因此需要选择正确的版本. ...