Java反射机制(一):认识Class类
一、 认识Class类
1.1 正常我们再使用一个类时,大多情况是先获取类的对象,然后通过对象去操作类中的属性或方法。 那,大家有没有想过,如果我们已经有了一个类的对象,我能否通过该对象去获取到类的信息呢?答案是肯定的。
java允许通过对象来找到其所在类的信息,那么实际上就是Class类的功能。
接下来,我们先看一个例子,定义一个类Person,并实例化Person类的一个对象,我们通过该对象去寻找到该Person类
package com.chen.yuan.ref;
class Person{ }
public class ClassDemo
{
public static void main(String[] args)
{
Person per = new Person();
Class c1 = per.getClass();
System.out.println(c1.getName());
}
}
输出: com.chen.yuan.ref.Person
我们发现,现在所有的操作都和我们以前的习惯是反着的了,即通过对象去获取类,而非通过类获取对象。
1.2 从上面的例子中我们可以看出,对象可以通过getClass()方法,获取对象的源头(类),那这个getClass()方法在哪里定义的呢? 查看JDK可以看到,在Object中定义了这个方法,此方法将被所有子类继承:
· public final Class getClass()
` 以上方法的返回值的类型是一个Class类,实际上此类是Java反射的源头。 所谓的反射从程序的运行结果可以这样理解: 可以通过对象反射求出类的名称。
1.3 Class类中常用的方法:
forName(String className) |
传入完成的”包.类”名称实例化Class对象 |
getName() |
获得类的完整名字 |
getFields() |
获得类的public类型的属性 |
getDeclaredFields() |
获得类的所有属性 |
getMethods() |
获得类的public类型的方法 |
getDeclaredMethods() |
获得类的所有方法 |
getMethod(String name, Class[] parameterTypes) |
获取类的特定方法(name参数指定方法名字,parameterTypes参数指定方法参数类型) |
getConstructors() |
获得类的public类型的构造方法 |
getConstructor(Class[] parameterTypes) |
获得类的特定构造方法(parameterTypes参数指定构造方法的参数类型) |
newInstance() |
通过类的不带参数的构造方法创建这个类的一个对象 |
1.4 实例化Class的三种方法:
通过查看源码或JDK文档我们可以发现,Class类的构造方法是私有的,即它无法通过构造方法进行实例化。 Java提供了三种可以实例化Class对象的方法:
a) 使用Class类的静态方法forName:
Class.forName("java.lang.String");
b) 使用类的.class语法:
String.class;
c) 使用对象的getClass()方法(java.lang.Object类中的方法):
String s = "aa";
Class<?> clazz = s.getClass();
package com.chen.yuan.ref;
class Person{ }
public class ClassDemo
{
public static void main(String[] args) throws Exception
{
Class<?> c1 = null;
Class<?> c2 = null;
Class<?> c3 = null;
// 通过Class.forName()进行实例化
c1 = Class.forName("com.chen.yuan.ref.Person");
//通过对象.getClass()方法进行实例化
c2 = new Person().getClass();
// 通过类.class进行实例化
c3 = Person.class; System.out.println("类名称: "+c1.getName());
System.out.println("类名称: "+c2.getName());
System.out.println("类名称: "+c3.getName());
}
}
二、 使用Class类
上文已经提到了一些Class类中的常用方法,当然不全,可以查看JDK文档,了解更加详细的Class中的方法。
接下来,我们假设,如果现在已经实例化好了Class对象,则就可以通过Class类中提供的newInstance()方法,实例化源类(比如Person)的对象。
package com.chen.yuan.ref;
class Person{
private String name;
private int age;
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;
}
@Override
public String toString()
{
return "Person [name=" + name + ", age=" + age + "]";
}
}
public class ClassDemo
{
public static void main(String[] args) throws Exception
{
Class<?> c1 = null; // 通过Class.forName()进行实例化
c1 = Class.forName("com.chen.yuan.ref.Person");
Person per = (Person) c1.newInstance();
per.setName("无缘");
per.setAge(25);
System.out.println(per);
}
}
输出: Person [name=无缘, age=25]
通过以上代码,我们发现,即使不使用new关键字,对象也可以进行实例化操作,这就是反射的作用。
但是,我们需要记住,此时的操作,必须为Person类一共一个无参数的构造方法,否则无法实例化。
package com.chen.yuan.ref;
class Person{
private String name;
private int age; public Person(String name, int age)
{
super();
this.name = name;
this.age = age;
}
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;
}
@Override
public String toString()
{
return "Person [name=" + name + ", age=" + age + "]";
}
}
public class ClassDemo
{
public static void main(String[] args) throws Exception
{
Class<?> c1 = null; // 通过Class.forName()进行实例化
c1 = Class.forName("com.chen.yuan.ref.Person");
Person per = (Person) c1.newInstance();
per.setName("无缘");
per.setAge(25);
System.out.println(per);
}
}
运行之后,程序会报错:
Exception in thread "main" java.lang.InstantiationException: com.chen.yuan.ref.Person
at java.lang.Class.newInstance(Class.java:359)
at com.chen.yuan.ref.ClassDemo.main(ClassDemo.java:42)
所以说,使用以上的方式,实际上还是需要类中构造方法的支持,才能进行对象的实例化。
如果想要解决该问题,则必须明确指定要调用的构造方法,并传递参数。但是,在实际开发中,一般用到反射的地方我们都要求类中含有一个无参的构造方法,便于操作。
如果想调用有参数的构造方法进行反射实例化,可以按如下步骤:
1) 通过Class类的getConstructors()取得本类中的全部构造方法。
2) 向构造方法中传递一个对象数组进去,里面包含了构造方法中所需的各个参数。
3) 之后通过Constructor实例化对象。
package com.chen.yuan.ref; import java.lang.reflect.Constructor; class Person
{
private String name; private int age; public Person(String name, int age)
{
super();
this.name = name;
this.age = age;
} 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;
} @Override
public String toString()
{
return "Person [name=" + name + ", age=" + age + "]";
}
} public class ClassDemo
{
public static void main(String[] args) throws Exception
{
Class<?> c1 = null; // 通过Class.forName()进行实例化
c1 = Class.forName("com.chen.yuan.ref.Person");
Constructor<?>[] cons = c1.getConstructors();
System.out.println(cons.length);
Person per = (Person) cons[0].newInstance("无缘", 25);
per.setName("无缘");
per.setAge(25);
System.out.println(per);
}
}
输出: Person [name=无缘, age=25]
可以看出,通过这种方式,我们确实可以做到对象的实例化,但是,我们最好不要如此操作,最好能为类添加一个无参的构造方法。
Java反射机制(一):认识Class类的更多相关文章
- 利用java反射机制 读取配置文件 实现动态类载入以及动态类型转换
作者:54dabang 在spring的学习过程之中,我们能够看出通过配置文件来动态管理bean对象的优点(松耦合 能够让零散部分组成一个总体,而这些总体并不在意之间彼此的细节,从而达到了真正的物理上 ...
- Java反射机制demo(六)—获得并操作一个类的属性
Java反射机制demo(六)—获得并操作一个类的属性 获得并操作一个类的属性?! 不可思议啊,一个类的属性一般都是私有成员变量啊,private修饰符啊! 但是毫无疑问,这些东西在Java的反射机制 ...
- Java反射机制demo(五)—获得并调用一个类中的方法
Java反射机制demo(五)—获得并调用一个类中的方法 这个demo在使用反射机制操作属性之前,主要原因是因为在.class文件字节码中,方法排在属性的前面. 1,获得一个类中的方法 先看一下方法和 ...
- Java反射机制demo(四)—获取一个类的父类和实现的接口
Java反射机制demo(四)—获取一个类的父类和实现的接口 1,Java反射机制得到一个类的父类 使用Class类中的getSuperClass()方法能够得到一个类的父类 如果此 Class 表示 ...
- Java反射机制demo(三)—获取类中的构造函数
Java反射机制demo(三)—获取类中的构造函数 1,获取类中所有的构造函数 如下面的代码中所示,这个类中显式的构造函数有五个. 空构造: public UserInfo() 带参构造有四个: pu ...
- Java反射机制demo(二)—通过Class实例化任意类的对象
Java反射机制demo(二)—通过Class实例化任意类的对象 上一章节中,实例化了Class类对象的实例,这个部分的demo展示了如何使用Class对象的实例去获得其他类的对象的实例. 任意一个类 ...
- Java反射机制demo(一)—实例化Class对象,并获得其他类包名和类型
Java反射机制demo(一)——实例化Class对象,通过对象获得包名和类型 使用Java中的 本地类作为实验对象,避免自定义的类带来的一些不便. public static void main(S ...
- java反射机制与动态加载类
什么是java反射机制? 1.当程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.我们认为java并不是动态语言,但是它却有一个非常突出的动态相关机制,俗称:反射. IT行业里这么说,没有 ...
- 利用Java反射机制对实体类的常用操作工具类ObjectUtil
代码: ObjectUtil类: import java.lang.reflect.Field; import java.math.BigDecimal; import java.text.Simpl ...
- 浅谈Java反射机制 之 使用类的 属性、方法和构造函数
前面两篇我们总结了Java反射机制如何获取类的字节码,如何获取构造函数,属性和方法, 这篇我们将进一步验证如何使用我们获取到的属性.方法以及构造函数 1.使用 反射 获取到的 属性 import ja ...
随机推荐
- leetcode 235-290 easy
235. Lowest Common Ancestor of a Binary Search Tree 公共的祖先必定大于左点小于右点,否则不断递归到合适. class Solution { publ ...
- 微信小程序--轮播图,标题,盒子,tab栏的合成例子
小程序是什么? 微信小程序,是一种不需要下载安装即可使用的应用,用户扫一扫或搜一下即可打开应用,在微信-发现-小程序可打开应用. 一.小程序的样式编写: 目录结构: app.json { " ...
- CF1067E Random Forest Rank
CF1067E Random Forest Rank 可以证明: 一个树的邻接矩阵的秩,等于最大匹配数*2(虽然我只能证明下界是最大匹配) 而树的最大匹配可以贪心, 不妨用DP模拟这个过程 f[x][ ...
- JSP Web第五章整理复习 JSP访问数据库
P164 例5-1 常用SQL语句 P178 数据库连接池 (1)连接池的作用 存储多个数据库连接对象,当程序需要时,从池中获取1个连接,程序执行完成后再还给连接池.避免数据库连接建立.关闭的开 ...
- 【linux】Ubuntu16.04中文输入法安装
最近刚给笔记本装了Ubuntu+win10双系统,但是ubuntu16.04没有自带中文输入法,所以经过网上的一些经验搜索整合,分享一下安装中文输入法的心得.本文主要介绍了谷歌拼音跟ibus中文输入法 ...
- java reference(转)
http://blog.163.com/xubin_3@126/blog/static/112987702200962211145825/ 在Java中的引用类型,是指除了基本的变量类型之外的所有类型 ...
- python 利用csv模块导入数据
- 洛谷2050 BZOJ2897美食节题解
放个链接 BZ链接 其实这题就是修车的加强版,做法差不多,还是对于每个厨师进行拆点 可是这样强行建图跑网络流会T飞 我们发现,如果一个厨师没有做倒数第x到菜,他一定不会做倒数第x+1到菜 我们的每次增 ...
- 免费报名 | 汇聚HBase&大数据最前沿 Apache HBaseConAsia2019盛会火热来袭
Apache HBase介绍 Apache HBase是基于Apache Hadoop构建的一个高可靠性.高性能.可伸缩的分布式存储系统,它提供了大数据背景下的高性能的随机读写能力,HBase是Goo ...
- 2019-5-25-win10-uwp-win2d-入门-看这一篇就够了
title author date CreateTime categories win10 uwp win2d 入门 看这一篇就够了 lindexi 2019-5-25 20:0:52 +0800 2 ...