【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的作 ...
随机推荐
- tomcat无法打开8080页面
tomcat已启动 app已经正常执行 但不能打开8080管理页面 可能是在webapps目录下没有ROOT目录
- 反向代理和HTTP重定向
1.什么是正向代理(前向代理)? 在NAT技术(Network Address Translation)出现之前,所有主机无法直接与外网相连,要想上网,需要连接到一台能够访问外网的Web服务器,再通过 ...
- [TCP/IP]TCP连接的建立和终止
TCP 是支持全双工通信的传输层协议,为了开发出更好的网络通信应用,清楚了解其中的交互过程是非常必要的. 下面用比较直白的话来描述&理解一下这个过程: TCP 连接建立:三次握手 服务器依次调 ...
- JMS基础篇(二)
简介 异构集成是消息发挥作用的一个领域,大型公司内部可能会遇到很多的平台,Java,.net或者公司自己的平台等. 传送消息还应该支持异步机制,以提高系统整体的性能.异步传输一条消息意味着,发送者不必 ...
- 洛谷 [P1402] 酒店之王
有两个约束条件的二分图匹配 我们回忆一下二分图匹配的匈牙利算法的具体流程,它是通过寻找增广路来判断最大匹配数的,我们再观察一下题目中的两个条件,只有两个条件都满足,才算找到一条增广路,所以我们可以分别 ...
- 2018/1/21 Netty通过解码处理器和编码处理器来发送接收POJO,Zookeeper深入学习
package com.demo.netty; import org.junit.Before;import org.junit.Test; import io.netty.bootstrap.Boo ...
- PhpStorm的破解 汉化
以前一直习惯使用sublime,最近发现phpstorm比submit稍微更强大些,其很多插件都是直接可以使用,不需要另外去拓展了 其中的破解.汉化步骤就需要借助一些资源 (1)破解 安装完毕后,直接 ...
- weblogic修改jdk版本遇到的问题与解决方法
1.修改setDomainEnv ,路径.../domains/xx_domain\bin\ 1.1修改JAVA_HOME为需要修改的路径 注意:BEA_JAVA_HOME路径不需修改 2.修改路径后 ...
- alertifyjs
<%@ page contentType="text/html; charset=UTF-8"%> <!DOCTYPE html PUBLIC "-// ...
- Gitbucket—快速建立自己的Github
GitBucket是一个用Scala语言编写的类似Github的应用,界面非常相似.它非常容易安装–容易到你只需要把它的war文件扔到tomcat中,然后启动tomcat就直接可以访问了.或者直接ja ...