Scala进阶之App特质
App特质的作用
App特质的作用那就是延迟初始化,从代码上看它继承自DelayedInit,里面有个delayedInit方法
trait App extends DelayedInit
DelayedInit特质里定义了延迟初始化方法:
def delayedInit(x: => Unit): Unit
开发者可以直接在初始化块里写逻辑,(这里指的是 extends App{//todo}里的//todo代码)
然后编译器会把这段初始化代码块里的逻辑封装成一个函数对象(是(() => Unit)类型)
override def delayedInit(body: => Unit) {
initCode += (() => body)
}
缓存起来(并没有运行),然后放到一个集合(ListBuffer)中,之后在main方法里一行一行调用并执行,
所以只有在执行到main方法的时候才会触发,从而达到延迟初始化的效果。
def main(args: Array[String]) = {
...
for (proc <- initCode) proc()
...
}
不过在实际开发中,经常需要一开始就初始化,不然会报错,如空指针异常,真正使用App的机会个人感觉都不多
object AppInternals extends App{
def testApp{
val c =new C
println("3. Hello spark")
}
}
trait Helper extends DelayedInit{
def delayedInit(body: => Unit)={
println("1. dummy text, printed before inititalization of C")
body //evaluates the initialization code of C
}
}
class C extends Helper{
println("2. this is the initialization code of C")
}
object AppTest {
def main(args: Array[String]) {
AppInternals.testApp
}
}
运行结果:
1. dummy text, printed before inititalization of C 2. this is the initialization code of C
3. Hello spark
问题: 是怎么把封装的初始化代码块传给delayedInit(body: => Unit)的?
用反编译工具jd-gui.exe把上面生成的.class反编译出来,可以看到多出了好多个类,
其中主要的有
class AppInternals, delayedInitbody,AppInternals, C, delayedInitbody,Helper,Helperclass
delayedInit$body出现两次,一次出现在AppInternals中,一次出现在C中, 它里面都有一个方法
apply()
分别对应的是
public final Object apply()//AppInternals
{
this.$outer.c_$eq(new C()); Predef..MODULE$.println("Hello Spark"); return BoxedUnit.UNIT;
} public final Object apply() { //C
Predef..MODULE$.println("this is the initialization code of C"); return BoxedUnit.UNIT;
}
从第一个apply中可以看出它已经把例子代码中的代码块封装起来了
object AppInternals extends App{
val c = new C
println(“Hello Spark”)
}
从第二个apply可以看出它把C中的
class C extends Helper {
println(“this is the initialization code of C”)
}
代码块封装起来了
最后在class Helperclass里面用publicstaticvoiddelayedInit(Helperthis, Function0 body)
{
Predef..MODULE$.println(“dummy text, printed before initialization of C”);
body.applymcVsp();
}
这就是初始化代码块的执行顺序,在main方法里调用delayedInit方法,先是执行
Predef..MODULE$.println(“dummy text, printed before initialization of C”);
之后调用 body.applymcVsp(); body.applymcVsp()里先调用 this.outer.c_$eq(new C()); Predef..MODULE$.println(“Hello Spark”);
所以是先执行 this.outer.c_$eq(new C());
new C执行时就执行语句
println(“this is the initialization code of C”)
接着执行
println(“Hello Spark”);
所以输出的结果就是这样的
dummy text, printed before initialization of C
this is the initialization code of C
Hello Spark
补充内容是自己猜测的居多,不保证正确,个人觉得背后的原理其实不用深究,知道有这么回事就行了。
Scala进阶之App特质的更多相关文章
- Scala进阶之路-Scala中的高级类型
Scala进阶之路-Scala中的高级类型 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.类型(Type)与类(Class)的区别 在Java里,一直到jdk1.5之前,我们说 ...
- Scala进阶之路-Scala中的Ordered--Ordering
Scala进阶之路-Scala中的Ordered--Ordering 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 说道对象的比较,在Java中大家最熟悉不过的就是实现类本身实 ...
- Scala进阶之路-Spark独立模式(Standalone)集群部署
Scala进阶之路-Spark独立模式(Standalone)集群部署 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们知道Hadoop解决了大数据的存储和计算,存储使用HDFS ...
- Scala进阶之路-并发编程模型Akka入门篇
Scala进阶之路-并发编程模型Akka入门篇 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Akka Actor介绍 1>.Akka介绍 写并发程序很难.程序员不得不处 ...
- Scala进阶之路-Scala中的泛型介绍
Scala进阶之路-Scala中的泛型介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 通俗的讲,比如需要定义一个函数,函数的参数可以接受任意类型.我们不可能一一列举所有的参数类 ...
- Scala进阶之路-Scala特征类与unapply反向抽取
Scala进阶之路-Scala特征类与unapply反向抽取 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Scala特征类分析 1>.Unit 答:用于定义返回值类型, ...
- Scala进阶之路-面向对象编程之类的成员详解
Scala进阶之路-面向对象编程之类的成员详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Scala中的object对象及apply方法 1>.scala 单例对象 ...
- Scala进阶之路-高级数据类型之集合的使用
Scala进阶之路-高级数据类型之集合的使用 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala 的集合有三大类:序列 Seq.集 Set.映射 Map,所有的集合都扩展自 ...
- Scala进阶之路-Scala中的枚举用法案例展示
Scala进阶之路-Scala中的枚举用法案例展示 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的枚举值和Java中的枚举值有点差别,不过使用起来也都差大同小异,我这 ...
随机推荐
- java反射知识点总结
一.java反射基础 1.1 什么叫java反射? 答:程序运行期间,动态的获取类的基本信息.比如:创建对象,调用类的方法,获得类的基本结构.这样给程序设计提供了很大的灵活性.个人总结就是:根据动态需 ...
- nginx做TCP代理实现群集
nginx做TCP代理实现群集 nginx从版本1.9开始,既能做HTTP代理,又能做TCP代理,这就非常完美了. 配置nginx.conf. 为了简单起见,笔者故意去掉了HTTP代理配置部分,只保留 ...
- RobotFramework自动化3-搜索案例
前言 RF系列主要以案例为主,关键字不会的可以多按按F5,里面都有很详细的介绍,要是纯翻译的话,就没太大意义了,因为小编本来英语就很差哦! 前面selenium第八篇介绍过定位一组搜索结果,是拿百度搜 ...
- 《MacTalk·人生元编程》
<MacTalk·人生元编程> 基本信息 <MacTalk·人生元编程> 基本信息 作者:池建强 出版社:人民邮电出版社 ISBN:9787115342232 上架时间:201 ...
- 《阿里巴巴JAVA开发手册》里面写超过三张表禁止join这是为什么?
分库分页.应用里做join 多表join性能很差 参考: 1.https://www.zhihu.com/question/56236190
- MySQL对索引的使用
什么是索引 使用索引可快速访问数据库表中的特定信息.索引是对数据库表中一列或多列的值进行排序的一种结构,例如 order 表的订单号(orderNum)列.如果要按订单号查找特定订单,与必须搜索表中的 ...
- pat 1060 比较科学计数法
trick: 1.前导0 如:000001,000.000001 2.出现0时也要按照科学计数法输出 e.g. 4 00000.00000 0001 NO 0.0000*10^0 0.1*10^1 3 ...
- Vue路由history模式踩坑记录:nginx配置解决404问题
问题背景: vue-router 默认是hash模式,使用url的hash来模拟一个完整的url,当url改变的时候,页面不会重新加载.但是如果我们不想hash这种以#号结尾的路径时候的话,我们可以使 ...
- linux系统用户下的crontab任务不执行问题处理
需求:需要每一天对数据库做一个备份,oracle数据库,linux系统. 备份命令采用最简单的导出\导入. 首先确认服务器是否开启任务计划服务,只有root用户才能对crond服务进行开启和关闭 [r ...
- ystem.Windows.Forms.SplitContainer : ContainerControl, ISupportInitialize
#region 程序集 System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 ...