一、 认识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类的更多相关文章

  1. 利用java反射机制 读取配置文件 实现动态类载入以及动态类型转换

    作者:54dabang 在spring的学习过程之中,我们能够看出通过配置文件来动态管理bean对象的优点(松耦合 能够让零散部分组成一个总体,而这些总体并不在意之间彼此的细节,从而达到了真正的物理上 ...

  2. Java反射机制demo(六)—获得并操作一个类的属性

    Java反射机制demo(六)—获得并操作一个类的属性 获得并操作一个类的属性?! 不可思议啊,一个类的属性一般都是私有成员变量啊,private修饰符啊! 但是毫无疑问,这些东西在Java的反射机制 ...

  3. Java反射机制demo(五)—获得并调用一个类中的方法

    Java反射机制demo(五)—获得并调用一个类中的方法 这个demo在使用反射机制操作属性之前,主要原因是因为在.class文件字节码中,方法排在属性的前面. 1,获得一个类中的方法 先看一下方法和 ...

  4. Java反射机制demo(四)—获取一个类的父类和实现的接口

    Java反射机制demo(四)—获取一个类的父类和实现的接口 1,Java反射机制得到一个类的父类 使用Class类中的getSuperClass()方法能够得到一个类的父类 如果此 Class 表示 ...

  5. Java反射机制demo(三)—获取类中的构造函数

    Java反射机制demo(三)—获取类中的构造函数 1,获取类中所有的构造函数 如下面的代码中所示,这个类中显式的构造函数有五个. 空构造: public UserInfo() 带参构造有四个: pu ...

  6. Java反射机制demo(二)—通过Class实例化任意类的对象

    Java反射机制demo(二)—通过Class实例化任意类的对象 上一章节中,实例化了Class类对象的实例,这个部分的demo展示了如何使用Class对象的实例去获得其他类的对象的实例. 任意一个类 ...

  7. Java反射机制demo(一)—实例化Class对象,并获得其他类包名和类型

    Java反射机制demo(一)——实例化Class对象,通过对象获得包名和类型 使用Java中的 本地类作为实验对象,避免自定义的类带来的一些不便. public static void main(S ...

  8. java反射机制与动态加载类

    什么是java反射机制? 1.当程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.我们认为java并不是动态语言,但是它却有一个非常突出的动态相关机制,俗称:反射. IT行业里这么说,没有 ...

  9. 利用Java反射机制对实体类的常用操作工具类ObjectUtil

    代码: ObjectUtil类: import java.lang.reflect.Field; import java.math.BigDecimal; import java.text.Simpl ...

  10. 浅谈Java反射机制 之 使用类的 属性、方法和构造函数

    前面两篇我们总结了Java反射机制如何获取类的字节码,如何获取构造函数,属性和方法, 这篇我们将进一步验证如何使用我们获取到的属性.方法以及构造函数 1.使用 反射 获取到的 属性 import ja ...

随机推荐

  1. leetcode 235-290 easy

    235. Lowest Common Ancestor of a Binary Search Tree 公共的祖先必定大于左点小于右点,否则不断递归到合适. class Solution { publ ...

  2. 微信小程序--轮播图,标题,盒子,tab栏的合成例子

    小程序是什么? 微信小程序,是一种不需要下载安装即可使用的应用,用户扫一扫或搜一下即可打开应用,在微信-发现-小程序可打开应用. 一.小程序的样式编写: 目录结构: app.json { " ...

  3. CF1067E Random Forest Rank

    CF1067E Random Forest Rank 可以证明: 一个树的邻接矩阵的秩,等于最大匹配数*2(虽然我只能证明下界是最大匹配) 而树的最大匹配可以贪心, 不妨用DP模拟这个过程 f[x][ ...

  4. JSP Web第五章整理复习 JSP访问数据库

    P164  例5-1  常用SQL语句 P178  数据库连接池 (1)连接池的作用 存储多个数据库连接对象,当程序需要时,从池中获取1个连接,程序执行完成后再还给连接池.避免数据库连接建立.关闭的开 ...

  5. 【linux】Ubuntu16.04中文输入法安装

    最近刚给笔记本装了Ubuntu+win10双系统,但是ubuntu16.04没有自带中文输入法,所以经过网上的一些经验搜索整合,分享一下安装中文输入法的心得.本文主要介绍了谷歌拼音跟ibus中文输入法 ...

  6. java reference(转)

    http://blog.163.com/xubin_3@126/blog/static/112987702200962211145825/ 在Java中的引用类型,是指除了基本的变量类型之外的所有类型 ...

  7. python 利用csv模块导入数据

  8. 洛谷2050 BZOJ2897美食节题解

    放个链接 BZ链接 其实这题就是修车的加强版,做法差不多,还是对于每个厨师进行拆点 可是这样强行建图跑网络流会T飞 我们发现,如果一个厨师没有做倒数第x到菜,他一定不会做倒数第x+1到菜 我们的每次增 ...

  9. 免费报名 | 汇聚HBase&大数据最前沿 Apache HBaseConAsia2019盛会火热来袭

    Apache HBase介绍 Apache HBase是基于Apache Hadoop构建的一个高可靠性.高性能.可伸缩的分布式存储系统,它提供了大数据背景下的高性能的随机读写能力,HBase是Goo ...

  10. 2019-5-25-win10-uwp-win2d-入门-看这一篇就够了

    title author date CreateTime categories win10 uwp win2d 入门 看这一篇就够了 lindexi 2019-5-25 20:0:52 +0800 2 ...