Java不懂反射,和咸鱼有什么区别啊!

Java反射的功能非常之强大,能够深入的理解其思想,对自己的开发能力将有非常大的提高。

反射概述

Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的內部信息,并能直接操作任意对象的内部属性及方法。

在这里还是要推荐下我自己建的Java学习裙:574加上253再加上075,群里都是学Java开发的,如果你正在学习Java ,小编欢迎你加入,大家都是软件开发党,不定期分享干货(只有Java软件开发相关的),包括我自己整理的一份2018最新的Java进阶资料和高级开发教程,欢迎进阶中和进想深入java的小伙伴

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

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

· 在运行时获取任意一个类所具有的成员变量和方法

· 在运行时调用任意一个对象的方法(属性)

· 生成动态代理

Class 是一个类; 一个描述类的类.

封装了描述方法的 Method,

描述字段的 Filed,

描述构造器的 Constructor 等属性.

获取Class对象的三种方式

1.通过类名获取 类名.class

2.通过对象获取 对象名.getClass()

3.通过全类名获取 Class.forName(全类名)

public class ReflectionTest {

@Test

public void testClass() throws ClassNotFoundException {

Class clazz = null;

//1.通过类名

clazz = Person.class;

//2.通过对象名

//这种方式是用在传进来一个对象,却不知道对象类型的时候使用

Person person = new Person();

clazz = person.getClass();

//上面这个例子的意义不大,因为已经知道person类型是Person类,再这样写就没有必要了

//如果传进来是一个Object类,这种做法就是应该的

Object obj = new Person();

clazz = obj.getClass();

//3.通过全类名(会抛出异常)

//一般框架开发中这种用的比较多,因为配置文件中一般配的都是全类名,通过这种方式可以得到Class实例

String className=" com.atguigu.java.fanshe.Person";

clazz = Class.forName(className);

//字符串的例子

clazz = String.class;

clazz = "javaTest".getClass();

clazz = Class.forName("java.lang.String");

System.out.println();

}

}

反射中最常用的几个类:

Method

Field

Constructor

Annotation

如何描述方法-Method

public class ReflectionTest {

@Test

public void testMethod() throws Exception{

Class clazz = Class.forName("com.atguigu.java.fanshe.Person");

//

//1.获取方法

// 1.1 获取取clazz对应类中的所有方法--方法数组(一)

// 不能获取private方法,且获取从父类继承来的所有方法

Method[] methods = clazz.getMethods();

for(Method method:methods){

System.out.print(" "+method.getName());

}

System.out.println();

//

// 1.2.获取所有方法,包括私有方法 --方法数组(二)

// 所有声明的方法,都可以获取到,且只获取当前类的方法

methods = clazz.getDeclaredMethods();

for(Method method:methods){

System.out.print(" "+method.getName());

}

System.out.println();

// 1.3.获取指定的方法

// 需要参数名称和参数列表,无参则不需要写

// 对于方法public void setName(String name) { }

Method method = clazz.getDeclaredMethod("setName", String.class);

System.out.println(method);

// 而对于方法public void setAge(int age) { }

method = clazz.getDeclaredMethod("setAge", Integer.class);

System.out.println(method);

// 这样写是获取不到的,如果方法的参数类型是int型

// 如果方法用于反射,那么要么int类型写成Integer: public void setAge(Integer age) { }

// 要么获取方法的参数写成int.class

//

//2.执行方法

// invoke第一个参数表示执行哪个对象的方法,剩下的参数是执行方法时需要传入的参数

Object obje = clazz.newInstance();

method.invoke(obje,2);

//如果一个方法是私有方法,第三步是可以获取到的,但是这一步却不能执行

//私有方法的执行,必须在调用invoke之前加上一句method.setAccessible(true);

}

}

如何描述字段-Field

@Test

public void testField() throws Exception{

String className = "com.atguigu.java.fanshe.Person";

Class clazz = Class.forName(className);

//1.获取字段

// 1.1 获取所有字段 -- 字段数组

// 可以获取公用和私有的所有字段,但不能获取父类字段

Field[] fields = clazz.getDeclaredFields();

for(Field field: fields){

System.out.print(" "+ field.getName());

}

System.out.println();

// 1.2获取指定字段

Field field = clazz.getDeclaredField("name");

System.out.println(field.getName());

Person person = new Person("ABC",12);

//2.使用字段

// 2.1获取指定对象的指定字段的值

Object val = field.get(person);

System.out.println(val);

// 2.2设置指定对象的指定对象Field值

field.set(person, "DEF");

System.out.println(person.getName());

// 2.3如果字段是私有的,不管是读值还是写值,都必须先调用setAccessible(true)方法

// 比如Person类中,字段name字段是公用的,age是私有的

field = clazz.getDeclaredField("age");

field.setAccessible(true);

System.out.println(field.get(person));

}

如何描述构造器-Constructor

@Test

public void testConstructor() throws Exception{

String className = "com.atguigu.java.fanshe.Person";

Class<Person> clazz = (Class<Person>) Class.forName(className);

//1. 获取 Constructor 对象

// 1.1 获取全部

Constructor<Person> [] constructors =

(Constructor<Person>[]) Class.forName(className).getConstructors();

for(Constructor<Person> constructor: constructors){

System.out.println(constructor);

}

// 1.2获取某一个,需要参数列表

Constructor<Person> constructor = clazz.getConstructor(String.class, int.class);

System.out.println(constructor);

//2. 调用构造器的 newInstance() 方法创建对象

Object obj = constructor.newInstance("zhagn", 1);

}

如何描述注解 -- Annotation

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)

@Target(value={ElementType.METHOD})

public @interface AgeValidator {

public int min();

public int max();

}

反射小结

1. Class: 是一个类; 一个描述类的类.

封装了描述方法的 Method,

描述字段的 Filed,

描述构造器的 Constructor 等属性.

2. 如何得到 Class 对象:

2.1 Person.class

2.2 person.getClass()

2.3 Class.forName("com.atguigu.javase.Person")

3. 关于 Method:

3.1 如何获取 Method:

1). getDeclaredMethods: 得到 Method 的数组.

2). getDeclaredMethod(String methondName, Class ... parameterTypes)

3.2 如何调用 Method

1). 如果方法时 private 修饰的, 需要先调用 Method 的 setAccessible(true), 使其变为可访问

2). method.invoke(obj, Object ... args);

4. 关于 Field:

4.1 如何获取 Field: getField(String fieldName)

4.2 如何获取 Field 的值:

1). setAccessible(true)

2). field.get(Object obj)

4.3 如何设置 Field 的值:

field.set(Obejct obj, Object val)

5. 了解 Constructor 和 Annotation

ClassLoader

类加载的顺序

public class ReflectionTest {

@Test

public void testClassLoader() throws ClassNotFoundException, FileNotFoundException{

//1. 获取一个系统的类加载器(可以获取,当前这个类PeflectTest就是它加载的)

ClassLoader classLoader = ClassLoader.getSystemClassLoader();

System.out.println(classLoader);

//2. 获取系统类加载器的父类加载器(扩展类加载器,可以获取).

classLoader = classLoader.getParent();

System.out.println(classLoader);

//3. 获取扩展类加载器的父类加载器(引导类加载器,不可获取).

classLoader = classLoader.getParent();

System.out.println(classLoader);

//4. 测试当前类由哪个类加载器进行加载(系统类加载器):

classLoader = Class.forName("com.atguigu.java.fanshe.ReflectionTest")

.getClassLoader();

System.out.println(classLoader);

//5. 测试 JDK 提供的 Object 类由哪个类加载器负责加载(引导类)

classLoader = Class.forName("java.lang.Object")

.getClassLoader();

System.out.println(classLoader);

}

}

//结果:

//sun.misc.Launcher$AppClassLoader@5ffdfb42

//sun.misc.Launcher$ExtClassLoader@1b7adb4a

//null

//sun.misc.Launcher$AppClassLoader@5ffdfb42

//null

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容