欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

java jdk动态代理详解

程序员文章站 2023-12-20 17:50:40
jdk动态代理要对一个类进行代理,被代理的类必须实现至少一个接口,并且只有接口中的方法才能被代理。 jdk实现动态代理一般分为三步: 1. 编写接口和实现类。 2....

jdk动态代理要对一个类进行代理,被代理的类必须实现至少一个接口,并且只有接口中的方法才能被代理。

jdk实现动态代理一般分为三步:

1. 编写接口和实现类。

2. 写一个处理器,该处理器实现invocationhandler接口,该接口只有一个方法,其签名为public object invoke(object proxy, method method, object[] args)
throws throwable;可在该处理器的实现方法中,在方法调用前和调用后加入自己的代码,从而进行动态拦截。要注意的是proxy为生成的动态代理类,并不是真的我们被代理的类,所以可在处理器中加入object类型的成员变量,指向我们真正要求被代理的类(即第1步中的实现类)。

3. 使用java.lang.reflect.proxy类的newproxyinstance方法生成动态代理类。对所有要代理方法的调用,都是直接调用生成的动态代理类的方法即可,但是要先对它进行强制类型转换,转换成我们要调用的方法的接口。

jdk原理分析:
通过分析proxy的源代码,可看到动态代理类的详细生成。newproxyinstance方法首先生成动态代理类的class实例,再调用它的参数类型为invocationhandler的构造函数来生成动态代理类并返回。

动态代理类的class实例是怎么生成的呢,是通过proxygenerator类来生成动态代理类的class字节流,把它载入方法区。

分析class字节流生成的过程可以看到它使用proxy为它的父类,实现所有要代理的接口的方法,每个方法的实现体里主要都是调用处理器的invoke方法。

class字节流的生成过程的主要代码如下:

复制代码 代码如下:

private byte[] generateclassfile()
    {
        addproxymethod(hashcodemethod, java/lang/object);
        addproxymethod(equalsmethod, java/lang/object);
        addproxymethod(tostringmethod, java/lang/object);
        for(int i = 0; i < interfaces.length; i++)
        {
            method amethod[] = interfaces[i].getmethods();
            for(int k = 0; k < amethod.length; k++)
                addproxymethod(amethod[k], interfaces[i]);

        }

        list list;
        for(iterator iterator = proxymethods.values().iterator(); iterator.hasnext(); checkreturntypes(list))
            list = (list)iterator.next();

        try
        {
            methods.add(generateconstructor());
            for(iterator iterator1 = proxymethods.values().iterator(); iterator1.hasnext();)
            {
                list list1 = (list)iterator1.next();
                iterator iterator2 = list1.iterator();
                while(iterator2.hasnext())
                {
                    proxymethod proxymethod = (proxymethod)iterator2.next();
                    fields.add(new fieldinfo(proxymethod.methodfieldname, "ljava/lang/reflect/method;", 10));
                    methods.add(<span style="color: red">proxymethod.generatemethod()</span><span style="color: #000000">);</span>       


复制代码 代码如下:

        } 
    } 

    methods.add(generatestaticinitializer()); 

catch(ioexception ioexception) 

    throw new internalerror("unexpected i/o exception"); 

if(methods.size() > 65535) 
    throw new illegalargumentexception("method limit exceeded"); 
if(fields.size() > 65535) 
    throw new illegalargumentexception("field limit exceeded"); 
cp.getclass(dottoslash(classname)); 
cp.getclass("java/lang/reflect/proxy"); 
for(int j = 0; j < interfaces.length; j++) 
    cp.getclass(dottoslash(interfaces[j].getname())); 

cp.setreadonly(); 
bytearrayoutputstream bytearrayoutputstream = new bytearrayoutputstream(); 
dataoutputstream dataoutputstream = new dataoutputstream(bytearrayoutputstream); 
try 

    dataoutputstream.writeint(-889275714); 
    dataoutputstream.writeshort(0); 
    dataoutputstream.writeshort(49); 
    cp.write(dataoutputstream); 
    dataoutputstream.writeshort(49); 
    dataoutputstream.writeshort(cp.getclass(dottoslash(classname))); 
    dataoutputstream.writeshort(cp.getclass("java/lang/reflect/proxy")); 
    dataoutputstream.writeshort(interfaces.length); 
    for(int l = 0; l < interfaces.length; l++) 
        dataoutputstream.writeshort(cp.getclass(dottoslash(interfaces[l].getname()))); 

    dataoutputstream.writeshort(fields.size()); 
    fieldinfo fieldinfo; 

复制代码 代码如下:

//添加属性
for(iterator iterator3 = fields.iterator(); iterator3.hasnext(); fieldinfo.write(dataoutputstream))

fieldinfo = (fieldinfo)iterator3.next();
            //添加方法
            dataoutputstream.writeshort(methods.size());
            methodinfo methodinfo;
            for(iterator iterator4 = methods.iterator(); iterator4.hasnext(); methodinfo.write(dataoutputstream))
                methodinfo = (methodinfo)iterator4.next();

            dataoutputstream.writeshort(0);
        }
        catch(ioexception ioexception1)
        {
            throw new internalerror("unexpected i/o exception");
        }
        return bytearrayoutputstream.tobytearray();
    }


注:代码中加红部分proxymethod.generatemethod()为每个方法生成方法体,通过查看源码可以看出都是在调用invocationhandler接口的实现处理器的invoke方法。

上一篇:

下一篇: