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特质的更多相关文章

  1. Scala进阶之路-Scala中的高级类型

    Scala进阶之路-Scala中的高级类型 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.类型(Type)与类(Class)的区别 在Java里,一直到jdk1.5之前,我们说 ...

  2. Scala进阶之路-Scala中的Ordered--Ordering

    Scala进阶之路-Scala中的Ordered--Ordering 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.   说道对象的比较,在Java中大家最熟悉不过的就是实现类本身实 ...

  3. Scala进阶之路-Spark独立模式(Standalone)集群部署

    Scala进阶之路-Spark独立模式(Standalone)集群部署 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们知道Hadoop解决了大数据的存储和计算,存储使用HDFS ...

  4. Scala进阶之路-并发编程模型Akka入门篇

    Scala进阶之路-并发编程模型Akka入门篇 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Akka Actor介绍 1>.Akka介绍 写并发程序很难.程序员不得不处 ...

  5. Scala进阶之路-Scala中的泛型介绍

    Scala进阶之路-Scala中的泛型介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 通俗的讲,比如需要定义一个函数,函数的参数可以接受任意类型.我们不可能一一列举所有的参数类 ...

  6. Scala进阶之路-Scala特征类与unapply反向抽取

    Scala进阶之路-Scala特征类与unapply反向抽取 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Scala特征类分析 1>.Unit 答:用于定义返回值类型, ...

  7. Scala进阶之路-面向对象编程之类的成员详解

    Scala进阶之路-面向对象编程之类的成员详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Scala中的object对象及apply方法 1>.scala 单例对象 ...

  8. Scala进阶之路-高级数据类型之集合的使用

    Scala进阶之路-高级数据类型之集合的使用 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala 的集合有三大类:序列 Seq.集 Set.映射 Map,所有的集合都扩展自 ...

  9. Scala进阶之路-Scala中的枚举用法案例展示

    Scala进阶之路-Scala中的枚举用法案例展示 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的枚举值和Java中的枚举值有点差别,不过使用起来也都差大同小异,我这 ...

随机推荐

  1. C#(静态String类)

    [转]http://blog.csdn.net/angelazy/article/details/8501776 C#中提供了比较全面的字符串处理方法,很多函数都进行了封装为我们的编程工作提供了很大的 ...

  2. java代码声明引用变量经验

    1.static只能修饰类的成员变量,不能修饰方法里的局部变量. 因为static是在类加载时候将成员变量存储进方法区的. 加载类的时候,是不去执行方法里的函数的.所以不会馆方法里的代码,自然就不会读 ...

  3. Arcgis10.5 python按属性分割图层,属性相同分为一个图层

    # coding=utf-8 """ Source code for potential gp tool to create outputs based on attri ...

  4. Log4net 配置输出文本, 按年月日分文件夹 z

    在项目中新建 “log4net.config” 文件 <?xml version="1.0" encoding="utf-8" ?> <con ...

  5. windows下androidNDK环境配置

    一:什么是NDK? NDK 提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库,并能自动将so 和java 应用一起打包成apk.这些工具对开发者的帮助是巨大的. NDK 集成了交叉编译器, ...

  6. python urlopen SSL: CERTIFICATE_VERIFY_FAILED

    1.使用ssl创建未经验证的上下文,在urlopen中传入上下文参数 import sslimport urllib2 context = ssl._create_unverified_context ...

  7. 阿里云服务器IIS启用HTTPS协议(转)

    https://www.cnblogs.com/randytech/p/7017188.html

  8. OpenCV学习(17) 细化算法(5)

    本章我们看下Pavlidis细化算法,参考资料http://www.imageprocessingplace.com/downloads_V3/root_downloads/tutorials/con ...

  9. [21] Mesh法线的生成算法

    // 生成顶点法线 bool YfCalculateVertexNormal ( void* pNormalsBuffer, Yuint normalStriding, Yuint normalPos ...

  10. Gh0st整理资料1

    题首 Gh0st是一款开源的远程控制软件.界面友好,性能高效.网上流传很多版本,比如红狼,饭客,败笔,大灰狼版本以及多如牛毛的个人修改的如外星人,Drat等个人修改版本.但内核都是基于Gh0st3.6 ...