Scala中的空

Scala的有即Any,Scala的无是Null,null,Nil,Nothing,None,Unit.那么这几种空有什么区别呢?

一、Null&null

很多人一辈子都没有走出这个无。Null是一个Trait,你不能创建她它的实例。但是Scala在语言层面上存在一个Null的实例,那就是null。Java中的null意味着引用并没有指向任何对象。但存在一个悖论,一切都是对象,那没有对象是不是也是对象呢?Scala定义了一个类似于对象语义的Null,和一个值语义的null。这样面向对象在空引用的情况下完备了。如果你写了一个带有Null作为参数的对象,那么你传入的参数只能是null,或者指向Null的引用。

scala> def tryit(thing: Null): Unit = { println("That worked!"); }
tryit: (Null)Unit

scala> tryit("hey")
<console>:6: error: type mismatch;
 found   : java.lang.String("hey")
 required: Null
       tryit("hey")
             ^

scala> val someRef: String = null
someRef: String = null

scala> tryit(someRef)
<console>:7: error: type mismatch;
 found   : String
 required: Null
       tryit(someRef)
             ^

scala> tryit(null)
That worked!

scala> val nullRef: Null = null
nullRef: Null = null

scala> tryit(nullRef)
That worked!

第四行我们试图传入一个String,自然不能工作。第14行我们传入一个null引用,但是任然不能工作,为什么呢?因为null引用指向的是String类型。它可能在运行时是null,但是在编译时类型检查却不认同,编译器认为他是个String

二、Nil

Nil是一个继承List[Nothing]的对象,我们随后讨论Nothing。它就是一个空的列表,下面是一些使用实例:

scala> Nil
res4: Nil.type = List()

scala> Nil.length
res5: Int = 0

scala> Nil + "ABC"
res6: List[java.lang.String] = List(ABC)

scala> Nil + Nil
res7: List[object Nil] = List(List())

可以看出,Nil就是一个可以封装任何东西的空容器。它的长度为0。它并不是一无所有,它是一个容器,一个列表,只是没有存放内容而已。

三、Nothing

Nothing可能比较难理解。Nothing也是一个Trait,它继承自Any,而Any是整个Scala类型系统的根。Nothing是没有实例的,但它时任何对象的子类,他是List的子类,是String的子类,是Int的子类,是任何用户自定义类型的子类。

前面提到Nil是一个空的List[Nothing]。由于Nothing是任何类型的子类,那么Nil就可以当做是一个空的String List,空的Int List,甚至使Any List。Nothing比较适合用来定义基类容器。

scala> val emptyStringList: List[String] = List[Nothing]()
emptyStringList: List[String] = List()

scala> val emptyIntList: List[Int] = List[Nothing]()
emptyIntList: List[Int] = List()

scala> val emptyStringList: List[String] = List[Nothing]("abc")
<console>:4: error: type mismatch;
 found   : java.lang.String("abc")
 required: Nothing
       val emptyStringList: List[String] = List[Nothing]("abc")

第一行,我们将一个List[Nothing]赋值给一个List[String]。一个Nothing是一个String,因此是正确的。第四行,我们将一个List[Nothing]赋值给一个List[Int]。一个Nothing是一个Int,因此是正确的。Nothing是任何类型的子类,但是上面的List[Nothing]都不包含任何成员。当我们创建一个包含一个String的List[Nothing]的List,并把它赋值给List[String],会出现什么情况?它会失败,因为Nothing并不是任何类型的父类,并且也不存在Nothing的实例。任何Nothing的容器必然是空的,是Nil。

另一种Nothing的用法是作为不返回函数的返回值。因为Nothing没有任何实例,而函数的返回值必定是一个值,是一个对象,这样定义为Nothing为返回值的函数实际上不可能返回。

四、None

写Java程序的时候,经常会碰到没有有意义的东西可以返回,我们返回null。但返回null有一些问题,调用方必须检查返回值,不然会有NullPointerException的异常。这逼迫我们去check函数的返回值。还有一种解决办法是使用异常,但增加try/catch块,并不是明智的选择。

Scala内置一种解决办法。如果你想返回一个String,但可能有的时候得不到有意义的返回值,我们可以让函数返回Option[String]。

scala> def getAStringMaybe(num: Int): Option[String] = {
     |   if ( num >= 0 ) Some("A positive number!")
     |   else None // A number less than 0?  Impossible!
     | }

getAStringMaybe: (Int)Option[String]

scala> def printResult(num: Int) = {
     |   getAStringMaybe(num) match {
     |     case Some(str) => println(str)
     |     case None => println("No string!")
     |   }
     | }
printResult: (Int)Unit

scala> printResult(100)
A positive number!

scala> printResult(-50)
No string!

函数getAStringMaybe返回Option[String]。Option是一个抽象类,它有两个子类:Some和None。因此,初始化一个Option有两种方法。getAStringMaybe返回的只可能是Some[String]或None。Some和None又是Case Class,可以直接用来做模式匹配,很容易用来处理返回值。

类似地,Option[T]作为返回值,意味着调用者可能会收到Some[T]或None。

其实Option作为返回值,并没有比返回null好多少。代码里面充满了Option和模式匹配的代码,也不优雅,所以对待Option还是慎重。

五、Unit

Unit跟java的void一样,表示函数没有返回值。

scala> def doThreeTimes(fn: (Int) => Unit) = {
 | fn(1); fn(2); fn(3);
 | }
 doThreeTimes: ((Int) => Unit)Unit

scala> doThreeTimes(println)
 1
 2
 3
时间: 2024-12-24 05:40:39

Scala中的空的相关文章

Scala中的None,Nothing,Null,Nil

在scala中这四个类型名称很类似,作用确实完全不同的. None是一个object,是Option的子类型,定义如下 [java] view plaincopyprint? case object None extends Option[Nothing] { def isEmpty = true def get = throw new NoSuchElementException("None.get") } scala推荐在可能返回空的方法使用Option[X]作为返回类型.如果有值

Scala 中的函数式编程基础(三)

主要来自 Scala 语言发明人 Martin Odersky 教授的 Coursera 课程 <Functional Programming Principles in Scala>. 3. Data and Abstraction 3.1 Class Hierarchies 这一集字幕不同步-,-,听得有点费力! 类的概念和其他语言里面很相似,基类,子类,父类啥的叫法差不多.在 Scala 中,所有用户自定义的类都是另外一个类的子类,如果没有显式给定父类,java 里面默认继承 java.

scala中List的泛型分析以及::类和Nil对象

学习了scala中List的泛型分析以及::类和Nil对象,List是对scala的类型系统的一个非常好的使用,进一步巩固List的知识对于理解scala的类型系统是非常有价值的.本讲主要分析了List的泛型.::类和Nil对象. List有两个非常重要的子类,一个是::,一个是Nil,这两个子类都是case class.Nil表示一个空的列表,而::表达的是一个非空的列表. 例子如下: case object Nil extends List[Nothing]{ override def is

Scala中的集合:Iterator、BitSet、Set、Map、Stack、Vector、List、Array

 5.   util包 5.1.     架构 http://www.scala-lang.org/docu/files/collections-api/collections.html The following figure shows all collections in package scala.collection. These are all high-level abstract classes or traits, which generally have mutable

scala中的字段和成员方法的底层class字节文件分析

本文基于class字节码来分析在Scala语言中, 一个类中的字段和方法是如何实现的, 并且对比和java实现方式的区别. 首先看一段简单的源码: [java] view plain copy class FieldMethodTest{ private var i = 0 private val j = 0 def add() : Int = i + j } 这个类很简单, 其中有两个字段和一个方法: i字段被声明为var, 是可变的,类似于java中的普通变量: j字段被声明为val, 是不

在Scala中函数和方法有什么区别

方法可以作为一个表达式的一部分出现(调用函数并传参),但是方法(带参方法)不能作为最终的表达式, 但是函数可以作为最终的表达式出现: scala> //定义一个方法 scala> def m(x:Int) = 2*x m: (x: Int)Int scala> //定义一个函数 scala> val f = (x:Int) => 2*x f: Int => Int = <function1> scala> //方法不能作为最终表达式出现 scala&g

第3节 Scala中的模式匹配:1 - 5

7.    模式匹配和样例类 Scala有一个十分强大的模式匹配机制,可以应用到很多场合:如switch语句.类型检查等.并且Scala还提供了样例类,对模式匹配进行了优化,可以快速进行匹配. 7.1.   匹配字符串 package cn.itcast.cases import scala.util.Random object CaseDemo01 extends App{   val arr = Array("hadoop", "zookeeper", &quo

在Scala中免费验证

优锐课带你详细了解如何在Scala中实施免费的monad验证.抽丝剥茧,细说架构那些事! 由于业务数据的复杂性,已经在数据验证上花费了很多精力.在Scala中,提出了使用应用程序进行验证的方法,并被广泛认为是一种有效的方法.受应用验证和免费monad的思想启发,在本文中,我们介绍了一个monadic验证框架,该框架“免费”构建验证.我们将进一步讨论此方法,并通过示例代码演示实现. 在Scala中验证 当检测到错误时,可以以不同的形式找到验证.遇到第一个错误(或异常)时,验证可以立即返回:验证结果

scala中trait学习笔记

scala中提供的trait(特质)和Java中的Interface有很多相似之处.都可以持有方法的声明和属性,但是trait还有比interface强大的多的其他用法. 1. trait可以带有方法实现: 2. trait与interface一样,可以互相继承.但是trait可以继承自某个类,但是这种特质只能够混入父类的子类中,不能随意混入: 3. trait中可以在运行时动态调用方法. 下面举一个trait使用的例子. 首先定义一个虚类IntQueue和特质Logger abstract c