Scala进阶之路-Scala高级语法之隐式(implicit)详解
Scala进阶之路-Scala高级语法之隐式(implicit)详解
作者:尹正杰
版权声明:原创作品,谢绝转载!否则将追究法律责任。
我们调用别人的框架,发现少了一些方法,需要添加,但是让别人为你一个人添加是不现实的,因此很多很多时候需要我们自己动手。掌握implicit的用法是阅读Spark源码的基础,也是学习Scala其它的开源框架的关键,implicit可分为隐式参数,隐式转换类型以及隐式类三种类型。
一.Scala中的隐士参数
/*
@author :yinzhengjie
Blog:http://www.cnblogs.com/yinzhengjie/tag/Scala%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/
EMAIL:y1053419035@qq.com
*/
package cn.org.yinzhengjie.scalaImplicit object ScalaImplicit { /**
* 定义一个隐式值,编译器在查找隐式值的时候,不能出现歧义,也就是说,编译器在编译代码的时候,不能找到
* 两个类型一致的隐式值,不然编译器是会编译不通过的!
*/
implicit val default:Int = 500 /**
* @param Name : 此处我们将Name设置为隐士的参数
*/
def sayHello(implicit Name:String = "YinZhengJie") = {
println(s"I'm ${Name},I love Beijing !")
} /**
* @param x : 柯里化函数(Currying)的第一个参数
* @param y :柯里化函数(Currying)的第二个参数,其类型为一个隐式的参数哟
* @return : 返回值类型为Int
*/
def add(x:Int)(implicit y:Int):Int={
x + y
} /**
* 方法的参数如果有多个隐式参数的话,只需要使用一个implicit关键字即可,隐式参数列表必须放在方法的参数列表后面
*/
def sum(a:Int)(implicit b:Int,c:Int):Int={
a + b + c
} def main(args: Array[String]): Unit = { sayHello("yinzhengjie") /**
* sayHello方法参数是隐式参数,如果你没有给sayHello传递参数的话,编译器在编译的时候会自动从当前的上下文中
* 找一个隐式值(符合参数类型的隐式值),如果有则使用,如果没有就使用sayHello方法参数的默认值,指的注意的是,当一
* 各类中出现了多个implicit变量时,貌似默认值也不好使了!当前的类我就定了2个implicit变量就是一个很好的例子!
*/
// sayHello implicit val msg:String = "尹正杰"
sayHello /**
* 在调用柯里化函数(Currying)的时候,我们仅仅出入了第一个参数,第二个参数默认就是类中的成员变量用implicit修饰的default的值。
*/
val res = add(20)
println(s"res =====》 ${res}") /**
* 本来需要传入3个参数的,但是我们就传了一个,其他两个参数就会默认使用类中的成员变量用implicit修饰的default的值。
*/
val res2 = sum(10)
println(s"res2 =====》 ${res2}")
}
} /*
以上代码输出结果如下:
I'm yinzhengjie,I love Beijing !
I'm 尹正杰,I love Beijing !
res =====》 520
res2 =====》 1010
*/
二.Scala中的隐式类型转换
/*
@author :yinzhengjie
Blog:http://www.cnblogs.com/yinzhengjie/tag/Scala%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/
EMAIL:y1053419035@qq.com
*/
package cn.org.yinzhengjie.scalaImplicit object ScalaImplicit { /**
* 使用关键字implicit定义一个隐式方法。
*/
implicit def doubleToInt(double: Double):Int = {
println("====== 调用了方法:doubleToInt ======")
double.toInt
}
/**
* 使用关键字implicit定义一个隐式函数。
*/
implicit val doubleToInt2 = (double:Double) => {
println("====== 调用了函数:doubleToInt2 ======")
double.toInt
} def main(args: Array[String]): Unit = { /**
* year是一个Int类型的变量,但是赋值的确实一个浮点型数字,此刻编译器会在当前上下文中找一个
* 隐式转换,找一个能把浮点型变成Int类型的隐式转换的函数或者方法,如果有函数实现了这个功能就优先
* 调用函数的功能,若没有函数实现该功能就回去找是否有方法实现了该功能!如果函数或者方法都没有实现
* 该功能编译器是会报错的!
*/
val year:Int = 2018.7
println(s"year的结果是: ${year}")
}
} /*
以上代码输出结果如下:
====== 调用了函数:doubleToInt2 ======
year的结果是: 2018
*/
三.Scala中的隐式类
/*
@author :yinzhengjie
Blog:http://www.cnblogs.com/yinzhengjie/tag/Scala%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/
EMAIL:y1053419035@qq.com
*/
package cn.org.yinzhengjie.scalaImplicit import java.io.File import scala.io.Source object ScalaImplicit {
/**
* 定义隐式类,只能在静态对象(使用Object修饰)中使用
*/
implicit class FileRead(file:File){
def myRead() = Source.fromFile(file).mkString
} def main(args: Array[String]): Unit = {
val file = new File("D:\\10.Java\\IDE\\yhinzhengjieData\\Scala\\1.txt")
val lineCounts = file.myRead()
println(s"file文件对象里面的内容如下:\n${lineCounts}")
}
} /*
以上代码输出结果如下:
file文件对象里面的内容如下:
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
hello wolrd
*/
四.小试牛刀(封装File对象,新增lineCount方法,用来统计文件的行数,要求使用隐式方法)
/*
@author :yinzhengjie
Blog:http://www.cnblogs.com/yinzhengjie/tag/Scala%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/
EMAIL:y1053419035@qq.com
*/
package cn.org.yinzhengjie.scalaImplicit import java.io.{BufferedReader, File, FileReader} /**
* 自定义一个RichFile类,封装File类
*/
class RichFile(file:File){
/**
* 定义方法返回文件的记录行数
*/
def linesCount():Int={
val fileReader = new FileReader(file)
val bufferReader = new BufferedReader(fileReader) var sum = 0
try {
var line = bufferReader.readLine()
while (line != null) {
sum += 1
line = bufferReader.readLine()
}
} catch {
case _: Exception => sum
} finally {
fileReader.close()
bufferReader.close()
}
sum
}
} object ScalaImplicit {
/**
* 定义一个隐式方法,将File类型转换成RichFile类型。
*/
implicit def fileToRichFile(file: File) = {
new RichFile(file)
} def main(args: Array[String]): Unit = {
val file = new File("D:\\10.Java\\IDE\\yhinzhengjieData\\Scala\\1.txt") val lineCounts = file.linesCount()
println(s"file文件对象里面的行数是:${lineCounts}")
}
} /*
以上代码输出结果如下:
file文件对象里面的行数是:18
*/
Scala进阶之路-Scala高级语法之隐式(implicit)详解的更多相关文章
- Scala进阶之路-Scala中的高级类型
Scala进阶之路-Scala中的高级类型 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.类型(Type)与类(Class)的区别 在Java里,一直到jdk1.5之前,我们说 ...
- Scala进阶之路-Scala的基本语法
Scala进阶之路-Scala的基本语法 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.函数式编程初体验Spark-Shell之WordCount var arr=Array( ...
- Scala进阶之路-Scala中的Ordered--Ordering
Scala进阶之路-Scala中的Ordered--Ordering 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 说道对象的比较,在Java中大家最熟悉不过的就是实现类本身实 ...
- Scala进阶之路-Scala中的泛型介绍
Scala进阶之路-Scala中的泛型介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 通俗的讲,比如需要定义一个函数,函数的参数可以接受任意类型.我们不可能一一列举所有的参数类 ...
- Scala进阶之路-Scala特征类与unapply反向抽取
Scala进阶之路-Scala特征类与unapply反向抽取 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Scala特征类分析 1>.Unit 答:用于定义返回值类型, ...
- Scala进阶之路-Scala函数篇详解
Scala进阶之路-Scala函数篇详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.传值调用和传名调用 /* @author :yinzhengjie Blog:http: ...
- Scala进阶之路-Scala中的枚举用法案例展示
Scala进阶之路-Scala中的枚举用法案例展示 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的枚举值和Java中的枚举值有点差别,不过使用起来也都差大同小异,我这 ...
- Scala进阶之路-高级数据类型之集合的使用
Scala进阶之路-高级数据类型之集合的使用 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala 的集合有三大类:序列 Seq.集 Set.映射 Map,所有的集合都扩展自 ...
- Scala进阶之路-高级数据类型之数组的使用
Scala进阶之路-高级数据类型之数组的使用 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.数组的初始化方式 1>.长度不可变数组Array 注意:顾名思义,长度不可变数 ...
随机推荐
- Open Source CRM
https://www.odoo.com/zh_CN/page/crm 试用: https://none53.odoo.com/web#home https://none.mypscloud.com/ ...
- Appium学习笔记3_Genymotion模拟器安装
如果你已经配置好了安卓的运行环境,也配置好了自带的模拟器AVD,而且也launch了你的安卓模拟器,那么我相信你是不再愿意launch安卓模拟器第二次了,因为实在是太卡了(当然如果你电脑的配置够高,你 ...
- Typecho博客迁移
在新的机器上先搭建好一个新的Typecho博客,数据库名称和原博客相同(可以省不少事). 备份原来博客的usr目录. 备份mysql数据库,命令: mysqldump -uroot -p --all- ...
- PHP 4种输出的方式
<?php //测试用的数组 $info = array('11'=>'aaa', '22'=>'bbb', '33'=>'ccc'); //第一种,将整个数组作为一个对象输出 ...
- 3.27PSP及体会
首先,我还是第一次了解老师这种先喂鸡汤,再打鸡血的行为,大老板的出现让我有些措手不及,我 的 天 啊! 话说这周alpha版本实现,真的是好费脑筋,因为预定是4~6周的项目一周弄完,而且还是在拥有几个 ...
- 云时代的IT运维面临将会有哪些变化
导读 每一次IT系统的转型,运维系统和业务保障都是最艰难的部分.在当前企业IT系统向云架构转型的时刻,运维系统再一次面临着新的挑战.所以在数据中心运维的时候,运维人员应该注意哪些问题? 在云计算时代, ...
- UVALive5876-Writings on the Wall-KMP
有两段字符串,第一段的尾和第二段的头可能重合.问有多少种组合的可能. 需要理解一下next数组的意义. #include <cstdio> #include <cstring> ...
- virtualenv 模块
作用 安装虚拟环境 下载组件 pip3 install virtualenv 使用 命令行创建使用 """ 创建虚拟环境 """ virtu ...
- day30 hashlib模块
hashlib 提供摘要算法 最常见的就是MD5,当然一般来说MD5已经被足够了 不管算法多不一样,但是摘要的功能是不变的 对于相同的字符串进行摘要,使用同一个算法得到的值总是不变的 不同算法的话,会 ...
- day10 递归
死循环,因此递归必须要定义一个明确的结束条件 def calc(n): print(n) calc(n) calc(10) return 表示终止符号,最终会得出一个确切的返回值,且可以赋值 def ...