【java学习笔记】反射基础
一.反射
反射就是在剖析一个类,了解这个类的构造,创建这个类对应的对象。
| Class | 代表字节码的类,代表类的类 |
| Field | 代表属性的类 |
| Method | 代表方法的类 |
| Constructor | 代表构造方法的类 |
| Annotation | 代表注解的类 |
| Package | 代表包的类 |
二.Class类
2.1 获取Class对象:必须有意义
①通过类名.class的方式来获取对应类的字节码对象
import java.util.List;
public class ClassDemo {
@SuppressWarnings("rawtypes")
public static void main(String[] args) {
// 每一个Class对象对应了一个实际的类,因此要求每一个Class都必须有意义
// 如果允许随意创建Class对象,就会导致Class对象没有实际对应的类
// Class clz0 = new Class();
// 通过类名.class的方式来获取一个字节码对象
// clz表示的就是String的字节码
Class<String> clz = String.class;
System.out.println(clz);
// clz2表示的就是List的字节码
Class<List> clz2 = List.class;
System.out.println(clz2);
// clz3表示的就是String[]的字节码
Class<String[]> clz3 = String[].class;
System.out.println(clz3);
}
}
类名.class
②通过对象.getClass()方法来获取这个对象对应的实际类的字节码对象
public class ClassDemo {
@SuppressWarnings("unchecked")
public static void main(String[] args) {
Object str = "abc";
Class<String> clz = (Class<String>) str.getClass();
System.out.println(clz);
}
}
//输出:class java.lang.String
对象.getClass()
③通过Class.forName(类的全路径名)的方法来获取指定类的字节码对象
import java.util.List;
public class ClassDemo {
@SuppressWarnings("unchecked")
public static void main(String[] args) throws ClassNotFoundException {
Class<String> clz = (Class<String>) Class.forName("java.lang.String");
System.out.println(clz);
Class<List> clz2 = (Class<List>) Class.forName("java.util.List");
System.out.println(clz2);
}
}
//输出:
//class java.lang.String
//interface java.util.List
Class.forName(类的全路径名)
2.2创建由此类对象表示的类的新实例
2.2.1 newInstance
字节码对象.newInstance();
用处之举例:
Cat和Dog都实现了Animal接口,利用向上造型,将字符串利用Properties进行更改,可实现任意创建Cat对象或者Dog对象。
public class TestClassDemo {
@SuppressWarnings("unchecked")
public static void main(String[] args) throws Exception {
// 表示获取了Animal类对应的字节码
//"cn.hdu.reflection.Cat"可以用Properties进去获取
Class<Animal> clz = (Class<Animal>) Class.forName("cn.hdu.reflection.Cat");
// 要求类中必须有无参构造
Animal a = clz.newInstance(); //产生对象
System.out.println(a);
}
}
interface Animal {
}
class Dog implements Animal {
@Override
public String toString() {
return "Dog";
}
}
class Cat implements Animal {
@Override
public String toString() {
return "Cat";
}
}
AnimalCatDog
2.2.2 通过构造方法创建新实例
//只能获取public构造方法
Constructor<?>[] getConstructors()
Constructor<T> getConstructor(类<?>... parameterTypes) //不区分是否是public的构造方法
Constructor<?>[] getDeclaredConstructors()
Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
//需要用 Constructor对象.setAccessible(true);进行设置
举例:
import java.lang.reflect.Constructor;
public class ClassGetConstructorDemo {
public static void main(String[] args) throws Exception {
// clz代表String类的字节码
Class<String> clz = String.class;
// String(String)
// 只能获取public修饰的构造方法
Constructor<String> constructorPublic = clz.getConstructor(String.class);
String str = constructorPublic.newInstance("abc");
System.out.println(str);
// String(byte[], int, int)
Constructor<String> constructorParameters = clz.getConstructor(byte[].class, int.class, int.class);
String str2 = constructorParameters.newInstance(new byte[] { 97, 98, 99, 100 }, 1, 3);
System.out.println(str2);
// 获取指定的构造方法,不区分是否是public的
Constructor<String> constructorPrivate = clz.getDeclaredConstructor(char[].class, boolean.class);
// 暴力破解/暴力拆除
constructorPrivate.setAccessible(true);
String str3 = constructorPrivate.newInstance(new char[] { 'a', 'b', 'c' }, true);
System.out.println(str3);
}
}
getConstructor
在String源码中是默认的,非public

2.3 获取方法
//public
Method getMethod(String name, 类<?>... parameterTypes)
Method[] getMethods() //获取所有公有方法(包含了父类的方法也包含Object类) //非public
Method getDeclaredMethod(String name, 类<?>... parameterTypes)
Method[] getDeclaredMethods() //获取所有的成员方法,包括私有的(不包括继承的)
//m.setAccessible(true); //使用所指定的方法:Method的对象.invoke(Object obj, Object... args);
import java.lang.reflect.Constructor;
import java.lang.reflect.Method; public class ClassGetMethodDemo { public static void main(String[] args) throws Exception { Class<String> clz = String.class; // 表示String(String)
Constructor<String> c = clz.getConstructor(String.class);
String str = c.newInstance("abcdefg"); // 获取charAt(int)
// 只能获取public修饰的方法
Method m = clz.getMethod("charAt", int.class);
// 执行方法对象
// 相当于char ch = str.charAt(3);
char ch = (char) m.invoke(str, 3);
System.out.println(ch); // 获取指定的方法
Method m2 = clz.getDeclaredMethod("lastIndexOfSupplementary", int.class, int.class);
m2.setAccessible(true);
int i = (int) m2.invoke(str, 2, 3);
System.out.println(i);
} }
getMethod
2.4 获取属性
//public
Field getField(String name)
Field[] getFields() //非public
Field getDeclaredField(String name)
Field[] getDeclaredFields()
//f.setAccessible(true); //Field类
获取:Object get(Object obj)
设置:void set(Object obj, Object value)
void setChar(Object obj, char c)
void setDouble(Object obj, double d)
void setInt(Object obj, int i)
... ...
import java.lang.reflect.Constructor;
import java.lang.reflect.Field; public class ClassGetFieldDemo {
public static void main(String[] args) throws Exception { Class<String> clz = String.class; // 表示String(String)
Constructor<String> c = clz.getConstructor(String.class);
String str = c.newInstance("abcdefg"); // 获取类中指定的属性
Field f = clz.getDeclaredField("hash");
f.setAccessible(true);
System.out.println(f.get(str));
// 设置属性的值
f.set(str, 125);
// 获取属性的值
System.out.println(f.get(str));
}
}
getField
2.5 获取注解
<A extends Annotation>A getAnnotation(类<A> annotationClass)
Annotation[] getAnnotations() <A extends Annotation>A getDeclaredAnnotation(类<A> annotationClass)
Annotation[] getDeclaredAnnotations()
2.6 Class类其它常用方法
import java.lang.reflect.TypeVariable;
import java.util.ArrayList;
import java.util.List; public class ClassDemo { @SuppressWarnings("rawtypes")
public static void main(String[] args) { Class<String> clz = String.class; // 获取这个类实现的所有的接口
Class[] ins = clz.getInterfaces();
for (Class c : ins) {
System.out.println(c);
} // 获取父类
Class superc = clz.getSuperclass();
System.out.println(superc); // 获取类的全路径名
System.out.println(clz.getName());
// 获取类的简称
System.out.println(clz.getSimpleName()); // 获取所在的包
System.out.println(clz.getPackage()); List<String> list = new ArrayList<>();
Class lclz = list.getClass();
TypeVariable[] ts = lclz.getTypeParameters();
for (TypeVariable typeVariable : ts) {
System.out.println(typeVariable);
} // 判断是否是一个枚举
System.out.println(clz.isAnnotation());
// 判断是否是一个基本类型
System.out.println(clz.isPrimitive()); // 判断参数是否是指定类型的实例的
Object str = "abc";
System.out.println(clz.isInstance(str));
System.out.println(str instanceof String); // 判断两个类之间是否有继承关系
System.out.println(Object.class.isAssignableFrom(String.class)); } }
【java学习笔记】反射基础的更多相关文章
- Java学习笔记之---基础语法
Java学习笔记之---基础语法 一. Java中的命名规范 (一)包名 由多个单词组成时,所有字母小写(例如:onetwo) (二)类名和接口 由多个单词组成时,所有单词首字母大写(例如:OneTw ...
- java学习笔记之基础篇
java选择语句之switch //switch可以用于等值判断 switch (e) //int ,或则可以自动转化成int 的类型,(byte char short)枚举jdk 7中可以防止字 ...
- Java学习笔记--反射
什么是Java反射 概念 java反射是指java能够在运行时确定类的类型信息,包括其方法.字段.构造函数等,并能够通过反射调用类或者类对象的方法.在Java中,java.lang.Class类与ja ...
- C#学习笔记----反射基础
反射基础 反射用于在程序运行过程中,获取类里面的信息或发现程序集并运行的一个过程.通过反射可以获得.dll和.exe后缀的程序集里面的信息.使用反射可以看到一个程序集内部的类,接口,字段,属性,方法, ...
- 0034 Java学习笔记-反射-初步2-操作对象
通过反射创建对象 通过反射创建对象有两种方式,一种通过Class对象的newInstance()方法,一种是获取到Class对象的Constructor后,再调用newInstance()方法,前者要 ...
- 0033 Java学习笔记-反射-初步1
先看看通过反射能干嘛 示例:修改对象的private实例变量 package testpack; import java.lang.reflect.Field; public class Test1 ...
- Java学习笔记--反射API
反射API 1.反射API的介绍 通过反射API可以获取Java程序在运行时刻的内部结构.比如Java类中包含的构造方法.域和方法等元素,并可以与这些元素进行交换. 按照 一般地面向对象的设计 ...
- Java 学习笔记 反射与迭代器
反射 使用反射获得类 Class cls = Class.forName("全类名") //包括包名 Class cls = xx.Class;//xx代表类名 使用反射获得构造方 ...
- JAVA 学习笔记 - 反射机制
1. JAVA反射机制的概念 2. 怎样实例化一个 Class对象 Class.forName(包名.类名); 对象.getClass(); 类.class; ================== ...
- Java学习笔记——反射
反射就是把Java类中的各种成分映射成相应的java类. Class类-->java程序中的各个java类属于同一事物,描述这类事物的Java类名就是Class. Class.forName的作 ...
随机推荐
- PHP读取XML文件
xml主键被json取代,大概了解一下就OK了 简要: 加载xml文件:$xml = simplexml_load_file('sa.xml');//$xml是一个对象 读取节点:echo $xml- ...
- 图片文档倾斜矫正算法 附完整c代码
2年前在学习图像算法的时候看到一个文档倾斜矫正的算法. 也就是说能将一些文档图像进行旋转矫正, 当然这个算法一般用于一些文档扫描软件做后处理 或者用于ocr 文字识别做前处理. 相关的关键词: 抗倾斜 ...
- 模块(module)
1.模块加载 import math //import后面跟模块名 from module1 import module11 //module1是一个大模块,里边有子模块module11,调用这个 ...
- LINUX打开文件
body, table{font-family: 微软雅黑; font-size: 10pt} table{border-collapse: collapse; border: solid gray; ...
- Effective Java 之-----静态工厂与构造器
一. 考虑用静态工厂方法代替构造器: 1)静态工厂方法与构造器不同的第一大优势在于:他们有名称.当一个类需要多个带有相同签名的构造器时,就用静态方法代替构造器,并慎重的选择名称以突出他们间的区别: 2 ...
- oracle用户、权限操作
oracle用户操作,权限操作: 1.创建用户,并让用户默认表空间为tb1: create user 用户名 identified by 密码 default namespace tb1 2.授权: ...
- Spring源码情操陶冶-AnnotationConfigBeanDefinitionParser注解配置解析器
本文承接前文Spring源码情操陶冶-自定义节点的解析,分析spring中的context:annotation-config节点如何被解析 源码概览 对BeanDefinitionParser接口的 ...
- oracle 字符串截取substr和instr
SUBSTR(string,start_position,[length]) 求子字符串,返回字符串解释:string 元字符串 start_position 开始位置(从0开始 ...
- .NET Core版本七牛云SDK使用
一.问题背景 公司目前正在将一部分的业务从.NET平台准备迁移到.NET Core上去,同时也准备启用docker进行.NET Core的部署,在项目迁移过程中,不可避免的碰到有些SDK只有在.NET ...
- H5动画
1.参考:http://blog.csdn.net/whqet/article/details/42911059?readlog https://developer.mozilla.org/zh-CN ...