使用 Javassist 动态生成 Hello World

开发 后端
字节码编程在实际的业务开发(CRUD)中并不常用,但是随着网络编程,RPC、动态字节码增强技术和自动化测试以及零侵入APM监控的不断发展与大量使用,越来越多的技术需要使用到字节码编程。

[[436837]]

大家好,我是冰河~~

字节码编程在实际的业务开发(CRUD)中并不常用,但是随着网络编程,RPC、动态字节码增强技术和自动化测试以及零侵入APM监控的不断发展与大量使用,越来越多的技术需要使用到字节码编程。

好了,我们今天就使用Javassist动态生成一个HelloWorld案例,相关的程序案例代码可以关注公众号:冰河技术 获取,也可以直接到Github和Gitee获取。

Github:https://github.com/sunshinelyz/bytecode

Gitee:https://gitee.com/binghe001/bytecode

开发环境

  • JDK 1.8
  • IDEA 2018.03
  • Maven 3.6.0

Maven依赖

在项目的pom.xml文件中添加如下环境依赖。

  1. <properties> 
  2.     <javassist.version>3.20.0-GA</javassist.version> 
  3. </properties> 
  4.  
  5. <dependencies> 
  6.     <dependency> 
  7.         <groupId>org.javassist</groupId> 
  8.         <artifactId>javassist</artifactId> 
  9.         <version>${javassist.version}</version> 
  10.     </dependency> 
  11. </dependencies> 

 

 

案例效果

整体案例效果其实也是很简单的,学习Java语言时,我们会在命令行打印第一个Hello World程序。今天,我们学习Javassist字节码编程时,也来实现一个HelloWorld程序。

案例的效果就是要生成如下的程序代码。

  1. package io.binghe.bytecode.javassist.test; 
  2.  
  3. public class HelloWorld { 
  4.     public static void main(String[] var0) { 
  5.         System.out.println("Javassist Hello World by 冰河(公众号:冰河技术)"); 
  6.     } 
  7.  
  8.     public HelloWorld() { 
  9.     } 

看看这个效果,像不像我们自己在IDEA中写的Java代码呢?就让我们一起使用Javassist来实现它吧。

案例实现

这个案例其实还是蛮简单的,这里就先直接给出源代码了。

  1. /** 
  2.  * @author binghe (公众号:冰河技术) 
  3.  * @version 1.0.0 
  4.  * @description 测试使用Javassist生成第一个类HelloWorld 
  5.  */ 
  6. public class GenerateHelloWorldClass { 
  7.  
  8.     /** 
  9.      * 创建HelloWorld的类,并返回HelloWorld的Class实例 
  10.      */ 
  11.     public static Class createHelloWorld()throws Exception{ 
  12.         //使用默认的ClassPool 
  13.         ClassPool pool = ClassPool.getDefault(); 
  14.         //创建一个空类 
  15.         CtClass ctClass = pool.makeClass("io.binghe.bytecode.javassist.test.HelloWorld"); 
  16.         //添加一个main方法 
  17.         CtMethod ctMethod = new CtMethod(CtClass.voidType, "main", new CtClass[]{pool.get(String[].class.getName())}, ctClass); 
  18.         //将main方法声明为public static类型 
  19.         ctMethod.setModifiers(Modifier.PUBLIC + Modifier.STATIC); 
  20.         //设置方法体 
  21.         ctMethod.setBody("{" + 
  22.                 "System.out.println(\"Javassist Hello World by 冰河(公众号:冰河技术)\");" + 
  23.                 "}"); 
  24.         ctClass.addMethod(ctMethod); 
  25.  
  26.         //将生成的类的class文件输出的磁盘 
  27.         ctClass.writeFile(); 
  28.  
  29.         //返回HelloWorld的Class实例 
  30.         return ctClass.toClass(); 
  31.  
  32.     } 
  33.  
  34.     public static void main(String[] args) throws Exception { 
  35.         Class clazz = createHelloWorld(); 
  36.         Object obj = clazz.newInstance(); 
  37.         Method mainMethod = clazz.getMethod("main", new Class[]{String[].class}); 
  38.         mainMethod.invoke(obj, new String[1]); 
  39.     } 

接下来,我们根据上述代码来看看Javassist是如何生成完整字节码的。

(1) 在createHelloWorld()方法中创建一个ClassPool,ClassPool本质上就是个CtClass对象容器。

(2) 调用ClassPool的makeClass()方法,传入完整的包名+类名生成一个空的类信息。这里传入的完整的包名+类名是io.binghe.bytecode.javassist.test.HelloWorld。

(3) 给类添加方法,并设置方法的返回类型、方法名称、参数名(入参和出参)、访问修饰符以及方法体。这里设置的完整方法体如下:

  1. public static void main(String[] var0) { 
  2.     System.out.println("Javassist Hello World by 冰河(公众号:冰河技术)"); 

(4) 尽管我们在上述代码中没有显示的创建无参构造函数,但是在编译时,Javassist会自动创建一个HelloWorld类的无参构造函数。

(5) 通过 CtClass的writeFile()方法将内存中的类信息输出到磁盘,这样我们就可以通过IDEA清晰的看到Javassist生成的HelloWorld类了。

(6) 最终在createHelloWorld()方法中调用CtClass的toClass()方法返回Class对象。

(7) 在main()方法中调用createHelloWorld()方法获取Class对象。

(8) 通过反射实例化对象,并通过反射调用生成的HelloWorld类的main()方法。

效果演示

运行GenerateHelloWorldClass类的main()方法,会在顶级工程目录下的io/binghe/bytecode/javassist/test 目录下生成HelloWorld.class文件,具体如下所示。

查看IDEA的输出信息时,发现会输出如下内容。

  1. Javassist Hello World by 冰河(公众号:冰河技术) 
  2.  
  3. Process finished with exit code 0 

案例总结

我们使用Javassist实现了创建一个HelloWorld类的功能,字节码编程听起来貌似挺难的,但是在Javassist强大的API下,实现起来还是蛮简单的。

在接下来的一段时间里,冰河会持续输出关于字节码编程的文章,让我们一起精通字节码编程。

好了,今天就到这儿吧,我是冰河,我们下期见~~

本文转载自微信公众号「冰河技术」,可以通过以下二维码关注。转载本文请联系冰河技术公众号。

 

责任编辑:武晓燕 来源: 冰河技术
相关推荐

2014-12-19 10:07:10

C

2017-11-23 17:45:46

Yii框架IntelYii框架深度剖析

2009-08-11 10:32:23

什么是Groovy

2009-07-30 13:21:17

Scala入门Hello World

2009-09-16 17:15:19

OSGi Bundle

2011-06-08 14:39:06

Qt 教程

2023-01-06 08:18:44

2023-09-04 07:30:03

Wasm汇编语言

2022-04-27 10:51:00

PythonMLCubePodman

2012-02-20 14:26:48

JavaPlay Framew

2009-09-28 14:49:08

Spring DMOSGi服务

2023-05-23 08:01:10

Netty网络通信

2024-04-11 13:13:27

2011-08-05 09:48:46

iPhone Interface

2009-08-14 16:54:19

C# Hello Wo

2014-04-11 11:36:42

NDKAndroid开发终端

2011-09-08 10:41:12

Node.js

2017-06-26 08:55:52

2010-01-07 13:27:22

Linux驱动程序

2013-12-12 17:30:03

Lua例子
点赞
收藏

51CTO技术栈公众号