Reference:

Java编程思想

java 反射(Reflect)

Java系列笔记(2) - Java RTTI和反射机制

Java Reflection in Action, 有空再补

----------------------------------------------------------------------------------------------

对于面向对象, 以多态的方式编写程序是首选的方式, 即使用基类reference, 而不是具体的类对象来编写程序

在这种情况下, 你根本就不需要知道基类reference具体指向的是那个类对象, 因为编译器会自动调用相应的逻辑, 大大的简化了编程的复杂度

比如下面的例子, 对于draw, 只需要写Shape.draw(), 此时Shape可以指向circle, square, triangle, 任一个对象

所以对于绝大部分情况下, 其实是不需要知道RTTI (Run-time Type Identification)的, 所以当你用到RTTI, 首先确认你真的需要吗, 设计没有问题吗, 不应该使用多态吗?

当然对于一些特殊和复杂的情况下, 获取RTTI是有较大帮助的,

比如上面的例子, 你只想把circle画成红色来高亮强调, 或则在调用旋转操作的时候想要跳过circle(因为旋转对circle没有意义), 这个时候就需要RTTI

对于Java的RTTI, 无论什么形式, 没有什么神秘的, 毫无例外的都是通过class对象来获得信息.

传统的RTTI和新的Reflection的唯一差别, 就是在何时去check class对象文件
传统的RTTI是在编译时候去check, 这个其实可以满足绝大部分需求

但是对于某些需求,
从磁盘, 或网络连接中读取一个对象, 在编译的时候还没有读, 所以你无法在编译的时候check
在RMI(远程方法调用)中, 需要知道你调用的类的各种信息
这些情况下, class对象文件必须到运行时才能获取(从磁盘或网络上), 这是传统的RTTI无法支持的
这时就需要使用Reflection接口

Class对象

可以看到class对象是RTTI的核心, 下面就具体来看看这个对象

在Java中, 每个class都有一个相应的Class对象(每个类编译完成后, 存放在生成的.class文件中), 用于表示这个类的类型信息.
Class对象用于创建类的所有普通的instance, 并且RTTI也是通过Class对象实现, 哪怕是最简单的cast

当类第一次被使用时, class对象会被类加载器加载到JVM中,

确切的说, 当第一次引用该类的static member时, class对象会被加载. 构造函数是隐含的static member, 所以用new创建对象的时候也会被当作引用static member

This happens when the program makes the first reference to a static member of that class.
It turns out that the constructor is also a static method of a class, even though the static keyword is not used for a constructor. Therefore, creating a new object of that class using the new operator also counts as a reference to a static member of the class.

类加载器, 会先check该class对象是否已经被加载, 如果没有加载, 首先找到对应的.class文件, 然后check这个class文件是否被破坏或含有不良代码, 一旦class对象被载入, 就可以用来创建普通对象

Class对象的生成方式

1. 使用Class的静态成员forName

Class.forName("类名字符串") (注意: 类名字符串必须是全称, 包名+类名), 如果找不到该类会抛ClassNotFoundException异常

2. Class literals, 更加安全和高效的方法, 类名.class

Class literals work with regular classes as well as interfaces, arrays, and primitive types.

可用于接口, 数组和基本类型

需要注意的是, 使用Class literals来创建class对象时, 会有惰性, 对象初始化会被延迟到真正使用, 即引用静态成员时发生

对于基本类型, 还可以使用包装器类的Type

3.实例对象.getClass()

public class ClassTest {
public static void main(String [] args)throws Exception{
String str1="abc";
Class cls1=str1.getClass();
Class cls2=String.class;
Class cls3=Class.forName("java.lang.String");
System.out.println(cls1==cls2); //True
System.out.println(cls1==cls3); //True
}
}

RTTI的形式

总结一下传统的RTTI的形式, 如下3种

1. 向上转型或向下转型(upcasting and downcasting)

向上转型(子类cast到父类)是绝对安全的, 所以不需要check, 可以通过赋值操作完成
但是向下转型(父类cast到子类)却是有风险的, 编译器需要check向下转型是否合理, 这就需要使用RTTI来check实际类型

2. Class对象

传统的RTTI和反射都依赖于Class对象, 什么时候是传统RTTI, 什么时候是反射?

Class<?> c = Class.forName(args[0]); //这种情况就是反射, 只有在运行时产能得到args[0]
Method[] methods = c.getMethods();
Constructor[] ctors = c.getConstructors();

3. instanceof或isInstance()

从下面看出两种的不同用法,

//x是否是Dog类的instance
x instanceof Dog // obj instanceof classname
Dog.class.isInstance(x) //动态的instanceof, classobj.isInstance(obj)

Reflection, 反射

反射, 即在Java运行时环境中动态获取类的信息, 以及动态调用对象的方法的功能, 让Java跨入半动态语言的行列, 毕竟Java不允许动态的更改.

Java 反射机制主要提供了以下功能:

- 在运行时判断任意一个对象所属的类

- 在运行时构造任意一个类的对象

- 在运行时判断任意一个类所具有的成员变量和方法

- 在运行时调用任意一个对象的方法

- 生成动态代理

在JDK中, 主要由以下类来实现Java 反射机制, 这些类都位于java.lang.reflect包中:

- Java.lang.Class类: 代表一个类

- Field类: 代表类的成员变量(成员变量也称为类的属性)

- Method类: 代表类的方法

- Constructor类: 代表类的构造方法

- Array类: 提供了动态创建数组 以及访问数组元素的静态方法

使用的例子, 参考java 反射(Reflect)

动态代理

普通的代理, 静态代理, 很简单, 问题就是必须手工的写代理
那么如果真实的对象中有100个方法, 那么在代理类中就需要写100个代理接口, 是不是很麻烦

动态代理, 就是可以自动生成代理类, 其实代理类是很简单的, 实现代理类的关键就是知道真正的对象中有哪些接口, Reflection出场...

Reflection包中封装实现了动态代理, 可以直接使用

下面给出两种创建动态代理的方法

一种, 先生成代理类, 再创建代理对象

另一种, 直接生成代理对象

/**** 方式一****/
//创建InvocationHandler对象
InvocationHandler handler = new MyInvocationHandler(...);
//创建动态代理类
Class proxyClass = Proxy.getProxyClass(Foo.class.getClassLoader(), new Class[] { Foo.class });
//创建动态代理类的实例
Foo foo = (Foo) proxyClass.getConstructor(new Class[] { InvocationHandler.class }).newInstance(new Object[] { handler }); /**** 方式二****/
//创建InvocationHandler对象
InvocationHandler handler = new MyInvocationHandler(...);
//直接创建动态代理类的实例
Foo foo = (Foo) Proxy.newProxyInstance(Foo.class.getClassLoader(), new Class[] { Foo.class }, handler);

Java RTTI and Reflection的更多相关文章

  1. JAVA RTTI

    基础类可接收我们发给派生类的任何消息,因为两者拥有完全一致的接口.我们要做的全部事情就是从派生上溯造型,而且永远不需要回过头来检查对象的准确类型是什么.所有细节都已通过多态性获得了完美的控制. 但经过 ...

  2. Java进阶之reflection(反射机制)——反射概念与基础

    反射机制是Java动态性之一,而说到动态性首先得了解动态语言.那么何为动态语言? 一.动态语言 动态语言,是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化.比如常见 ...

  3. Java - 反射机制(Reflection)

    Java - 反射机制(Reflection)     > Reflection 是被视为 动态语言的关键,反射机制允许程序在执行期借助于 Reflection API 取得任何类的       ...

  4. 浅析Java RTTI 和 反射的概念

    一.概念: RTTI(Run-Time Type Identification,运行时类型识别)的含义就是在运行时识别一个对象的类型,其对应的类是Class对象,怎么理解这个Class对象呢?如果说类 ...

  5. java 反射(Reflection)

    看了很多关于java 反射的文章,自己把所看到的总结一下.对自己,对他人或多或少有帮助吧. Java Reflection是什么? 首先来看看官方文档Oracle里面对Reflection的描述: R ...

  6. Java系列笔记(2) - Java RTTI和反射机制

    目录 前言 传统的RTTI 反射 反射的实现方式 反射的性能 反射与设计模式 前言 并不是所有的Class都能在编译时明确,因此在某些情况下需要在运行时再发现和确定类型信息(比如:基于构建编程,),这 ...

  7. Java RTTI机制与反射机制

    1.1 什么是RTTI? 维基百科的定义:In computer programming, RTTI (Run-Time Type Information, or Run-Time Type Iden ...

  8. Java RTTI和反射

    一.Java的RTTI  RTTI(Run-Time Type Identification,通过运行时类型识别)的含义就是在运行时识别一个对象的类型,其对应的类是Class对象,每个java里面的类 ...

  9. java RTTI笔记 之Class学习笔记(摘自java编程思想)

    1.java 使用Class对象来执行其RTTI.java 中每个类在编译后都会对应产生一个Class对象(更恰当地说是被保存在一个同名的.class文件中),甚至void和基本类型也都对应一个cla ...

随机推荐

  1. C 语言文件拷贝

    相关的方法: int fputs(const char*s,FILE *stream); int gets(char *s,int size,FILE *stream); 具体代码如下 /** *@a ...

  2. MongoDB GridFS最佳应用概述

    <MongoDB GridFS最佳应用概述> 作者:chszs,转载需注明.博客主页:http://blog.csdn.net/chszs GridFS是MongoDB数据库之上的一个简单 ...

  3. jmeter测试http请求使用csv参数

    创建参数化文件 线程组添加CSV Data Set Config配置元件 为请求添加参数 重新运行查看结果 使用随机参数(JMeter有好多函数可以使用) 查看结果发现使用随机生成的数字 3

  4. [k8s]kubectl windows配置(kubernetic) && kubectl config set-context使用Kubernetic

    参考: https://feisky.gitbooks.io/kubernetes/components/kubectl.html https://kubernetes.io/docs/tasks/t ...

  5. C++忽略字符大小写比较

    在项目中用到对两个字符串进行忽略大小写的比较,有两个方法实现 1.使用C++提供的忽略大小写比较函数实现 代码实现: /* 功能 :忽略大小写进行字符串比较 */ #ifdef __LINUX__ # ...

  6. Windows/OS X下制作Mac安装U盘

    Windows下制作: 方法一:(适用于OSX 10.9以前) 前期准备:一台windows电脑 UltraISO软件 Mac系统镜像dmg(这里使用Mac os x 10.8.4) 至少8GB的U盘 ...

  7. java大数模板

    java 大数计算   这几天做了几道用大数的题,发现java来做大数运算十分方便.对acmer来说是十分实用的 1.valueOf(parament); 将参数转换为制定的类型 比如 int a=3 ...

  8. Angular4中的依赖注入

    在Angular中使用依赖注入,可以帮助我们实现松耦合,可以说只有在组件中使用依赖注入才能真正 的实现可重用的组件. 如果我们有个服务product.service.ts,其中export了一个Pro ...

  9. datagrid后台分页js

    $(function () { gridbind(); bindData(); }); //表格绑定function gridbind() { $('#dg').datagrid({ title: ' ...

  10. Could not load type ‘System.ServiceModel.Activation.HttpModule’ from&

    1. 部署网站到IIS7.5,Window 2008的时候出现这个错误   2. 错误信息 Server Error in ‘/’ Application. Could not load type ‘ ...