1.高阶函数
-
概念
scala混合了面向对象和函数式的特性,通常将可以作为参数传递到方法中的表达式叫做函数.在函数式编程语言中,函数式"头等公民",高阶函数包含:作为值的函数,匿名函数,闭包,柯里化 等- 作为值得函数
可以像任何其他数据类型一样被传递和操作的函数,每当你想要给算法传入具体动作时这个特性就会变得非常有用
- 作为值得函数
定义函数时格式: val变量名 =(输入参数类型和个数) => 函数实现和返回值类型和个数
"=" 表示 将函数赋给一个变量
"=>"左面表示输入参数名称,类型和个数,右边表示函数的实现和返回值类型和参数个数
- 匿名函数
在scala中,你不需要给每一个函数命名,没有将函数赋给变量的函数叫匿名函数
由于Sala可以自动推出参数的类型,所有可以写的更精简一些
还可以在进行简写,最终形式
- 将方法转换成函数
在Scala中没方法和函数是不一样的,最本质的区别是函数可以像任何其他数据类型一样被传递和操作,函数是一个对象,自称自FunctionN
函数对象有 apply,curried,toString,tupled 这些方法
而方法不具有这些特性,如果想把方法转换成一个函数,可以用方法名跟上下划线的方式
-
柯里化
- 什么是柯里化
柯里化(Currying)指的是把原来接受多个参数的函数变换成接受一个参数的函数过程,并且返回接受余下的参数且返回结果为一个新函数的技术
- 什么是柯里化
- 举例
(1) 一个普通的非柯里化的函数定义,实现一个加法函数
scala> def plainOldSum(x:Int,y:Int)=x+y
plainOldSum: (x: Int, y: Int)Int
scala> plainOldSum(1,2)
res0: Int = 3
(2)使用“柯里化”技术来定义这个加法函数,原来函数使用一个参数列表,“柯里化”,把函数定义为多个参数列表:
scala> def curriedSum(x:Int)(y:Int)=x+y
curriedSum: (x: Int)(y: Int)Int
scala> curriedSum(1)(2)
res1: Int = 3
当你调用curriedSum (1)(2)时,实际上是依次调用两个普通函数(非柯里化函数),
第一次调用使用一个参数x,返回一个函数类型的值,
第二次使用参数y调用这个函数类型的值。
(3)使用下面两个分开的定义在模拟curriedSum柯里化函数:
首先定义第一个函数:
scala> def first(x:Int)=(y:Int)=>x+y
first: (x: Int)Int => Int
然后我们使用参数1调用这个函数来生成第二个函数:
scala> val second =first(1)
second: Int => Int = <function1>
scala> second(2)
res2: Int = 3
(4)使用curriedSum来定义second
scala> val onePlus=curriedSum(1)_
onePlus: Int => Int = <function1>
下划线“_” 作为第二参数列表的占位符, 这个定义的返回值为一个函数,当调用时会给调用的参数加一。
scala> onePlus(2)
res3: Int = 3
调用生成的函数,给函数传入参数,即可得到我们想要的结果。
- 总结
scala柯里化风格的使用可以简化主函数的复杂度,提高主函数的自闭性,提高功能上的可扩张性、灵活性。可以编写出更加抽象,功能化和高效的函数式代码。
-
闭包
- 什么是闭包
闭包是一个函数,返回值依赖于声明在函数外部的一个或多个变量。
- 什么是闭包
闭包通常来讲可以简单的认为是可以访问一个函数里面局部变量的另外一个函数。
- 例子
package cn.itcast.implic
/**
scala中的闭包
闭包是一个函数,返回值依赖于声明在函数外部的一个或多个变量。
*/
object ClosureDemo {
def main(args: Array[String]): Unit = {
val y=10
//函数在变量不处于其有效作用域时,还能够对变量进行访问
val add=(x:Int)=>{
x+y
}
//在add中有两个变量:x和y。其中的一个x是函数的形式参数,
//在add方法被调用时,x被赋予一个新的值。
// 然而,y不是形式参数,而是*变量
println(add(5)) // 结果15
}
}
-
隐式转换和隐式参数
- 隐式转换
Scala提供的隐式转换和隐式参数功能,是非常有特色的功能。是Java等编程语言所没有的功能。它可以允许你手动指定,将某种类型的对象转换成其他类型的对象或者是给一个类增加方法。通过这些功能,可以实现非常强大,而且特殊的功能
- 隐式转换
Scala的隐式转换,其实最核心的就是定义隐式转换方法,即implicit conversion function。定义的隐式转换方法,只要在编写的程序内引入,就会被Scala自动使用。Scala会根据隐式转换方法的签名,在程序中使用到隐式转换方法接收的参数类型定义的对象时,会自动将其传入隐式转换方法,转换为另外一种类型的对象并返回。这就是“隐式转换”。其中所有的隐式值和隐式方法必须放到object中.
然而使用Scala的隐式转换是有一定限制的,总结如下:
1)implicit关键字只能用来修饰方法,变量(参数)和伴随对象
2)隐式转换的方法(变量和伴随对象) 在当前范围内才有效.如果隐式转换不在当前范围内定义(例如:定义在另一个类或包含在某个对象中),那么必须通过import语句将其导入
- 隐式参数
所谓的隐式参数,指的是在函数或者方法中,定义一个用implicit修饰的参数,此时Scala会尝试找到一个指定的类型,用implicit修饰的对象,即隐式值并注入参数
Scala会在两个范围内查找:
1)当前作用域内可见的val或var的隐式变量;
2)一种是隐式参数类型的伴生对象内的隐式值;
- 隐式转换方法作用域与导入
- Scala默认会使用两种隐式转换:
一种是源类型,或者目标类型的伴生对象内的隐式转换函数;
一种是当前程序作用域内的可以用唯一标识表示的隐式转换方法
- 隐式转换的时机
- 当方法中的参数的类型与目标类型不一致时
- 当对象调用类中不存在的方法或成员时,编译器会自动将对象进行隐式转换
- 案例作证
(1)隐式转换案例一(让File类具备RichFile类中的read方法)
package cn.itcast.implic
import java.io.File
import scala.io.Source
/**
*/
object MyPredef{
//定义隐式转换方法
implicit def fileReadToRichFile(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 context=f.read()
println(context)
}
}
(2)隐式转换案例二(超人变身)
package cn.itcast.implic
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
}
}
(3)隐式转换案例三(一个类隐式转换成具有相同方法的多个类)
package cn.itcast.implic
class 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()
}
}
(4)隐式参数案例四(员工领取薪水)
package cn.itcast.implic
object Company{
//在object中定义隐式值 注意:同一类型的隐式值只允许出现一次,否则会报错
implicit val aaa="zhangsan"
implicit val bbb=10000.00
}
class Boss {
//定义一个用implicit修饰的参数,为隐式参数。类型为String
//Scala会在两个范围内查找:
//1.当前作用域内可见的val或var定义的隐式变量;
//2.是隐式参数类型的伴生对象内的隐式值
//注意参数匹配的类型 它需要的是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())
}