`
m635674608
  • 浏览: 5061812 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

慎用动态编译

    博客分类:
  • java
 
阅读更多

动态编译一直是Java的梦想,从Java 6版本它开始支持动态编译了,可以在运行期直接编译.java文件,执行.class,并且能够获得相关的输入输出,甚至还能监听相关的事件。不过,我们 最期望的还是给定一段代码,直接编译,然后运行,也就是空中编译执行(on-the-fly),来看如下代码:

  1. public class Client {  
  2.      public static void main(String[] args) throws Exception {  
  3.          //Java源代码  
  4.          String sourceStr = "public class Hello{    public String sayHello (String name) {return \"Hello,\" + name + \"!\";}}";  
  5.          //类名及文件名  
  6.          String clsName = "Hello";  
  7.          //方法名  
  8.          String methodName = "sayHello";  
  9.          //当前编译器  
  10.          JavaCompiler cmp = ToolProvider.getSystemJavaCompiler();  
  11.          //Java标准文件管理器  
  12.          StandardJavaFileManager fm = cmp.getStandardFileManager(null,null,null);  
  13.          //Java文件对象  
  14.          JavaFileObject jfo = new StringJavaObject(clsName,sourceStr);  
  15.          //编译参数,类似于javac <options>中的options  
  16.          List<String> optionsList = new ArrayList<String>();  
  17.          //编译文件的存放地方,注意:此处是为Eclipse工具特设的  
  18.          optionsList.addAll(Arrays.asList("-d","./bin"));  
  19.          //要编译的单元  
  20.          List<JavaFileObject> jfos = Arrays.asList(jfo);  
  21.          //设置编译环境  
  22.          JavaCompiler.CompilationTask task = cmp.getTask(null, fm, null, optionsList,null,jfos);  
  23.          //编译成功  
  24.          if(task.call()){  
  25.              //生成对象  
  26.              Object obj = Class.forName(clsName).newInstance();  
  27.              Class<? extends Object> cls = obj.getClass();  
  28.              //调用sayHello方法  
  29.              Method m = cls.getMethod(methodName, String.class);  
  30.              String str = (String) m.invoke(obj, "Dynamic Compilation");  
  31.              System.out.println(str);  
  32.         }  
  33.     }  
  34. }  
  35. //文本中的Java对象  
  36. class StringJavaObject extends SimpleJavaFileObject{  
  37.      //源代码  
  38.      private String content = "";  
  39.      //遵循Java规范的类名及文件  
  40.      public StringJavaObject(String _javaFileName,String _content){  
  41.            super(_createStringJavaObjectUri(_javaFileName),Kind.SOURCE);  
  42.            content = _content;  
  43.      }  
  44.      //产生一个URL资源路径  
  45.      private static URI _createStringJavaObjectUri(String name){  
  46.         //注意此处没有设置包名  
  47.         return URI.create("String:///" + name + Kind.SOURCE.extension);  
  48.      }  
  49.      //文本文件代码  
  50.      @Override  
  51.      public CharSequence getCharContent(boolean ignoreEncodingErrors)  
  52.             throws IOException {  
  53.         return content;  
  54.     }  

 

上面的代码较多,这是一个动态编译的模板程序,读者可以拷贝到项目中使用,代码中的中文注释也较多,相信读者看得懂,不多解释,读者只要明白一件事:只要是在本地静态编译能够实现的任务,比如编译参数、输入输出、错误监控等,动态编译就都能实现。

 

Java的动态编译对源提供了多个渠道。比如,可以是字符串(例子中就是字符串),可以是文本文件,也可以是编译过的字节码文件(.class文 件),甚至可以是存放在数据库中的明文代码或是字节码。汇总成一句话,只要是符合Java规范的就都可以在运行期动态加载,其实现方式就是实现 JavaFileObject接口,重写getCharContent、openInputStream、openOutputStream,或者实现 JDK已经提供的两个SimpleJavaFileObject、ForwardingJavaFileObject,具体代码可以参考上个例子。

 

动态编译虽然是很好的工具,让我们可以更加自如地控制编译过程,但是在我目前所接触的项目中还是使用得较少。原因很简单,静态编译已经能够帮我们处 理大部分的工作,甚至是全部的工作,即使真的需要动态编译,也有很好的替代方案,比如JRuby、Groovy等无缝的脚本语言。

 

另外,我们在使用动态编译时,需要注意以下几点:

 

(1)在框架中谨慎使用

 

比如要在Struts中使用动态编译,动态实现一个类,它若继承自ActionSupport就希望它成为一个Action。能做到,但是debug很困难;再比如在Spring中,写一个动态类,要让它动态注入到Spring容器中,这是需要花费老大功夫的。

 

(2)不要在要求高性能的项目使用

 

动态编译毕竟需要一个编译过程,与静态编译相比多了一个执行环节,因此在高性能项目中不要使用动态编译。不过,如果是在工具类项目中它则可以很好地 发挥其优越性,比如在Eclipse工具中写一个插件,就可以很好地使用动态编译,不用重启即可实现运行、调试功能,非常方便。

 

(3)动态编译要考虑安全问题

 

如果你在Web界面上提供了一个功能,允许上传一个Java文件然后运行,那就等于说:“我的机器没有密码,大家都来看我的隐私吧”,这是非常典型的注入漏洞,只要上传一个恶意Java程序就可以让你所有的安全工作毁于一旦。

 

(4)记录动态编译过程

建议记录源文件、目标文件、编译过程、执行过程等日志,不仅仅是为了诊断,还是为了安全和审计,对Java项目来说,空中编译和运行是很不让人放心的,留下这些依据可以更好地优化程序。

 

转载至:http://book.51cto.com/art/201202/317468.htm

 

分享到:
评论

相关推荐

    APK+Dex文件反编译及回编译工具(APKDB)v1.8.1正式版 支持繁体

     从而导致APK文件无法删除,建议配置低的电脑慎用。 __________________________________________________________ 安装及卸载说明  纯属绿色安装,文件直接复制到你指定的目录,同时写入注册表。  卸载方式:...

    jad class文件反编译工具.绿色免安装.方便

    jad反编译工具,可反编译class文件还原成java文件.适合用于查看class文件.请慎用,严禁用于破解他人代码用途

    简单使用GCC编译Keil下工程C文件

    如果你最开始使用Keil重新新建的一个工程并添加了Keil自动生成的启动代码的话(startup_xxx.s),请慎用在ASM选项卡下添加“--gnu”,因为ARMCC下的汇编格式是与GCC完全不一样的,所以用GCC来编译Keil下生成的汇编是...

    浅谈vue中慎用style的scoped属性

    浅谈 Vue 中慎用 style 的 scoped 属性 本篇文章主要介绍了 Vue 中慎用 style 的 scoped 属性的重要性。scoped 属性是在 Vue 组件中使用 style 标签时添加的特殊属性,用于使样式私有化(模块化),避免全局样式...

    MAC版本:opencv4nodejs.tar.gz

    MAC版本的opencv4nodejs的已经编译版本,直接解压放在/usr/local/lib/node_modules目录下...编译环境:MacOS:10.14.6,node:v12.8.0(npm:6.11.2),适用于:appium:1.14.2,其他环境下可能存在不兼容问题,慎用。

    汇编语言集成开发工具3.6

    【使用方法】 一、安装 1.请按照步骤正确安装。...ShutdownMyComputer 关机(慎用) RebootMyComputer 重启(慎用) LogoffMyComputer 注销(慎用) ShutdownAbort 放弃关机 ShowCyndi 显示美女,学习累了可以看看啊,呵呵

    VisualSVN 2.0.2 破解 Crack

    然后覆盖压缩包内的dll到安装目录的bin下的相应文件即可(如C:\Program Files (x86)\VisualSVN\bin)。...(注:覆盖之前请关闭visual studio,适用于2.0.2版本,其它版本请慎用。 本补丁仅供研究) 附:支持vs 2010。

    VisualSVN 2.0.3 破解 Crack

    然后覆盖压缩包内的dll到安装目录的bin下的相应文件即可(如C:\Program Files (x86)\VisualSVN\bin)。...(注:覆盖之前请关闭visual studio,适用于2.0.3版本,其它版本请慎用。 本补丁仅供研究) 附:支持vs 2010。

    opencv 3.4.6 + cuda 10.1 cmake好的资源,包括需要外网下载的资源也打包好

    文件太大,无法上传,300+M,附件中是baidu网盘连接地址,介意慎下 opencv3.4.6 with cuda 10.1 + opencv_contrib 里面有需要从外网下载的一些数据,在路径opencv-3.4.6/opencv-3.4.6/.cache文件夹中,包括FFmpeg ...

    518021910273刘慎恒2

    这意味着它涉及到编译原理中的词法分析、语法分析和语义分析阶段。 2. **界面组件**: - **Code console**:用户在此输入Basic代码,相当于源代码编辑器。 - **INPUT响应区**:在遇到INPUT命令时,系统显示“?”...

    get-version:查看npm包依赖的最新版本号的软件

    慎用查找全部,因为会有超时和网速的影响。软件下载技术栈pug + sass + ECMA8 + react + antd + webpack + nwjs。编译命令输入命令 $ npm start 运行开发环境。输入命令 $ npm build 编译到文件夹。输入命令 $ npm ...

    proxmark3-20241008-HEAD

    冰人固件,rrp分支,最新编译的,需要512k的存储芯片,256的用户慎用..该固件是 pm3 easy版使用, 不适应rdv4 ...里面内置的图像操作页面大家可以下载试用. 里面的bat文件中的 端口号自动修改. ...20241008 编译

    windows64位平台的hadoop2.6插件包(hadoop.dll,winutils.exe)

    在windows8 64位系统安装时自己编译的,基于hadoop2.6的源码,版本低的请慎用。 包含 hadoop.dll hadoop.pdb hadoop.lib hadoop.exp winutils.exe winutils.pdb libwinutils.lib 测试可用。

    SVN 使用规范学习

    * 在项目中要慎用锁定的功能。 * 只有在编辑那些无法合并的文件(例如图片文件,flash 文件等)时,才适当的采用锁定操作。 九、版本管理 * 对已经成熟稳定的版本,可标记为“发布版”。 * 管理员将该版本向技术...

    jiqigou.rar_SRB_影子系统_扇区_磁盘 结构_还原软件

    取文件在扇区的位置后,向磁盘驱动发送srb命令读写扇区,来穿透冰点等还原软件。 编译时注意:FileSystemControl的数据结构需要自己添加。 警告:本代码有一定的危险性,纯粹为了研究、学习,勿...慎用!后果自负!!!

    proxmark3-20211009.rar

    冰人固件,rrp分支,最新编译的,需要512k的存储芯片,256的用户慎用..该固件是 pm3 easy版使用, 不适应rdv4 ...里面内置的图像操作页面大家可以下载试用. 里面的bat文件中的 端口号请自行修改. ...

    proxmark3 rrp iceman固件2020-10-02

    冰人固件,rrp分支,最新编译的,需要512k的存储芯片,256的用户慎用..该固件是 pm3 easy版使用, 不适应rdv4 ...里面内置的图像操作页面大家可以下载试用. 里面的bat文件中的 端口号请自行修改. ...

    eval函数的一些用法

    它的功能是将字符串作为代码执行,这在处理动态生成的代码或者解析JSON数据时尤为...理解和慎用`eval()`,以及寻求更安全的替代方案,是每个开发者的必备技能。在实际开发中,要充分评估安全性和性能,做出明智的选择。

Global site tag (gtag.js) - Google Analytics