java中把生成Class对象和实例对象弄混了,更何况生成Class对象和生成 instance都有多种方式。所以只有弄清其中的原理,才可以深入理解。首先要生成Class对象,然后再生成Instance。那Class对象的生 成方式有哪些呢,以及其中是如何秘密生成的呢?
Class对象的生成方式如下:
1.Class.forName(“类名字符串”) (注意:类名字符串必须是全称,包名+类名)
2.类名.class
3.实例对象.getClass()
例如:
public class TestClass {
public static void main(String[] args)
{
try {
//测试Class.forName()
Class testTypeForName=Class.forName(“TestClassType”);
System.out.println(“testForName—”+testTypeForName);
//测试类名.class
Class testTypeClass=TestClassType.class;
System.out.println(“testTypeClass—”+testTypeClass);
//测试Object.getClass()
TestClassType testGetClass= new TestClassType();
System.out.println(“testGetClass—”+testGetClass.getClass());
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
class TestClassType{
//构造函数
public TestClassType(){
System.out.println(“----构造函数—”);
}
//静态的参数初始化
static{
System.out.println(“—静态的参数初始化—”);
}
//非静态的参数初始化
{
System.out.println(“----非静态的参数初始化—”);
}
}
控制台输出如下:
—静态的参数初始化—
testForName—class TestClassType
testTypeClass—class TestClassType
----非静态的参数初始化—
----构造函数—
testGetClass—class TestClassType
根据结果可以发现,三种生成的Class对象一样的。并且三种生成Class对象只打印一次“静态的参数初始化”。
我们知道,静态的方法属性初始化,是在加载类的时候初始化。而非静态方法属性初始化,是new类实例对象的时候加载。
因此,这段程序说明,三种方式生成Class对象,其实只有一个Class对象。在生成Class对象的时候,首先判断内存中是否已经加载。
所以,生成Class对象的过程其实是如此的:
当我们编写一个新的java类时,JVM就会帮我们编译成class对象,存放在同名 的.class文件中。在运行时,当需要生成这个类的对象,JVM就会检查此类是否已经装载内存中。若是没有装载,则把.class文件装入到内存中。若 是装载,则根据class文件生成实例对象。
================================================================
Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的內部信息,并能直接操作任意对象的内部属性及方法。
Java反射机制主要提供了以下功能:
-
在运行时构造任意一个类的对象
-
在运行时获取任意一个类所具有的成员变量和方法
-
在运行时调用任意一个对象的方法(属性)
-
生成动态代理
Class 是一个类;
-
一个描述类的类.
-
封装了描述方法的 Method,
-
描述字段的 Filed,
-
描述构造器的 Constructor 等属性.
例如:描述方法-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);
}
}
主要用到的两个方法:
/**
-
@param name the name of the method
-
@param parameterTypes the list of parameters
-
@return the {@code Method} object that matches the specified
*/
public Method getMethod(String name, Class<?>… parameterTypes){
}
/**
-
@param obj the object the underlying method is invoked from
-
@param args the arguments used for the method call
-
@return the result of dispatching the method represented by
*/
public Object invoke(Object obj, Object… args){
}
自定义一个方法
比如Person里有一个方法
public void test(String name,Integer age){
System.out.println(“调用成功”);
}
- 把类对象和类方法名作为参数,执行方法
/**
-
@param obj: 方法执行的那个对象.
-
@param methodName: 类的一个方法的方法名. 该方法也可能是私有方法.
-
@param args: 调用该方法需要传入的参数
-
@return: 调用方法后的返回值
*/
public Object invoke(Object obj, String methodName, Object … args) throws Exception{
//1. 获取 Method 对象
// 因为getMethod的参数为Class列表类型,所以要把参数args转化为对应的Class类型。
Class [] parameterTypes = new Class[args.length];
for(int i = 0; i < args.length; i++){
parameterTypes[i] = args[i].getClass();
System.out.println(parameterTypes[i]);
}
Method method = obj.getClass().getDeclaredMethod(methodName, parameterTypes);
//如果使用getDeclaredMethod,就不能获取父类方法,如果使用getMethod,就不能获取私有方法
//
//2. 执行 Method 方法
//3. 返回方法的返回值
return method.invoke(obj, args);
}
调用:通过对象名,方法名,方法参数执行了该方法
@Test
public void testInvoke() throws Exception{
Object obj = new Person();
invoke(obj, “test”, “wang”, 1);
}
=================================================================================
/**
-
@param className: 某个类的全类名
-
@param methodName: 类的一个方法的方法名. 该方法也可能是私有方法.
-
@param args: 调用该方法需要传入的参数
-
@return: 调用方法后的返回值
*/
public Object invoke(String className, String methodName, Object … args){
Object obj = null;
try {
obj = Class.forName(className).newInstance();
//调用上一个方法
return invoke(obj, methodName, args);
}catch(Exception e) {
e.printStackTrace();
}
return null;
}
调用:
@Test
public void testInvoke() throws Exception{
invoke(“com.atguigu.java.fanshe.Person”,
“test”, “zhagn”, 12);
}
使用系统方法(前提是此类有一个无参的构造器(查看API))
@Test
public void testInvoke() throws Exception{
Object result =
invoke(“java.text.SimpleDateFormat”, “format”, new Date());
System.out.println(result);
}
这种反射实现的主要功能是可配置和低耦合。只需要类名和方法名,而不需要一个类对象就可以执行一个方法。如果我们把全类名和方法名放在一个配置文件中,就可以根据调用配置文件来执行方法.
前面说一般使用getDeclaredMethod获取方法(因为此方法可以获取类的私有方法,但是不能获取父类方法)
如何获取父类方法呢,上一个例子format方法其实就是父类的方法(获取的时候用到的是getMethod)
首先我们要知道,如何获取类的父亲:
比如有一个类,继承自Person
使用:
public class ReflectionTest {
@Test
public void testGetSuperClass() throws Exception{
String className = “com.atguigu.java.fanshe.Student”;
Class clazz = Class.forName(className);
Class superClazz = clazz.getSuperclass();
System.out.println(superClazz);
}
}
//结果是 “ class com.atguigu.java.fanshe.Person ”
此时如果Student中有一个方法是私有方法method1(int age); Person中有一个私有方法method2();
怎么调用
定义一个方法,不但能访问当前类的私有方法,还要能父类的私有方法.
/**
-
@param obj: 某个类的一个对象
-
@param methodName: 类的一个方法的方法名.
-
该方法也可能是私有方法, 还可能是该方法在父类中定义的(私有)方法
-
@param args: 调用该方法需要传入的参数
-
@return: 调用方法后的返回值
*/
public Object invoke2(Object obj, String methodName,
Object … args){
//1. 获取 Method 对象
Class [] parameterTypes = new Class[args.length];
for(int i = 0; i < args.length; i++){
parameterTypes[i] = args[i].getClass();
}
try {
Method method = getMethod(obj.getClass(), methodName, parameterTypes);
method.setAccessible(true);
//2. 执行 Method 方法
//3. 返回方法的返回值
return method.invoke(obj, args);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
-
获取 clazz 的 methodName 方法. 该方法可能是私有方法, 还可能在父类中(私有方法)
-
如果在该类中找不到此方法,就向他的父类找,一直到Object类为止
-
这个方法的另一个作用是根据一个类名,一个方法名,追踪到并获得此方法
-
@param clazz
-
@param methodName
-
@param parameterTypes
-
@return
*/
public Method getMethod(Class clazz, String methodName,
Class … parameterTypes){
for(;clazz != Object.class; clazz = clazz.getSuperclass()){
try {
Method method = clazz.getDeclaredMethod(methodName, parameterTypes);
return method;
} catch (Exception e) {}
}
return null;
}
@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));
}
但是如果需要访问父类中的(私有)字段:
/**
-
//创建 className 对应类的对象, 并为其 fieldName 赋值为 val
-
//Student继承自Person,age是Person类的私有字段/
public void testClassField() throws Exception{
String className = “com.atguigu.java.fanshe.Student”;
String fieldName = “age”; //可能为私有, 可能在其父类中.
Object val = 20;
Object obj = null;
//1.创建className 对应类的对象
Class clazz = Class.forName(className);
//2.创建fieldName 对象字段的对象
Field field = getField(clazz, fieldName);
//3.为此对象赋值
obj = clazz.newInstance();
setFieldValue(obj, field, val);
//4.获取此对象的值
Object value = getFieldValue(obj,field);
}
public Object getFieldValue(Object obj, Field field) throws Exception{
field.setAccessible(true);
return field.get(obj);
}
public void setFieldValue(Object obj, Field field, Object val) throws Exception {
field.setAccessible(true);
field.set(obj, val);
}
public Field getField(Class clazz, String fieldName) throws Exception {
Field field = null;
for(Class clazz2 = clazz; clazz2 != Object.class;clazz2 = clazz2.getSuperclass()){
field = clazz2.getDeclaredField(fieldName);
}
return field;
}
@Test
public void testConstructor() throws Exception{
String className = “com.atguigu.java.fanshe.Person”;
Class clazz = (Class) Class.forName(className);
//1. 获取 Constructor 对象
// 1.1 获取全部
Constructor [] constructors =
(Constructor[]) Class.forName(className).getConstructors();
for(Constructor constructor: constructors){
System.out.println(constructor);
}
// 1.2获取某一个,需要参数列表
Constructor 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();
}
此注解只能用在方法上
@AgeValidator(min=18,max=35)
public void setAge(int age) {
this.age = age;
}
那么我们在给Person类对象的age赋值时,是感觉不到注解的存在的
@Test
public void testAnnotation() throws Exception{
Person person = new Person();
person.setAge(10);
}
必须通过反射的方式为属性赋值,才能获取到注解
/** Annotation 和 反射:
-
- 获取 Annotation
-
getAnnotation(Class annotationClass)
-
getDeclaredAnnotations()
*/
@Test
public void testAnnotation() throws Exception{
String className = “com.atguigu.java.fanshe.Person”;
Class clazz = Class.forName(className);
Object obj = clazz.newInstance();
Method method = clazz.getDeclaredMethod(“setAge”, int.class);
int val = 6;
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
完结
Redis基于内存,常用作于缓存的一种技术,并且Redis存储的方式是以key-value的形式。Redis是如今互联网技术架构中,使用最广泛的缓存,在工作中常常会使用到。Redis也是中高级后端工程师技术面试中,面试官最喜欢问的问题之一,因此作为Java开发者,Redis是我们必须要掌握的。
Redis 是 NoSQL 数据库领域的佼佼者,如果你需要了解 Redis 是如何实现高并发、海量数据存储的,那么这份腾讯专家手敲《Redis源码日志笔记》将会是你的最佳选择。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。[外链图片转存中…(img-Wurv87hi-1713453247673)]
[外链图片转存中…(img-POK94phs-1713453247677)]
[外链图片转存中…(img-nop989Og-1713453247679)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
完结
Redis基于内存,常用作于缓存的一种技术,并且Redis存储的方式是以key-value的形式。Redis是如今互联网技术架构中,使用最广泛的缓存,在工作中常常会使用到。Redis也是中高级后端工程师技术面试中,面试官最喜欢问的问题之一,因此作为Java开发者,Redis是我们必须要掌握的。
Redis 是 NoSQL 数据库领域的佼佼者,如果你需要了解 Redis 是如何实现高并发、海量数据存储的,那么这份腾讯专家手敲《Redis源码日志笔记》将会是你的最佳选择。
[外链图片转存中…(img-8WxKswTv-1713453247681)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!