Scala高级特性

Scala高级特性



Scala混合了面向对象和函数式的特性,我们通常将可以作为参数传递到方法中的表达式叫做函数。在函数式编程语言中,函数是“头等公民”,高阶函数包含:作为值的函数、匿名函数、闭包、柯里化等等。

[if !supportLists]2.2.  [endif] 作为值的函数

可以像任何其他数据类型一样被传递和操作的函数,每当你想要给算法传入具体动作时这个特性就会变得非常有用。

定义函数时格式:val 变量名=(输入参数类型和个数)=>函数实现和返回值类型

“=”表示将函数赋给一个变量

“=>”左面表示输入参数名称、类型和个数,右边表示函数的实现和返回值类型

[if !supportLists]2.3.  [endif] 匿名函数

在Scala中,你不需要给每一个函数命名,没有将函数赋给变量的函数叫做匿名函数。

由于Scala可以自动推断出参数的类型,所有可以写的跟精简一些

还记得神奇的下划线吗?这才是终极方式

[if !supportLists]2.4.  [endif] 柯里化

[if !supportLists]2.4.1.  [endif] 什么是柯里化

柯里化(Currying)指的是把原来接受多个参数的函数变换成接受一个参数的函数过程,并且返回接受余下的参数且返回结果为一个新函数的技术。


[if !supportLists]2.4.2.  [endif] 例子

[if !supportLists](1)    [endif]一个普通的非柯里化的函数定义,实现一个加法函数:


scala> def plainOldSum(x:Int,y:Int)=x+y

plainOldSum: (x: Int, y: Int)Int


scala> plainOldSum(1,2)

res0: Int = 3


[if !supportLists](2)     [endif]使用“柯里化”技术来定义这个加法函数,原来函数使用一个参数列表,“柯里化”,把函数定义为多个参数列表:

scala> def curriedSum(x:Int)(y:Int)=x+y

curriedSum: (x: Int)(y: Int)Int


scala> curriedSum(1)(2) (等价于下面俩句)


val f1 = curriedSum(1) _

f1(2)





当你调用curriedSum (1)(2)时,实际上是依次调用两个普通函数(非柯里化函数),

第一次调用使用一个参数x,返回一个函数类型的值,

第二次使用参数y调用这个函数类型的值。


[if !supportLists](3)     [endif]使用下面两个分开的定义在模拟curriedSum柯里化函数:

首先定义第一个函数:

scala> def first(x:Int)=(y:Int)=>x+y

first: (x: Int)Int => Int


然后我们使用参数1调用这个函数来生成第二个函数:

scala> val second =first(1)

second: Int => Int =

scala> second(2)

res2: Int = 3



[if !supportLists](4)    [endif]使用curriedSum 来定义second

scala>  val onePlus=curriedSum(1)_

onePlus:

  Int => Int =


下划线“_” 作为第二参数列表的占位符, 这个定义的返回值为一个函数,当调用时会给调用的参数加一。


scala>  onePlus(2)

res3:  Int = 3

调用生成的函数,给函数传入参数,即可得到我们想要的结果。




[if !supportLists]2.4.3.  [endif] 总结

scala柯里化风格的使用可以简化主函数的复杂度,提高主函数的自闭性,提高功能上的可扩张性、灵活性。可以编写出更加抽象,功能化和高效的函数式代码。

[if !supportLists]2.5.  [endif]闭包

[if !supportLists]2.5.1.  [endif] 什么是闭包

闭包是一个函数,返回值依赖于声明在函数外部的一个或多个变量。闭包通常来讲可以简单的认为是可以访问不在当前作用域范围内的一个函数。

[if !supportLists]2.5.2.  [endif] 例子

package cn.itcast.closure

/**

  * scala中的闭包

  * 闭包是一个函数,返回值依赖于声明在函数外部的一个或多个变量。

  */

object ClosureDemo {


def main(args: Array[String]): Unit = {


val y=10


//变量y不处于其有效作用域时,函数还能够对变量进行访问


val add=(x:Int)=>{

          x+y

        }


//在add中有两个变量:x和y。其中的一个x是函数的形式参数,

    //在add方法被调用时,x被赋予一个新的值。

    // 然而,y不是形式参数,而是自由变量

    println(add(5)) // 结果15

  }

}







[if !supportLists]3.  [endif] 隐式转换和隐式参数

[if !supportLists]3.1.  [endif]  隐式转换

Scala提供的隐式转换和隐式参数功能,是非常有特色的功能。是Java等编程语言所没有的功能。它可以允许你手动指定,将某种类型的对象转换成其他类型的对象或者是给一个类增加方法。通过这些功能,可以实现非常强大、特殊的功能。

Scala的隐式转换,其实最核心的就是定义隐式转换方法,即implicit conversion function。定义的隐式转换方法,只要在编写的程序内引入,就会被Scala自动使用。Scala会根据隐式转换方法的签名,在程序中使用到隐式转换方法接收的参数类型定义的对象时,会自动将其传入隐式转换方法,转换为另外一种类型的对象并返回。这就是“隐式转换”。其中所有的隐式值和隐式方法必须放到object中。

然而使用Scala的隐式转换是有一定的限制的,总结如下:

[if !supportLists]Ø  [endif]implicit关键字只能用来修饰方法、变量(参数)。

[if !supportLists]Ø  [endif]隐式转换的方法在当前范围内才有效。如果隐式转换不在当前范围内定义(比如定义在另一个类中或包含在某个对象中),那么必须通过import语句将其导。

[if !supportLists]3.2.  [endif]  隐式参数

所谓的隐式参数,指的是在函数或者方法中,定义一个用implicit修饰的参数,此时Scala会尝试找到一个指定类型的,用implicit修饰的参数,即隐式值,并注入参数。

Scala会在两个范围内查找:

[if !supportLists]Ø  [endif]当前作用域内可见的val或var定义的隐式变量;

[if !supportLists]Ø  [endif]一种是隐式参数类型的伴生对象内的隐式值;



[if !supportLists]3.3.  [endif]  隐式转换方法作用域与导入

(1)Scala默认会使用两种隐式转换,一种是源类型或者目标类型的伴生对象内的隐式转换方法;一种是当前程序作用域内的可以用唯一标识符表示的隐式转换方法。

(2)如果隐式转换方法不在上述两种情况下的话,那么就必须手动使用import语法引入某个包下的隐式转换方法,比如import

test._。通常建议,仅仅在需要进行隐式转换的地方,用import导入隐式转换方法,这样可以缩小隐式转换方法的作用域,避免不需要的隐式转换。

[if !supportLists]3.4.  [endif]  隐式转换的时机

(1)当对象调用类中不存在的方法或成员时,编译器会自动将对象进行隐式转换

(2)当方法中的参数的类型与目标类型不一致时


[if !supportLists]3.5.  [endif]  隐式转换和隐式参数案例

[if !supportLists]① [endif]隐式转换案例一(让File类具备RichFile类中的read方法)

package cn.itcast.implic_demo

import java.io.File

import scala.io.Source

object MyPredef{

  //定义隐式转换方法


implicit def file2RichFile(file: File)=new RichFile(file)

}

class RichFile(val f:File) {


def read()=Source.fromFile(f).mkString

}

object RichFile{


def main(args: Array[String]) {


val f=new File("E://words.txt")


  //使用import导入隐式转换方法

    import MyPredef._

    //通过隐式转换,让File类具备了RichFile类中的方法


val content=f.read()

    println(conte

nt)

  }

}


[if !supportLists]② [endif]隐式转换案例二(超人变身)

package cn.itcast.implic_demo

class Man(val name:String)

class SuperMan(val name: String) {


def heat=print("超人打怪兽")

}

object SuperMan{


//隐式转换方法

  implicit def man2SuperMan(man:Man)=new SuperMan(man.name)


def main(args: Array[String]) {


val hero=new Man("hero")


//Man具备了SuperMan的方法

      hero.heat

  }

}

[if !supportLists]③ [endif]隐式转换案例三(一个类隐式转换成具有相同方法的多个类)

package cn.itcast.implic_democlass A(c:C) {


def readBook(): Unit ={

      println(

"A说:好书好书...")

    }

}

class B(c:C){


def readBook(): Unit ={

    println(

"B说:看不懂...")

  }


def writeBook(): Unit ={

    println(

"B说:不会写...")

  }

}

class C

object AB{


//创建一个类的2个类的隐式转换

  implicit def C2A(c:C)=new A(c)


implicit def C2B(c:C)=new B(c)

}

object B{


def main(args: Array[String]) {


//导包

    //1. import AB._ 会将AB类下的所有隐式转换导进来

    //2. import AB._C2A 只导入C类到A类的的隐式转换方法

    //3. import AB._C2B 只导入C类到B类的的隐式转换方法

    import AB._


val c=new C


//由于A类与B类中都有readBook(),只能导入其中一个,否则调用共同方法时代码报错

    //c.readBook()

    //C

类可以执行B类中的writeBook()


c.writeBook()

  }

}

[if !supportLineBreakNewLine]

[endif]

[if !supportLists]④ [endif]隐式参数案例四(员工领取薪水)

package cn.itcast.implic_demo

object Company{


//在object中定义隐式值    注意:同一类型的隐式值只允许出现一次,否则会报错

  implicit  valaaa="zhangsan"


implicit  valbbb=10000.00

}

class Boss {


//注意参数匹配的类型   它需要的是String类型的隐式值

  def callName()(implicit name:String):String={

    name+

" is coming !"


}


//定义一个用implicit修饰的参数

  //注意参数匹配的类型    它需要的是Double类型的隐式值

  def getMoney()(implicit money:Double):String={


" 当月薪水:"+money

  }

}

object Boss extends App{


//使用import导入定义好的隐式值,注意:必须先加载否则会报错

  import Company._


val boss =new Boss

  println(

boss.callName()+boss.getMoney())

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,194评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,058评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,780评论 0 346
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,388评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,430评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,764评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,907评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,679评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,122评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,459评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,605评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,270评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,867评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,734评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,961评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,297评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,472评论 2 348

推荐阅读更多精彩内容