美文网首页
Class对象

Class对象

作者: Winston825 | 来源:发表于2014-12-03 09:39 被阅读187次

    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 编程思想》

    相关文章

      网友评论

          本文标题:Class对象

          本文链接:https://www.haomeiwen.com/subject/bnmgtttx.html