1. Scala 的 反射

关于 Scala 反射的具体内容,可以参考官方文档:https://docs.scala-lang.org/overviews/reflection/overview.html

这篇文章写一点自己的理解:

  • 由于 Scala 编译出来的内容是与 Java 相同的字节码文件,所以可以使用 Java 反射的相关方法来实现 Scala 代码的反射。
  • Scala 自己写了一套基于 Scala 的反射,具体的实现在 scala.reflect 这个 package 下面。
  • 这篇文章主要介绍,反射 Scala 中的 class 和 object 类的方法。

先看基础代码:

package com.personal

object ProvisioningApp {
val strInObj = "123"
def sayHello(): Unit = println("say hello")
def sayHello2(from: String, to: String): Unit = println(from + " say hello, " + to)
} class ProvisioningApp {
val strInClazz = "234"
def sayGoodbye(): Unit = println("say goodbye")
}

2. Java Style

2.1 使用 Java 的方式反射 Scala class

和反射 Java 的 class 步骤完全一致,所以不赘述,直接贴代码:

  test("Should reflect Scala class in Java style") {
val app = new ProvisioningApp
val field = classOf[ProvisioningApp].getDeclaredField("strInClazz")
field.setAccessible(true)
field.set(app, "789")
app.strInClazz shouldBe "789" val method = classOf[ProvisioningApp].getDeclaredMethod("sayGoodbye")
method.setAccessible(true)
method.invoke(app)
}

2.2 使用 Java 的方式反射 Scala object 

Scala 中的 object,称之为 “伴生类”,想要反射获取它的类或者方法,首先要知道它编译之后是个什么东西:

通过 jd-gui,我们知道了 Scala object 的具体实现:

  • 编译之后的类名是 "类名+$" 形式
  • 属性的名字,有时会和在代码中定义的不同(在这个例子里面没有显现,具体原因我还不知道,比如这个 "strInObj", 有时候这个类名会变成 “$com.$personal.$$strInObj” 这样)
  • 可以发现,这是一个使用静态代码块模式的单例,详见 https://www.cnblogs.com/jing-an-feng-shao/p/7501617.html
  • 因此,暂时没有找到使用 Java 方式反射 Scala object 的方法

3. Scala style

3.1 使用 Scala 方式反射 Scala class

步骤如下:

  1. 通过 universe 和 classLoader 找到 JavaMirror
  2. 通过 JavaMirror 和 实例化对象 获取 InstanceMirror
  3. 通过 universe 获取目标类的 TypeTag
  4. 通过 TypeTag 获取目标类的 TermSymbol or MethodSymbol
  5. 获取 FieldMirror or MethodMirror
  6. 反射执行
  test("Should reflect Scala class in Scala type") {
import scala.reflect.runtime.universe val app = new ProvisioningApp()
// JavaMirror
val classMirror = universe.runtimeMirror(getClass.getClassLoader)
// InstanceMirror
val instanceMirror = classMirror.reflect(app)
// TypeTag
val typeTag = universe.typeOf[ProvisioningApp] // TermSymbol
val strInClazzSymbol = typeTag.decl(universe.TermName("strInClazz")).asTerm
val fieldMirror = instanceMirror.reflectField(strInClazzSymbol)
fieldMirror.set("789")
app.strInClazz shouldBe "789" // MethodSymbol
val sayGoodbyeSymbol = typeTag.decl(universe.TermName("sayGoodbye")).asMethod
// MethodMirror and invoke action
val result = instanceMirror.reflectMethod(sayGoodbyeSymbol)() // No input parameters here
result shouldBe "Goodbye"
}

3.2 使用 Scala 方式反射 Scala object

与 Scala class 不同,反射 Scala object 核心是通过 staticModule 获取 ModuleMirror:

  test("Should reflect Scala object in Scala style") {
import scala.reflect.runtime.universe // JavaMirror
val classMirror = universe.runtimeMirror(getClass.getClassLoader)
// The ModuleSymbol for object
val staticMirror = classMirror.staticModule("com.personal.ProvisioningApp")
// ModuleMirror can be used to create instances of the class
val moduleMirror = classMirror.reflectModule(staticMirror)
// ObjectMirror can be used to reflect the members of the object
val objectMirror = classMirror.reflect(moduleMirror.instance)
// TermSymbol represents val, var, def and object declarations
val strInObjSymbol = moduleMirror.symbol.typeSignature.member(universe.TermName("strInObj")).asTerm
// FieldMirror can be used to get and set the value of the field
val fieldMirror = objectMirror.reflectField(strInObjSymbol)
fieldMirror.set("789")
ProvisioningApp.strInObj shouldBe "789" val sayHelloSymbol = moduleMirror.symbol.typeSignature.member(universe.TermName("sayHello")).asMethod
val sayHelloSymbol2 = moduleMirror.symbol.typeSignature.member(universe.TermName("sayHello2")).asMethod
// MethodMirror and invoke action
objectMirror.reflectMethod(sayHelloSymbol)()
objectMirror.reflectMethod(sayHelloSymbol2)("Sai", "Gerrard") // Pass the input parameters one by one
}

Scala进阶(1)—— 反射 object 和 class的更多相关文章

  1. Scala进阶之路-反射(reflect)技术详解

    Scala进阶之路-反射(reflect)技术详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的反射技术和Java反射用法类似,我这里就不一一介绍反射是啥了,如果对 ...

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

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

  3. [.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦

    [.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦 本节导读:上篇文章简单介绍了.NET面向对象中一个重要的技术反射的基本应用,它可以让我们动态的调 ...

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

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

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

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

  6. Scala进阶之路-正则表达式案例

    Scala进阶之路-正则表达式案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 废话不多说,正则大家都很清楚,那在Scala如何使用正则了?我们直接上个案例,如下: /* @au ...

  7. Scala进阶之路-进程控制之执行shell脚本

    Scala进阶之路-进程控制之执行shell脚本 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 废话不多说,我这里直接放一个案例. /* @author :yinzhengjie ...

  8. Scala进阶之路-Spark底层通信小案例

    Scala进阶之路-Spark底层通信小案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Spark Master和worker通信过程简介 1>.Worker会向ma ...

  9. Scala进阶之路-Scala高级语法之隐式(implicit)详解

    Scala进阶之路-Scala高级语法之隐式(implicit)详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们调用别人的框架,发现少了一些方法,需要添加,但是让别人为你一 ...

随机推荐

  1. 2.go语言入门----变量类型、声明变量、数组、切片

    基本变量类型 介绍几种基本的变量类型:字符串.int.float.bool package main import ( "fmt" ) // 列举几种非常基本的数据类型 func ...

  2. 微软YARP初体验

    本文讨论了微软的反向代理--YARP.YARP是一个可以创建高性能.高度可定制的反向代理服务器的类库.那么什么是反向代理呢?反向代理是位于用户与目标服务器之间的中间连接点.它接收初始的HTTP连接请求 ...

  3. sql语句的练习,已练习的会以绿色标注!!!

    表架构 Student(S#,Sname,Sage,Ssex) 学生表 Course(C#,Cname,T#) 课程表 SC(S#,C#,score) 成绩表 Teacher(T#,Tname) 教师 ...

  4. .NET测试--模拟框架NSubstitute

    .NET测试--模拟框架NSubstitute .NET测试 NSubstitute在GitHub的开源地址:https://github.com/nsubstitute/nsubstitute/do ...

  5. 【知识点】 gcc和g++的联系和区别

    目前(2020-09)GCC 编译器已经更新至 10.2版本,其功能也由最初仅能编译 C 语言,扩增至可以编译多种编程语言,其中就包括 C++ . 除此之外,当下的 GCC 编译器还支持编译 Go.O ...

  6. 致被职场PUA的打工人

    作为打工人,除了每天面对着各种繁琐的工作,还要被动接受上级或多或少的PUA,实在是难上加难,甚至有人想不开而自杀.网络上最近流行了一个词:职场PUA,赋予了这种现象一个正式的名字. 职场PUA指的是职 ...

  7. FreeBSD 入门 哲学与玄学

    『哲学与玄学』 FreeBSD 是一种 UNIX 哲学(如模块化,一切皆文件等,见< UNIX 编程艺术>❩的发展,也是学院派的代表作品.她是一套工具集,她存在目的是为了让人们更好的生活. ...

  8. slickgrid ( nsunleo-slickgrid ) 5 增加子件

    slickgrid ( nsunleo-slickgrid ) 5 增加子件 上次把单元格切换的问题解决了,这次要最做的事情就是给slickgrid的treegird增加子件,我们先选中某一条记录,然 ...

  9. RabbitMQ镜像队列集群搭建、与SpringBoot整合

    镜像模式 集群模式非常经典的就是Mirror镜像模式,保证100%数据不丢失,在实际工作中也是用的最多的,并且实现集群比较的简单. Mirror镜像队列,目的是为了保证 RabbitMQ 数据的高可靠 ...

  10. requirejs的用法

    requirejs的用法 2014年11月6日 17164次浏览 之前我的一片文章介绍过requirejs,具体地址是:http://www.haorooms.com/post/RequireJS_m ...