9.1 隐式转换

  9.1.1 提出问题

      先看一个案例演示,引出隐式转换的实际需要=>指定某些数据类型的相互转化

object boke_demo01 {

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

    val num: Int = 3.5 //?错 高精度->低精度
println(num)
}
}

  9.1.2 隐式函数基本介绍

      隐式转换函数是以implicit关键字声明的带有单个参数的函数,这种函数将会自动应用,将值从一种类型转换为另一种类型

  9.1.3 隐式函数快速入门

      使用隐式函数可以优雅的解决数据类型转换

      案例演示

object boke_demo01 {

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

    //编写一个隐式函数转成 Double->Int 转换
//隐式函数应当在作用域才能生效
implicit def f1(d: Double): Int = { //底层 生成 f1$1
d.toInt
} implicit def f2(f: Float): Int = {
f.toInt
} //这里我们必须保证隐式函数的匹配只能是唯一的.
// implicit def f3(f1:Float): Int = {
// f1.toInt
// } val num: Int = 3.5 // 底层编译 f1$1(3.5)
val num2: Int = 4.5f //
println("num =" + num) }
}

      -反编译后的代码

      

  9.1.4 隐式转换的注意事项和细节

      1) 隐式转换函数的函数名可以是任意的,隐式函数与函数名无关,只与函数签名(函数参数类型和返回值类型)有关

      2) 隐式函数可以有多个(即:隐式函数列表),但是需要保证在当前环境下,只有一个隐式函数能被识别

9.2 隐式转换丰富类库功能

  9.2.1 快速入门案例

      使用隐式转换方式动态的给MySql类增加delete方法

  9.2.2 案例演示

object boke_demo01 {

  def main(args: Array[String]): Unit = {
//编写一个隐式函数,丰富mySQL功能
implicit def addDelete(msql: MySQL): DB = {
new DB
} //创建mysql对象
val mySQL = new MySQL
mySQL.insert()
mySQL.delete() // 编译器工作 分析 addDelete$1(mySQL).delete()
mySQL.update() }
} class MySQL {
def insert(): Unit = {
println("insert")
}
} class DB {
def delete(): Unit = {
println("delete")
} def update(): Unit = {
println("update")
}
}

9.3 隐式值

  9.3.1 基本介绍

      隐式值也叫隐式变量,将某个形参变量标记为implicit,所以编译器会在方法省略隐式参数的情况下去搜索作用域内的隐式值作为缺省参数

  9.3.2 快速入门案例

object boke_demo01 {
def main(args: Array[String]): Unit = { implicit val str1: String = "Jack" //这个就是隐式值 //implicit name: String :name就是隐式参数
def hello(implicit name: String): Unit = {
println(name + " hello")
} hello //底层 hello$1(str1); }
}

  9.3.3 一个案例说明隐式值,默认值,传值的优先级

//小结
//1. 当在程序中,同时有 隐式值,默认值,传值
//2. 编译器的优先级为 传值 > 隐式值 > 默认值
//3. 在隐式值匹配时,不能有二义性
//4. 如果三个 (隐式值,默认值,传值) 一个都没有,就会报错 object boke_demo01 {
def main(args: Array[String]): Unit = {
// 隐式变量(值)
// implicit val name: String = "Scala"
//implicit val name1: String = "World" //隐式参数
def hello(implicit content: String = "jack"): Unit = {
println("Hello " + content)
} //调用hello
hello //当同时有implicit 值和默认值,implicit 优先级高
def hello2(implicit content: String = "jack"): Unit = {
println("Hello2 " + content)
} //调用hello
hello2 //说明
//1. 当一个隐式参数匹配不到隐式值,仍然会使用默认值 implicit val name: Int = 10 def hello3(implicit content: String = "jack"): Unit = {
println("Hello3 " + content)
} //调用hello
hello3 // hello3 jack // //当没有隐式值,没有默认值,又没有传值,就会报错
// def hello4(implicit content: String ): Unit = {
// println("Hello4 " + content)
// } //调用hello
// hello4 // hello3 jack
}
}

9.4 隐式类

  9.4.1 基本介绍

      在Scala2.10后提供了隐式类,可以使用implicit声明类,隐式类非常强大,同样可以扩展类的功能,比前面使用隐式转换丰富类库功能更加的方便,在集合中隐式类会发挥重要作用

  9.4.2 隐式类使用有如下几个特点

      1) 其所带的构造参数有且只能有一个

      2) 隐式类必须被定义在“类”或“伴生对象”或“包对象”里,即隐式类不能是顶级的(top-level objects)

      3) 隐式类不能是case class(样例类)

      4) 作用域内不能有与之相同名称的标识符

  9.4.3 应用案例

      隐式类的案例,进一步认识隐式类

object boke_demo01 {

  def main(args: Array[String]): Unit = {
//DB会对应生成隐式类
//DB是一个隐式类, 当我们在该隐式类的作用域范围,创建MySQL实例
//该隐式类就会生效, 这个工作仍然编译器完成
//看底层..
implicit class DB(val m: MySQL) { //boke_demo01$DB$1
def addSuffix(): String = {
m + " scala"
}
} //创建一个MySQL实例
val mySQL = new MySQL
mySQL.sayOk()
mySQL.addSuffix() //研究 如何关联到 DB$2(mySQL).addSuffix(); implicit def f1(d: Double): Int = {
d.toInt
} def test(n1: Int): Unit = {
println("ok")
} test(10.1) }
} class DB {} class MySQL {
def sayOk(): Unit = {
println("sayOk")
}
}

9.5 隐式的转换时机

      1) 当方法中的参数的类型与目标类型不一致时,或者是赋值时

implicit def f1(d: Double): Int = {
d.toInt
} def test1(n1: Int): Unit = {
println("ok")
} test1(9.1)

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

9.6 隐式解析机制

      即编译器是如何查找到缺失信息的,解析具有以下两种规则:

      1) 首先会在当前代码作用域下查找隐式实体(隐式方法,隐式类,隐式对象)(一般是这种情况)

      2) 如果第一条规则查找隐式实体失败,会继续在隐式参数的类型的作用域里查找。类型的作用域是指与该类型相关联的全部伴生模块,一个隐式实体的类型T它的查找范围如下(第二种情况范围广且复杂在使用时,应当尽量避免出现)

        a) 如果 T 被定义为 T with A with B with C,那么 A,B,C 都是 T 的部分,在 T 的隐式解析过程中,它们的伴生对象都会被搜索

        b) 如果 T 是参数化类型,那么类型参数和与类型参数相关联的部分都算作 T 的部分,比如 List[String]的隐式搜索会搜索 List 的伴生对象和 String 的伴生对象

        c) 如果 T 是一个单例类型 p.T,即 T 是属于某个 p 对象内,那么这个 p 对象也会被搜索

        d) 如果 T 是个类型注入 S#T,那么 S 和 T 都会被搜索

9.7 在进行隐式转换时,需要遵守两个基本的前提

      1) 不能存在二义性

      2) 隐式操作不能嵌套使用  //比如:隐式函数转换

object boke_demo01 {
def main(args: Array[String]): Unit = { //1. 隐式转换不能有二义性
//2. 隐式转换不能嵌套使用 implicit def f1(d: Double): Int = {
d.toInt
//val num2:Int = 2.3 //底层 f1$1(2.3) //f1$1对应的就是f1,就会形成递归
} val num1: Int = 1.1
}
}

9. Scala隐式转换和隐式值的更多相关文章

  1. Scala 中的隐式转换和隐式参数

    隐式定义是指编译器为了修正类型错误而允许插入到程序中的定义. 举例: 正常情况下"120"/12显然会报错,因为 String 类并没有实现 / 这个方法,我们无法去决定 Stri ...

  2. Spark基础-scala学习(八、隐式转换与隐式参数)

    大纲 隐式转换 使用隐式转换加强现有类型 导入隐式转换函数 隐式转换的发生时机 隐式参数 隐式转换 要实现隐式转换,只要程序可见的范围内定义隐式转换函数即可.Scala会自动使用隐式转换函数.隐式转换 ...

  3. Scala学习之路 (八)Scala的隐式转换和隐式参数

    一.概念 Scala 2.10引入了一种叫做隐式类的新特性.隐式类指的是用implicit关键字修饰的类.在对应的作用域内,带有这个关键字的类的主构造函数可用于隐式转换. 隐式转换和隐式参数是Scal ...

  4. Scala隐式转换和隐式参数

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

  5. Scala基础:闭包、柯里化、隐式转换和隐式参数

    闭包,和js中的闭包一样,返回值依赖于声明在函数外部的一个或多个变量,那么这个函数就是闭包函数. val i: Int = 20 //函数func的方法体中使用了在func外部定义的变量 那func就 ...

  6. 12、scala隐式转换与隐式参数

    一.隐式转换 1.介绍 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象. 通过这些功能,可以实 ...

  7. Scala入门到精通——第十九节 隐式转换与隐式參数(二)

    作者:摇摆少年梦 配套视频地址:http://www.xuetuwuyou.com/course/12 本节主要内容 隐式參数中的隐式转换 函数中隐式參数使用概要 隐式转换问题梳理 1. 隐式參数中的 ...

  8. 02.Scala高级特性:第6节 高阶函数;第7节 隐式转换和隐式参数

    Scala高级特性 1.    课程目标 1.1.   目标一:深入理解高阶函数 1.2.   目标二:深入理解隐式转换 2.    高阶函数 2.1.   概念 Scala混合了面向对象和函数式的特 ...

  9. 大数据技术之_16_Scala学习_06_面向对象编程-高级+隐式转换和隐式值

    第八章 面向对象编程-高级8.1 静态属性和静态方法8.1.1 静态属性-提出问题8.1.2 基本介绍8.1.3 伴生对象的快速入门8.1.4 伴生对象的小结8.1.5 最佳实践-使用伴生对象解决小孩 ...

  10. 15、Scala隐式转换和隐式参数

    1.隐式转换 2.使用隐式转换加强现有类型 3.隐式转换函数的作用域与导入 4.隐式转换发生时机 5.隐式参数 1.隐式转换 要实现隐式转换,只要程序可见的范围内定义隐式转换函数即可.Scala会自动 ...

随机推荐

  1. Unity3D协程(转)

    这篇文章转自:http://blog.csdn.net/huang9012/article/details/38492937 协程介绍 在Unity中,协程(Coroutines)的形式是我最喜欢的功 ...

  2. Java操作ElasticSearch之创建客户端连接

    Java操作ElasticSearch之创建客户端连接 3 发布时间:『 2017-09-11 17:02』  博客类别:elasticsearch  阅读(3157) Java操作ElasticSe ...

  3. sqlalchemy相关操作(ORM)

    环境:python3.7,pycharm,mysql ORM(Object-Relational-Mapper) 对象关系映射(ORM)是一种允许您使用面向对象的范例从数据库查询和操作数据的技术,sq ...

  4. python遇到的模块集合

    J jieba jieba库是优秀的中文分词第三方库 ,它可以利用一个中文词库,确定汉字之间的关联概率将汉字间概率大的组成词组,形成分词结果,将中文文本通过分词获得单个的词语.jieba分词的三种模式 ...

  5. jquery-ui提供的拖拽方法

    项目当中遇到了任意拖动div标签的功能,找到了jqueryui提供的draggable的插件,这个插件可以实现任意的div的移动,也可以移动到整个屏幕或者在父元素的范围内进行移动. 插件的api    ...

  6. Jupyter Notebook---不需认证,与nginx搭配远程访问及下载

    如果默认启动,jupyter notebook需要token或是password登陆,且在nginx后面,下载会报403限制. Blocking Cross Origin request from h ...

  7. AjAX 异步通信

    <!DOCTYPE html> <html lang="en"> <head> <title>xmlhttprequest ajax ...

  8. Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.qingmu.mybaitsplus.mapper.UserMapper' available:

    java.lang.IllegalStateException: Failed to load ApplicationContext at org.springframework.test.conte ...

  9. JMeter【第四篇】参数化

    概念 参数化的原因,并不是网上说的真实模拟不同用户,真实反应服务器性能,而是: 数据唯一性(比如注册名不能一样) 避免数据库查询缓存 如何避免参数化: 去掉唯一性校验的约束 关闭数据库的查询缓存,my ...

  10. Game Publisher

    “Amazon Appstore https://developer.amazon.com/why-amazonApple Store https://developer.apple.com/prog ...