Scala进阶(1)—— 反射 object 和 class
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
- Scala 反射的核心是 scala.reflect.runtime.universe
- 很多东西自己也没有特别搞懂,内容参考了网上的博客:https://blog.csdn.net/feloxx/article/details/76034023
- 核心是通过 universe 对象,去依次获得 “Mirror” 这个东西
3.1 使用 Scala 方式反射 Scala class
步骤如下:
- 通过 universe 和 classLoader 找到 JavaMirror
- 通过 JavaMirror 和 实例化对象 获取 InstanceMirror
- 通过 universe 获取目标类的 TypeTag
- 通过 TypeTag 获取目标类的 TermSymbol or MethodSymbol
- 获取 FieldMirror or MethodMirror
- 反射执行
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的更多相关文章
- Scala进阶之路-反射(reflect)技术详解
Scala进阶之路-反射(reflect)技术详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的反射技术和Java反射用法类似,我这里就不一一介绍反射是啥了,如果对 ...
- Scala进阶之路-Scala中的高级类型
Scala进阶之路-Scala中的高级类型 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.类型(Type)与类(Class)的区别 在Java里,一直到jdk1.5之前,我们说 ...
- [.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦
[.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦 本节导读:上篇文章简单介绍了.NET面向对象中一个重要的技术反射的基本应用,它可以让我们动态的调 ...
- Scala进阶之路-Scala中的枚举用法案例展示
Scala进阶之路-Scala中的枚举用法案例展示 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Scala中的枚举值和Java中的枚举值有点差别,不过使用起来也都差大同小异,我这 ...
- Scala进阶之路-Scala中的Ordered--Ordering
Scala进阶之路-Scala中的Ordered--Ordering 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 说道对象的比较,在Java中大家最熟悉不过的就是实现类本身实 ...
- Scala进阶之路-正则表达式案例
Scala进阶之路-正则表达式案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 废话不多说,正则大家都很清楚,那在Scala如何使用正则了?我们直接上个案例,如下: /* @au ...
- Scala进阶之路-进程控制之执行shell脚本
Scala进阶之路-进程控制之执行shell脚本 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 废话不多说,我这里直接放一个案例. /* @author :yinzhengjie ...
- Scala进阶之路-Spark底层通信小案例
Scala进阶之路-Spark底层通信小案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Spark Master和worker通信过程简介 1>.Worker会向ma ...
- Scala进阶之路-Scala高级语法之隐式(implicit)详解
Scala进阶之路-Scala高级语法之隐式(implicit)详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们调用别人的框架,发现少了一些方法,需要添加,但是让别人为你一 ...
随机推荐
- 遇见ZooKeeper:初识
0. 什么是ZooKeeper ZooKeeper 是一个开源的分布式,它的设计目标是将那些复杂且容易出错的分布式协同服务封装起来,抽象出一个高效可靠的原语集,并以一系列简单的接口提供个用户使用. Z ...
- 学习笔记——JVM性能调优之 jmap
jmap jmap(JVM Memory Map)命令可生成head dump文件,还可查询finalize执行队列.Java堆和永久代的详细信息. 通过配置启动参数:-XX:+HeapDumpOnO ...
- Docker-compose编排微服务顺序启动
一.概述 docker-compose可以方便组合多个 docker 容器服务, 但是, 当容器服务之间存在依赖关系时, docker-compose 并不能保证服务的启动顺序.docker-comp ...
- tcp粘包情况分析
1 什么是粘包现象 TCP粘包是指发送方发送的若干包数据到接收方接收时粘成一包,从接收缓冲区看,后一包数据的头紧接着前一包数据的尾.在tcp长连接时,发送端发到buffer里面,接收端也有个buffe ...
- Springboot项目架构设计
导航 前言 流水线 架构的艺术 项目架构 理解阿里应用分层架构 superblog项目架构 结语 参考 本节是<Spring Boot 实战纪实>的第7篇,感谢您的阅读,预计阅读时长3mi ...
- Linux添加普通权限账号并授予root权限
命令创建账号和密码 adduser Mysticbinary #添加一个Mysticbinary用户 passwd Mysticbinary # 输入密码 授予可以切换root的权限 修改/etc/s ...
- 大家最常用的编程论坛是哪个呢,欢迎评论!!掘金16 juejin 简书41 jianshu 博客85 csdn137 csdn
软件编程交流论坛 掘金 16 juejin 简书 41 jianshu 博客 85 cnblogs csdn 137 csdn stackoverflow 0 思否 github 大家最常用的 ...
- golang——net/rpc/jsonrpc包学习
1.jsonrpc包 该实现了JSON-RPC的ClientCodec和ServerCodec接口,可用于rpc包. 可用于跨语言使用go rpc服务. 2.常用方法 (1)func Dial(net ...
- js 判断 是否在当前页面
1.使用visibilitychange 浏览器标签页被隐藏或显示的时候会触发visibilitychange事件. document.addEventListener("visibilit ...
- Jmeter性能常见问题集锦
1. java.net.BindException: Address already in use: connect 开始以为是单机运行脚本运行不过来,所以另加了一台负载机同时运行脚本 分布式环境部署 ...