一、前言

动态语言:程序运行时,可以改变程序结构或变量类型。典型的代表:Python,ruby,JavaScript

如JavaScript代码:

function test(){

var s="var a=3;var b=5;alert(a+b)"

eval(s)

}

但是 C、C++、Java不是动态语言,但是Java有一定的动态性,可以称之为准动态语言,可以利用反射机制、字节码操作获得类似动态语言的特性。

Java的动态性让编程更加灵活。

二、反射的概念

指的是程序已经运行起来了但是依然可以加载、探知、使用编译时完全未知的类。 程序在运行状态时,可以动态加载一个只有名称的类,对于任意一个已加载的类,

都能知道这个类所有的属性和方法,对于任意一个对象,都能够调用它的属性和方法。

Class c=Class.forName("com.sxt.test.User");

加载完类("com.sxt.test.User")之后,在堆的内存中,就产生了一个Class类型的对象c(一个类只有一个Class对象),这个对象c就包含了完整的类的结构信息。

我们可以通过这个对象c看到类("com.sxt.test.User")的结构。这个对象c就是一面镜子,透过这个镜子可以看到类的结构,所以形象的称之为:反射

(通俗理解:把一个类映射成一个对象,以便于好操纵这个类)

【基本概念】

/***
* 反射:把一个类映射成一个对象,以便于好操纵这个类
*/
package cn.sxt.jvm; @SuppressWarnings("all")
public class Test_0417_Reflection {
public static void main(String[] args) throws Exception {
String path="cn.sxt.jvm.Test_0417_User"; Class clz=Class.forName(path);//对象表示或封装一些数据,一个类被加载之后,jVM会创建一个对应于该类的Class对象clz
//类的整个结构信息会被加载到相应的Class对象(clz)中去。这个对象映射了这个类的全部信息。
System.out.println(clz);
System.out.println(clz.hashCode()); Class clz2=Class.forName(path);
System.out.println(clz2.hashCode());//输出结果是一样的,说明一个类只能对应一个反射对象(汽车图纸)。 Class strClass=String.class;//获取常用的 String类的映射对象
System.out.println(strClass);
Class strClass2=path.getClass();//path是个字符串对象,通过对象.getClass()获取映射对象
System.out.println(strClass2);
System.out.println(strClass==strClass2);//输出为真,说明获得的是同一个映射对象 Class intClass=int.class; int arr01[]=new int[10];
int arr02[]=new int[20];
int arr03[][]=new int[5][3];
double arr04[]=new double[10];
System.out.println(arr01.getClass().hashCode());
System.out.println(arr02.getClass().hashCode());//arr01[]和arr02[]输出结果是一样的,说明映射对象与数组大小无关
System.out.println(arr03.getClass().hashCode());//输出结果不一样,而是与维数(一维or二维)相关
System.out.println(arr04.getClass().hashCode());//输出结果不一样,映射对象与数据类型相关 }
}

【反射的作用】

/**
利用反射的API获取类的信息(名字、属性、方法、构造器)
*
*/
package cn.sxt.jvm; import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
@SuppressWarnings("all")
public class Test_0417_Reflection2 {
public static void main(String[] args) throws Exception {
Class clz=Class.forName("cn.sxt.jvm.Test_0417_User"); //获得类信息
System.out.println(clz.getName());//获取类的完整路径(包括包名+类名)
System.out.println(clz.getSimpleName());//获取类的简单路径(仅给出类名) //获得属性信息
Field[] fields=clz.getFields();//获取公开(public修饰的)的属性信息
Field[] fields2=clz.getDeclaredFields();//获取所有属性信息(包括public和private修饰的) Declared:宣布的,定义的
System.out.println(fields.length);//输出0 ,表示数组中没东西
System.out.println(fields2.length);//输出3 表示获得了3个私有的属性
for (Field temp : fields2) {
System.out.println("属性:"+temp);
}
System.out.println("获得单个属性:"+clz.getDeclaredField("name"));//获得单个属性name //获得方法(不含构造方法)信息
Method[] methods =clz.getDeclaredMethods();//获得所有方法(公开+私有的方法)
System.out.println(methods.length);//输出结果为6 即6个get和set方法
Method method=clz.getDeclaredMethod("setName", String.class);//参数为(方法的名字,方法中传进去参数的类型)
System.out.println(method); //获得构造方法的信息
Constructor[] constructors=clz.getDeclaredConstructors();
System.out.println(constructors.length);//输出为2
System.out.println(clz.getDeclaredConstructor(null));//获取所有的构造器中无参数的构造器
System.out.println(clz.getDeclaredConstructor(int.class,String.class,int.class));//获取所有的构造器中有如下参数的构造器 }
}
/***
* 通过反射API调用构造方法 构造对象、普通方法、属性
*/
package cn.sxt.jvm; import java.lang.reflect.Field;
import java.lang.reflect.Method; @SuppressWarnings("all")
public class Test_0417_Reflection3 {
public static void main(String[] args) throws Exception {
Class clz=Class.forName("cn.sxt.jvm.Test_0417_User"); //通过反射API调用构造方法 构造对象.
Test_0417_User user=(Test_0417_User)clz.newInstance();//其实是调用Test_0417_User的无参构造方法 Test_0417_User user2=(Test_0417_User)clz.getDeclaredConstructor(int.class,String.class,int.class).
newInstance(1001,"小李",18);//通过调用有参数的构造方法实例化一个对象
System.out.println(user2.getAge()); //通过反射API动态调用普通方法
Test_0417_User user3=(Test_0417_User)clz.newInstance();
Method method=clz.getDeclaredMethod("setName", String.class);//"setName"这里可以灵活多变,需要什么传什么
method.invoke(user3, "小王");//后2行代码等价于user3.setName("小王"); invoke:激活
System.out.println(user3.getName()); //通过反射API操作属性
Test_0417_User user4=(Test_0417_User)clz.newInstance();
Field field=clz.getDeclaredField("name");
field.setAccessible(true);//意思是访问name这个属性不用做安全检查,直接访问(否则name属性是私有的外边的类访问不了)
field.set(user4, "小张");//通过反射直接写属性的值
System.out.println(user4.getName());//通过反射直接读属性的值
System.out.println(field.get(user4));//输出结果一样
}
}

[19/04/17-星期三] Java的动态性_反射(Reflection)机制的更多相关文章

  1. [19/04/18-星期四] Java的动态性_动态编译(DynamicCompiler,Dynamic:动态的,Compiler:编译程序)

    一.概念 应用场景:如在线评测系统,客户端编写代码,上传到服务器端编译运行:服务器动态加载某些类文件进行编译 /*** * */ package cn.sxt.jvm; import java.io. ...

  2. [19/04/20-星期六] Java的动态性_字节码操作(Javassist类库(jar包),assist:帮助、援助)

    一.概念 [基本] /** * */ package cn.sxt.jvm; import javassist.ClassPool; import javassist.CtClass; import ...

  3. [19/04/19-星期五] Java的动态性_脚本(Script,脚本)引擎执行JavaScript代码

    一.概念 Java脚本引擎是jdk 6.0之后的新功能. 使得Java应用程序可以通过一套固定的接口与各种脚本引擎交互,从而达到在Java平台上调用各种脚本语言的目的. Java脚本API是连接Jav ...

  4. [19/04/03-星期三] IO技术_其它流(RandomAccessFile 随机访问流,SequenceInputStream 合并流)

    一.RandomAccessFile 随机访问流 [版本1] /* *RandomAccessFile 所谓随机读取就是 指定位置开始或指定位置结束 的读取写入文件 * 实现文件的拆分与合并 模拟下载 ...

  5. [19/04/02-星期二] IO技术_字符流分类总结(含字符转换流InputStreamReader/ OutputStreamWriter,实现字节转字符)

    一.概念 ------->1.BufferedReader/BufferedWriter [参考19.03.31文章] *Reader/Writer-------->2.InputStre ...

  6. [19/04/01-星期一] IO技术_字节流分类总结(含字节数组(Array)流、字节数据(Data)流、字节对象(Object)流)

    一.字节流分类概括 -->1.ByteArrayInputStream /ByteArrayOutputStream(数组字节输入输出)        InputStream/OutputStr ...

  7. [19/04/13-星期六] 网络编程_基本概念(关注传输层、数据传输,TCP和UDP)

    一.概念 ▪ 什么是计算机网络? 计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接起来,在网络操作系统, 网络管理软件及网络通信协议的管理和协调下,实现资源共享和信 ...

  8. JDK1.7新特性(4):java语言动态性之反射API

    直接通过一个代码示例来熟悉java中通过反射来对构造函数/域以及方法处理的相关API: package com.rampage.jdk7.chapter2; import java.lang.refl ...

  9. Java之注解与反射

    Java之注解与反射 注解(Annotation)简介 注解(Annotation)是从JDK5.0引入的新技术 Annotation作用:注解(Annotation)可以被其他程序如编译器等读取 A ...

随机推荐

  1. 编写更好的jQuery代码(转)

    这是一篇关于jQuery的文章,写到这里给初学者一些建议. 原文地址:http://flippinawesome.org/2013/11/25/writing-better-jquery-code/ ...

  2. Golang报错mixture of field:value and value initializers

    Golang 在使用匿名成员初始化时,如果出现 mixture of field:value and value initializers 是因为初始化的方式不对,见代码: package main ...

  3. Docker学习之Docker容器基本使用

    Docker学习之Docker容器基本使用 新建容器并启动 命令格式:docker run --options repository:tag 后台运行 命令格式:-d 已存在的容器相关操作 启动:do ...

  4. Spring-在IDEA2016中创建maven管理的SpringMVC项目

    这是一套我自己摸索出来的创建项目方法,基本是用在创建用maven管理的 Spring+SpringMVC+Mybatis的J2EE项目时. 创建一个maven管理的webapp项目 在创建项目时,选择 ...

  5. EF fluent API如何配置主键不自动增长

    在Dbcontext中作如下添加: protected override void OnModelCreating(DbModelBuilder modelBuilder) { modelBuilde ...

  6. 撩课-Web大前端每天5道面试题-Day14

    1. 请写出至少5个html5新增的标签,并说明其语义和应用场景? section:定义文档中的一个章节; nav:定义只包含导航链接的章节; header:定义页面或章节的头部; 它经常包含 log ...

  7. Django实现数据库中表格的增删查改

    1.urls.py """Django_demo1 URL Configuration The `urlpatterns` list routes URLs to vie ...

  8. 使用react——解决this.props.history.push无法跳转的问题

    转自: https://blog.csdn.net/yingzizizizizizzz/article/details/78751305 场景: 一个组件中,含有ul展开数组的组件,在每一行中,都能点 ...

  9. div实现水平和垂直都居中的三个超实用的方法

    本文仅仅介绍作者认为的三种不错的方式, 方式一:transform: translate(-50%,-50%)  示例代码如下: .div{ position: absolute; top: 50%; ...

  10. Oracle 通过出生日期计算年龄

    方法一: SELECT TRUNC(months_between(sysdate, birth)/12) AS age from mytable 方法二: select TRUNC((to_char( ...