一个类有多个组成部分,例如:成员变量,方法,构造方法等。反射就是加载类,并解剖出类的各个组成部分。

编程时什么情况下才需要加载类,并解剖出累的各个组成部分呢?

反射是用来做框架的。

从配置文件中解读类的方法等。

Java中有一个Class类用于代表某一个类的字节码。

Class类既然代表某个类的字节码,它当然就要提供加载某个类自己吗的方法:forName(). forName方法用于加载某个类的字节码到内存中,并使用class对象进行封装。

另外两种得到class对象的方式

类名.class

对象.getClass()

下面看三种类的加载方法

 package cn.itcast.reflect;

 public class Demo1 {

     public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
//加载类的字节码1,常用
Class clazz1 = Class.forName("cn.itcast.reflect.Person"); //加载类的字节码2
Class clazz2 = Person.class; //加载类的字节码3
Class clazz3 = new Person().getClass(); } }

Class对象提供了如下常用方法:

 public Constructor getConstructor(Class<?>... parameterTypes)
public Method getMethod(String name,Class<?>... parameterTypes)
public Field getField(String name) public Constructor getDeclaredConstructor(Class<?>... parameterTypes)
public Method getDeclaredMethod(String name,Class<?>... parameterTypes)
public Field getDeclaredField(String name)

这些方法分别用于从类中解剖出构造函数、方法和成员变量(属性)。解剖出的成员分别使用Constructor,Method,Field对象表示。

思考:假设你是一个框架的设计者,出这些成员后你会干什么?

先看反射类的构造方法。

 //Demo1.java
package cn.itcast.reflect;
import java.lang.reflect.Constructor;
import org.junit.Test;
public class Demo1 { public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
//加载类的字节码1,常用
Class clazz = Class.forName("cn.itcast.reflect.Person");
Constructor c = clazz.getConstructor(null); Object obj = c.newInstance(null); System.out.println(obj); } //反射无参的构造函数
@Test
public void test1() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person");
Constructor c = clazz.getConstructor(null); Object obj = c.newInstance(null); System.out.println(obj);
} //反射有参的构造函数
@Test
public void test2() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person");
Constructor c = clazz.getConstructor(String.class); Person p = (Person) c.newInstance("fix");
System.out.println(p);
} //反射私有的,有参的构造函数
@Test
public void test3() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person");
Constructor c = clazz.getDeclaredConstructor(int.class);
c.setAccessible(true);//暴力反射
Person p = (Person) c.newInstance(1);
System.out.println(p);
}
}

下面看反射类的方法

 package cn.itcast.reflect;

 import java.lang.reflect.Method;

 import org.junit.Test;

 public class Demo2 {
//反射方法 //反射不带参数的方法
@Test
public void test1() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("eat", null); method.invoke(clazz.newInstance(), null);
} //反射带参数的方法
@Test
public void test2() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("run", String.class); method.invoke(clazz.newInstance(), "上海");
} //反射带多个参数的方法
@Test
public void test3() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("run", String.class,int[].class,String[].class); method.invoke(clazz.newInstance(), "上海",new int[]{1,2,3},new String[]{"a","b","c"});
} //反射带返回值的方法
@Test
public void test4() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("test", String.class); String result = (String) method.invoke(clazz.newInstance(), "上海");
System.out.println(result);
} //反射私有方法
@Test
public void test5() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getDeclaredMethod("test2", String.class);
method.setAccessible(true);//暴力反射
String result = (String) method.invoke(clazz.newInstance(), "上海");
System.out.println(result);
} //反射静态方法
@Test
public void test6() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("test3", String.class);
String result = (String) method.invoke(null, "上海");
System.out.println(result);
} //反射main方法
//通过反射调用带数组的方法,要注意处理
@Test
public void test7() throws Exception{
Class clazz = Class.forName("cn.itcast.reflect.Person"); Method method = clazz.getMethod("main", String[].class);
//method.invoke(null, (Object)new String[]{"1","2"});
method.invoke(null, new Object[]{new String[]{"1","2"}});
}
}

利用Field访问属性

Field对象提供了如下方法,用于设置、获取对象属性的值:

public void set(Object obj,Object value)

public Object get(Object obj)

 package cn.itcast.reflect;

 import java.lang.reflect.Field;

 import org.junit.Test;

 public class Demo3 {

     /**
* 反射类的字段
* @throws Exception
*/
//反射公有成员变量
@Test
public void test1() throws Exception{ Person p = new Person();
Class clazz = Class.forName("cn.itcast.reflect.Person"); Field f = clazz.getField("name"); f.set(p, "flx");
System.out.println(p.getName()); } //反射公有成员变量
@Test
public void test2() throws Exception{ Person p = new Person();
p.setName("xxx");
Class clazz = Class.forName("cn.itcast.reflect.Person");
Field f = clazz.getField("name");
String result = (String)f.get(p);
System.out.println(result); } //反射final成员变量
@Test
public void test3() throws Exception{ Person p = new Person();
Class clazz = Class.forName("cn.itcast.reflect.Person");
Field f = clazz.getField("password");
String result = (String)f.get(p);
System.out.println(result); } //反射私有成员变量
@Test
public void test4() throws Exception{
Person p = new Person();
Class clazz = Class.forName("cn.itcast.reflect.Person");
Field f = clazz.getDeclaredField("age");
f.setAccessible(true);
f.set(p, 123);
int result = (Integer)f.getInt(p);
System.out.println(result);
}
}

Person类如下:

 package cn.itcast.reflect;

 public class Person {

     public String name;//字段或成员变量
private int age; public final String password="123"; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public int getAge() {
return age;
} public void setAge(int age) {
this.age = age;
} public Person(){
System.out.println("person");
} public Person(String name){
System.out.println(name);
} private Person(int name){
System.out.println(name);
} public void eat(){
System.out.println("eat!!!");
} public void run(String address){
System.out.println("跑到"+address);
} public void run(String address,int num[],String ss[]){
System.out.println("跑到"+address+"," +num);
} public String test(String str){
return str+"aaaaaa";
} private String test2(String str){
return str+"aaaaaa";
} public static String test3(String str){
return str+"aaaa";
} public static void main(String[] args) {
System.out.println(args[0]);
} }

如上就是反射所有常用的一些方法。

JavaWeb_反射的更多相关文章

  1. JavaWeb_(Spring框架)注解配置

    系列博文 JavaWeb_(Spring框架)xml配置文件  传送门 JavaWeb_(Spring框架)注解配置 传送门 Spring注解配置 a)导包和约束:基本包.aop包+context约束 ...

  2. 隐私泄露杀手锏 —— Flash 权限反射

    [简版:http://weibo.com/p/1001603881940380956046] 前言 一直以为该风险早已被重视,但最近无意中发现,仍有不少网站存在该缺陷,其中不乏一些常用的邮箱.社交网站 ...

  3. Java学习之反射机制及应用场景

    前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...

  4. 关于 CSS 反射倒影的研究思考

    原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...

  5. 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)

    建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...

  6. 运用Mono.Cecil 反射读取.NET程序集元数据

    CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...

  7. .NET面试题系列[6] - 反射

    反射 - 定义,实例与优化 在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等).如果答得好,还可能会 ...

  8. .NET基础拾遗(4)委托、事件、反射与特性

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  9. C++的性能C#的产能?! - .Net Native 系列五:.Net Native与反射

    此系列系小九的学堂原创翻译,翻译自微软官方开发向导,一共分为六个主题.本文是第五个主题:.Net Native与反射. 向导文链接:<C++的性能C#的产能?! - .Net Native 系列 ...

随机推荐

  1. InnoDB引擎的特点及优化方法

    1.什么是InnoDB引擎?      InnoDB引擎是MySQL数据库的另一个重要的存储引擎,正成为目前MySQL AB所发行的新版的标准,被包含在所有二进制安装包里,和其他存储引擎相比,Inno ...

  2. Android Volley完全解析(四),带你从源码的角度理解Volley

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/17656437 经过前三篇文章的学习,Volley的用法我们已经掌握的差不多了,但是 ...

  3. 【phpcms-v9】前台content模块中pc标签的调用说明

    内容模块PC标签调用说明 模块名:content 模块提供的可用操作 操作名 说明 lists 内容数据列表 relation 内容相关文章 hits 内容数据点击排行榜 category 内容栏目列 ...

  4. SQL夯实基础(三):聚合函数详解

    一.GROUP BY  Having 聊聚合函数,首先肯定要弄清楚group by 和having 的用法. SELECT id, COUNT(course) as numcourse, AVG(sc ...

  5. hdu5542 The Battle of Chibi[DP+BIT]

    求给定序列中长度为M的上升子序列个数.$N,M<=1000$. 很容易想到方法.$f[i,j]$表示以第$i$个数结尾,长度为$j$的满足要求子序列个数.于是转移也就写出来了$f[i][j]+= ...

  6. 微信小程序 报错Setting data field "variableName" to undefined is invalid.

    Setting data field "variableName" to undefined is invalid. 将数据字段“variableName”设置为未定义是无效的. ...

  7. CF 622F The Sum of the k-th Powers——拉格朗日插值

    题目:http://codeforces.com/problemset/problem/622/F 发现 sigma(i=1~n) i 是一个二次的多项式( (1+n)*n/2 ),sigma(i=1 ...

  8. HDU1584(蜘蛛牌)

    蜘蛛牌 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  9. MySQL导入MongoDB

    一.MongoDB的导入导出 mongoDB的导入导出,分为mongoDB官方提供的工具类,和第三方的工具类.下面依次介绍下: 1.1.mongoDB提供的工具 1.1.1.mongoimport工具 ...

  10. codeforce 510C Fox And Names(拓扑排序)

    Fox And Names time limit per test 2 seconds memory limit per test 256 megabytes input standard input ...