9. Scala隐式转换和隐式值
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隐式转换和隐式值的更多相关文章
- Scala 中的隐式转换和隐式参数
隐式定义是指编译器为了修正类型错误而允许插入到程序中的定义. 举例: 正常情况下"120"/12显然会报错,因为 String 类并没有实现 / 这个方法,我们无法去决定 Stri ...
- Spark基础-scala学习(八、隐式转换与隐式参数)
大纲 隐式转换 使用隐式转换加强现有类型 导入隐式转换函数 隐式转换的发生时机 隐式参数 隐式转换 要实现隐式转换,只要程序可见的范围内定义隐式转换函数即可.Scala会自动使用隐式转换函数.隐式转换 ...
- Scala学习之路 (八)Scala的隐式转换和隐式参数
一.概念 Scala 2.10引入了一种叫做隐式类的新特性.隐式类指的是用implicit关键字修饰的类.在对应的作用域内,带有这个关键字的类的主构造函数可用于隐式转换. 隐式转换和隐式参数是Scal ...
- Scala隐式转换和隐式参数
隐式转换 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象或者是给一个类增加方法.通过这些功能, ...
- Scala基础:闭包、柯里化、隐式转换和隐式参数
闭包,和js中的闭包一样,返回值依赖于声明在函数外部的一个或多个变量,那么这个函数就是闭包函数. val i: Int = 20 //函数func的方法体中使用了在func外部定义的变量 那func就 ...
- 12、scala隐式转换与隐式参数
一.隐式转换 1.介绍 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象. 通过这些功能,可以实 ...
- Scala入门到精通——第十九节 隐式转换与隐式參数(二)
作者:摇摆少年梦 配套视频地址:http://www.xuetuwuyou.com/course/12 本节主要内容 隐式參数中的隐式转换 函数中隐式參数使用概要 隐式转换问题梳理 1. 隐式參数中的 ...
- 02.Scala高级特性:第6节 高阶函数;第7节 隐式转换和隐式参数
Scala高级特性 1. 课程目标 1.1. 目标一:深入理解高阶函数 1.2. 目标二:深入理解隐式转换 2. 高阶函数 2.1. 概念 Scala混合了面向对象和函数式的特 ...
- 大数据技术之_16_Scala学习_06_面向对象编程-高级+隐式转换和隐式值
第八章 面向对象编程-高级8.1 静态属性和静态方法8.1.1 静态属性-提出问题8.1.2 基本介绍8.1.3 伴生对象的快速入门8.1.4 伴生对象的小结8.1.5 最佳实践-使用伴生对象解决小孩 ...
- 15、Scala隐式转换和隐式参数
1.隐式转换 2.使用隐式转换加强现有类型 3.隐式转换函数的作用域与导入 4.隐式转换发生时机 5.隐式参数 1.隐式转换 要实现隐式转换,只要程序可见的范围内定义隐式转换函数即可.Scala会自动 ...
随机推荐
- Fiddler模拟响应
1 启用规则 2 不匹配的规则通过,不勾选会导致不匹配的请求失败 3 匹配url 4 响应信息,Fiddler内置了一些响应脚步,你也可以选择一个保护响应信息的文本文件
- Qt之Q_PROPERTY宏理解
在初学Qt的过程中,时不时地要通过F2快捷键来查看QT类的定义,发现类定义中有许多Q_PROPERTY的东西,比如最常用的QWidget的类定义: Qt中的Q_PROPERTY宏在Qt中是很常用的,那 ...
- android studio学习---菜单栏BUILD功能
项目构建: 1.构建项目 2.重构项目 3.签名打包
- SELECT语句基础
列的查询 语法1-1 基本的SELECT语句 SELECT <列名>,... FROM <表名>; 语法1-2 查询出表中所有的列 SELECT * FROM <表名& ...
- sass的入门(安装以及sublime中如何编译sass)
很久之前就听到了sass和less,但是因为人懒(默默吐槽下自己),就没有去了解相关的知识.但是身为一个前端人员不了解sass好像有点low low 的,所以现在开始接触使用sass了. 万事开头难, ...
- php观察者模式(observer pattern)
... <?php /* The observer pattern implements a one-too-many dependency between objects. The objec ...
- JMeter【第四篇】参数化
概念 参数化的原因,并不是网上说的真实模拟不同用户,真实反应服务器性能,而是: 数据唯一性(比如注册名不能一样) 避免数据库查询缓存 如何避免参数化: 去掉唯一性校验的约束 关闭数据库的查询缓存,my ...
- Python input() 函数
Python3.x 中 input() 函数接受一个标准输入数据,返回为 string 类型. Python2.x 中 input() 相等于 eval(raw_input(prompt)) ,用来获 ...
- DataOps Reading Notes
质量.效率.成本.安全,是运维工作核心四要素. AIOps 技术会涉及到数据收集方面的基础监控,服务监控和业务监控,甚至会涉及到与持续交付流水线的数据和状态整合(比如在软件发布的阶段会自动关闭某些监控 ...
- html--JavaScript之DOM (文档对象模型)
一.简介 当网页被加载时,浏览器会创建页面的文档对象模型(Document Object Model). HTML DOM 定义了用于 HTML 的一系列标准的对象,以及访问和处理 HTML 文档的标 ...