学习于:掘金小册 《深度剖析apache dubbo 核心技术内幕》

web:Java动态字节技术之Javassist

看dubbo的时候重新看到了这东西,然后捡起一些东西来写写。

javassist是一个字节码修改工具,不需要了解字节码指令(同类型的ASM需要),性能比ASM稍差些,但是简单。操作起来跟java反射有些类似。

主要有 ClassPool ,CtClass,CtMethod,CtField,,构造器等。

顾名思义,基本可以理解是干啥的,classpool用来存储ctclass,工作方式与 JVM 类装载器非常相似。

概念啥的网上看看就差不多了,我这里也是基本层次的了解,写个demo比较好,写个复制一个类,然后修改它的方法体,然后加个属性,然后修改类名,然后调用方法的demo。

        <dependency>
            <groupId>org.javassist</groupId>
            <artifactId>javassist</artifactId>
            <version>3.20.0-GA</version>
        </dependency>
这是maven依赖。

 public static void main(String[] args) throws Exception{
        ClassPool cp = ClassPool.getDefault();
        //从池中获取全限定名的ctclass
        CtClass ct = cp.get("com.hbr.service.MyWorld");
        String name = ct.getName();
        System.out.println(name);
        String packageName = ct.getPackageName();
        System.out.println(packageName);
 
        //复制原来的类结构,创建一个新的类
        ct.setName("com.hbr.service.$MyWorld");
 
        //增加foo方法
        CtMethod ctMethod = new CtMethod(CtClass.intType,"foo",new CtClass[]{CtClass.intType,CtClass.intType},ct);
        ctMethod.setModifiers(Modifier.PUBLIC);
        ctMethod.setBody("return $1*$2;");  //$1  $2 代替形参
        ct.addMethod(ctMethod); //加入新的方法
 
        //增加一个属性
        CtField ctField = new CtField(CtClass.intType,"age",ct);
        ctField.setModifiers(Modifier.PUBLIC);
        ct.addField(ctField,"18");
 
        //写入到idea的maven的target目录下,写在其它目录可能不会加载,idea和maven的配置问题
        ct.writeFile(ClassPool.class.getClassLoader().getResource(".").getFile());
 
        //ctclass到class的转换
        Class clazz = ct.toClass();
        Object instance = clazz.newInstance();
        Method fooMethod = clazz.getMethod("foo", int.class, int.class);//这里不能使用integer.class
        Object result = fooMethod.invoke(instance, 2, 2);
        System.out.println(result);
    }
ClassPool cp = ClassPool.getDefault();

Returns the default class pool. The returned object is always identical since this method is a singleton factory.

The default class pool searches the system search path, which usually includes the platform library, extension libraries, and the search path specified by the -classpath option or the CLASSPATH environment variable.

这里注意它的搜索路径

When this method is called for the first time, the default class pool is created with the following code snippet:

ClassPool cp = new ClassPool();

cp.appendSystemPath();

这是它的替代方式

If the default class pool cannot find any class files, try ClassClassPath and LoaderClassPath.

总之就是返回一个包含了默认路径的class的class池。这个理解比较重要。

然后可以通过

cp.insertClassPath("D:codemygitdubbostudyoutyame");

这个方法设定搜索类的路径。(说实话,这个方法有点坑,特别是在maven工程中,跟target目录之类的相关)

String path=ClassPool.class.getClassLoader().getResource(".").getFile();

这个可以获取target下的class路径

CtClass ctClass2 = cp.makeClass("com.hbr.service.MyWorld");

ctClass2.writeFile(path);

这样就可以创建一个包名为com.hbr.service,类名为MyWorld的class。


这样就是单纯的没有.java文件,直接构造一个class文件,就这样


然后回到我们的demo

输入


输出:

结构是


结果增加了一个foo方法,然后反射调用了这个foo方法,然后我把class写入到了classes目录下。(其实是可以不写的) 大家可以自己尝试着操作里面的api,加深一下印象。

然后回到dubbo这里。dubbo会给每个服务提供者的实现类生产一个wrapper类,它最终调用服务提供方的接口实现类,wrapper类的存在是为了减少反射的调用。

可以查看 dubbo生成实现类的Wrapper类_Venture758的专栏-CSDN博客

重点:public class Wrapper1 extends Wrapper implements ClassGenerator.DC


反射传过来类名,方法名,参数类型和参数值。 通过类名我们可以反射获取实例,也就是上面的object,方法名直接是上面的string,参数之类的直接传递即可。

一般来说比如jdk的动态代理就是反射 method.invoke(instance,args)

而这里是invokeMethod直接调用你的这个方法,跟平常的调用没什么区别。 这里手动判断方法名,然后控制要调用的的方法,虽然生成这个类的操作没有反射操作这么简单,但是真正调用的时候,开销就比较小。 而且关键是,生成这个wrapper类的操作虽然比较麻烦,但是它是在dubbo服务启动时生成的,所以不会对运行时带来开销。

然后看看这个wrapper类的生成。


默认有两种方式,一个是JavassistProxyFactory,另一个是JdkProxyFactory。spi默认使用javassist。

看一下源码实现:


跟进去


进到ClassGenerator里面就是我们熟悉的javassist的api操作了。

然后反手看jdk那个


直接反射调用。没啥好说的。

值得注意的是上面的getproxy中的proxy,jdk的proxy是反射包的,javassist的proxy是dubbo包的,即javasist动态生成Proxy再调用newinstance。


也就是完全替代了动态代理的切面增强功能。。。定制化非常固化,适合框架使用,我们还是使用jdk的动态代理比较好,或者cglib吧。
————————————————
版权声明:本文为CSDN博主「暂时没名」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_42348371/article/details/112590002

Logo

腾讯云面向开发者汇聚海量精品云计算使用和开发经验,营造开放的云计算技术生态圈。

更多推荐