Class对象

2014-12-03  本文已影响187人  Winston825

Class对象:类型信息在运行时是如何表示的,包含了与类有关的信息,用于创建类的所有的“常规”对象的,Java使用Class对象来执行其RTTI。

类是程序的一部分,每个类都有一个Class对象,被保存在一个同名的.class文件中,为了生存这个类的对象,JVM将使用“类加载器"。

所有类的都在在对其第一次使用时,动态加载到JVM中的。当程序创建第一个对类的静态成员的引用时,就会加载这个类。这个证实构造器也是类的静态方法,即使在构造器之前并没有使用static关键字。因此,使用new操作符创建类的新对象也会被当作类的静态成员的引用。

Class的API:###

    getName():全限定的类名

    getSimpleName():不包含包名的类名

    getCanonicalName():全限定的类名

    getInterfaces():返回所有接口的Class对象

    getClassLoader():返回该类的类加载器。

    getComponentType():返回表示数组组件类型的 Class

    getSuperclass():返回表示此 Class 所表示的实体的超类的 Class。    

    isArray():判定此 Class 对象是否表示一个数组类。

Class实例对象的newInstance()方法来创建的类,必须带有默认构造器。

Class对象的应用方法:###

1.Class.forName()

2.类字面常量:类型.class

简单,安全,因为它在编译时就会受到检查,不需要try语句包围,根除了对forName()方法的调用,所以也更高效。

类字面常量不仅可以应用于普通的类,也可以应用于接口,数组以及基本数据类型。

当使用“.class”来创建对Class对象的引用时,不会自动地初始化该Class对象。初始化被延迟到了对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行。

泛化的Class引用###

Class<?>与Class等价,但优于平凡的Class,非具体的类引用

Class<? extends Number >:任何由Number派生的类

Class<? Super FancyToy>:某个类,它是FancyToy超类

instanceof与Class的等价性###

instanceof和isInstance()生成的结果完全一样,且保持类型的概念
Class的equals()和==也一样,比较实际的Class对象,没有考虑继承

注册工厂###

静态初始化器只有在类首先被加载的情况下才能被调用。
  使用工厂方法设计模式,将对象的创建工作交给类自己去完成。工厂方法可以被多态的调用,从而创建适当类型的对象。

精简版(工厂方法就是Factory接口中的create()方法):

public interface Factory<T> { T create();}

泛型参数T使得create()在每种Factory是实现中返回不同的类型

反射:运行时的类信息(RTTI)

编译时,编译器必须知道所有要通过RTTI来处理的类

反射提供了一种机制——用来检查可用的方法,并返回方法名。
运行时获取类的信息的另一个动机:希望提供在跨网络的远程平台上常见和运行对象的能力,即远程方法调用(RMI)。允许一个java程序将对象分布到多台机器上。

Class类与java.lang.reflect类库一起对反射的概念进行支持,该类库包含了Field、Method以及Constructor类(每个类都是先了Member接口)。这些类型的对象是由JVM在运行时创建的,用以表示未知类里对应的成员。

Constructor->创建新对象
get()和set()->读取和修改与Field对象关联的字段
invoke()方法->调用与Method对象关联的方法
getField(),getMethods()和getConstructors()->返回表示字段、方法以及构造器的对象数组

匿名对象的类信息就能在运行时被完全确定下来,而在编译时不需要知道任何信息。
详见JDK文档

对RTTI来说,编译器在编译时打开和检查.class文件;而对于反射机制来说,.class文件在编译时无法获取,所以在运行时打开和检查.class文件。

类方法提取器###

浏览实现了类定义的源代码或者其JDK文档,只能找到在这个类定义中被定义和被覆盖的方法。反射机制提供了一种方法,使我们能够编写自动展示完整接口的简单工具。

package typeinfo;
import static net.mindview.util.Print.print;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.regex.Pattern;

public class ShowMethod {
    private static Pattern p = Pattern.compile("\\w+\\.");
    public static void main(String[] args) {
        try {
            Class<?> c = Class.forName("typeinfo.ShowMethod");
            Method[] methods = c.getMethods();
            Constructor[] ctors = c.getConstructors();
            for (Method method : methods) {
            print(p.matcher(method.toString()).replaceAll(""));
            }
            for (Constructor constructor : ctors) {
                print(p.matcher(constructor.toString()).replaceAll(""));
            }
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

动态代理###

代理是基本的设计模式之一。它是你为了听过额外或不同的操作,而插入的用来替代“实际”对象的对象。这些操作通常涉及与“实际”对象的通信,因此代理通常充当中间人的角色。

package typeinfo;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

interface Interface{
    void doSomething();
    void somethingElse(String arg);
}

class RealObject implements Interface{
public void doSomething() {
    System.out.println("doSomething");
}

public void somethingElse(String arg) {
    System.out.println("somethingElse " +arg);
}
}

class DynamicProxyHandler implements InvocationHandler{

private Object proxied;
public DynamicProxyHandler(Object proxied) {
    this.proxied = proxied;
}

public Object invoke(Object proxy, Method method, Object[] args)
        throws Throwable {
    System.out.println("*** proxy: " + proxy.getClass() + 
            ", method: " + method + ", args: " + args);
    if (args != null) {
        for (Object arg : args) {
                System.out.println(" " + arg);
            }
        }
            return method.invoke(proxied, args);
        }
    }

public class SimpleDynamicProxy {
    
    public static void consumer(Interface iface){
        iface.doSomething();
        iface.somethingElse("bonobo");
    }
    public static void main(String[] args) {
        RealObject real = new RealObject();
        consumer(real);
        Interface proxy = (Interface) Proxy.newProxyInstance(
    Interface.class.getClassLoader(),
                new Class[] {Interface.class},
                new DynamicProxyHandler(real));
        consumer(proxy);
    }
}

通过调用静态方法Proxy.newProxyInstance()可以创建动态代理,这个方法需要一个类加载器(通常从已经加载的对象中获取其类加载器),一个希望该代理实现的接口列表(不是类或抽象类),以及InvocationHandler接口的一个实现。

动态代理可以将所有调用重定向到调用处理器,因此通常会向调用处理器的构造器传递给一个“实际”对象的引用,从而使得调用处理器在执行其中介任务时,可以将请求转发。

在invoke()内部,在代理上调用方法时需要格外小心,因为对接口的调用将被重定向为对代理的调用。
使用Method.invoke()将请求转发给被代理对象,并传入必须的参数。
注释:内容来自《Java 编程思想》

上一篇下一篇

猜你喜欢

热点阅读