素材牛VIP会员
Java反射代码 求解答,谢谢~
 素***2  分类:Java代码  人气:909  回帖:9  发布于6年前 收藏
package test;

import java.lang.reflect.Method;

public class JustTest {

    public static void main(String args[]) {
        try {
            // 将编译后的Just.class这个Class对象,指向c{Java中一切皆对象}
            Class<?> c = Just.class;
            // 获取Just这个类中的自定义的方法,包括private、protected、public、默认{c.getMethods()仅包括public方法}
            Method[] m = c.getDeclaredMethods();
            for (Method me : m) {

                // 获取方法名为“A”的方法名
                if (me.getName().equals("A")) {

                    // 实例化方法,在使用对象之前,先要先加载对应的类
                    Object obj = c.newInstance();
                    // 获取方法,参数为(方法名, Class数组), Class数组中存放着参数的类型
                    Method mm = c.getMethod(me.getName(), new Class[] { java.lang.String.class });
                    // 执行方法的调用,参数为(类的对象,参数数组)
                    mm.invoke(obj, new Object[] { "zxg" });
                }
            }
        } catch (Exception e) {
            // 可能会跑出ClassCastNotFoundation的异常
            e.printStackTrace();
        }
    }
}

class Just {

    public void A(String code) {
        if (code.equals("zxg")) {
            System.out.println("Great!");
        }
    }
}

如上所示的代码,有几点疑惑的求教:

  1. Class和Object之间的关系怎么理解,看了很多解释,还是不明白。
    2.步骤中,Object obj = c.newInstance();这个地方为什么要进行实例化。

求解答,谢谢~

 标签:jdk8java

讨论这个帖子(9)垃圾回帖将一律封号处理……

Lv6 码匠
无***圣 职业无 6年前#1

我们编写java代码,源文件使用以java为后缀命名,java文件编译后对应的以.class后缀命名;
当java虚拟机记载的时class文件,这些文件在虚拟机中必须以某种数据结构的形式存在,来表示对应的class的各种结构及数据,那么这个结构形式就是Class对象,它是对Class的一种描述
而Object是Class对象实例化的结果,在JVM中Class类对象只会有1份(同一个ClassLoader加载的,放在JVM的固定内存区域中,GC不会对其进行管理),但是可以有不同的多个实例对象存在

一个类的方法必须依赖一个具体的实例才可以被调用(除了静态方法),因为其要依赖具体实例的数据及状态~~~

Lv1 新人
BO***OS 职业无 6年前#2

您这里有2个提问:
第1个是实例与类之间关系的问题。这就好比模具与产品之间的关系,比如我们用一个做饼的模具做饼,用模具就好比类,用模具一压就生产出一个饼,每个饼的形状都是一样的,这是产品;模具的作用是使每个饼的大小等属性都一样。而反射的概念更加复杂,它描述的是连模具本身也有属性,比如这个模具有多大,多深,那么类本身也是类似,这个类有什么方法,哪些属性?如果知道了这些东西,就可以动态地实例化对象。我知道了这个模具的属性有哪些行为之后,也可以动态地生产饼,并且调用这个饼的方法;
第2个问题:因为用反射也要实例化对象,虽然没有用new,但也得实例化,不然报空指针。

Lv6 码匠
lo***ou 职业无 6年前#3

首先,Class用于抽象 类的信息,java是面向对象的,比如人这个类,Person,它可以抽象出姓名,性别,年龄等信息;而java提供对于类的抽象,java的类抽象成了Class类,它抽象了包名、类名、属性数组、方法数组等信息,这就是更高一层的抽象了。
这样就好理解反射了,Class<?> c = Just.class;这里的c就是Class的一个对象了,这个对象实例中包含的是Just类,但是你要获取Just类的实例就还要调用c.newInstance()方法了。

Lv1 新人
风***扬 Web前端工程师 6年前#4

类对象Class和方法对象Method也是Object,有自己的属性和方法。
类只有实例化为一个对象,才可以调用对象的实例方法,Class对象是Class类的一个实例。

Lv3 码奴
ch***az 职业无 6年前#5

1.我也解释不清楚
2.A这个方法是成员方法,如果没有实例化的对象是不能调用的,如果是静态方法就不需要了

Lv6 码匠
zh***ni 职业无 6年前#6

class对对象的抽象,对象(Object)对客观事物的抽象

Lv5 码农
sh***ao 职业无 6年前#7

Class是类对象,它当然也继承于Object。这个类对象比较区别于具体的实例对象,一般情况下一个类只有一个类对象,我们可以通过Just.class(这是其中一个办法,差不多有三种办法其实)获得的。使用newnewInstance可以创建多个实例对象。
建议题主看看反射入门知识,然后自己体会类对象的概念,这个是反射的基础。

Lv2 入门
th***as 职业无 6年前#8
  1. 一个Class对象不是class本身,可以认为它是一个用来访问实际class的界面。

  2. invoke一个实例方法需要一个那个类的实例. invoke一个static方法时就可以传null进去。

Lv1 新人
编***魔 职业无 6年前#9

看到问题都解决了,不过我还是从内存结构的层面来补充一下。

我们来看一下实例的在堆内存的结构:

即每个对象实例在 heap 内存中有4部分组成,重点说一下标志位信息(mark word)和 Class 实例指针,标志位信息包括是否有锁,GC 标志位,Youth GC 次数等信息。

Class 实例代表的是一个已经被 Classloader 加载到 jvm 的类,它存储了这个类的描述信息,类名,字段名和方法信息等。

所以楼主的问题也就清楚了,一个 Class 实例只是表示类的描述信息,它并不能等于类的实例,而每个实例会包含一个 Class 实例的指针引用。

通常调用 Class.newInstance() 会做3件事情:

1.开辟一个堆内存空间
2.调用对应类的无参构造方法
3.申请栈的地址,将引用地址指向这个堆内存空间

补充,Class 的实例化由 classloaderdefine() 方法来进行。

 文明上网,理性发言!   😉 阿里云幸运券,戳我领取