美文网首页
改善Java程序建议17

改善Java程序建议17

作者: KUN叔 | 来源:发表于2017-03-08 14:37 被阅读6次

建议 17: 慎用动态编译。

关于动态编译的代码:

public class Client {
    public static void main(String[] args) throws Exception {
            //Java源代码
            String sourceStr = "public class Hello{public String sayHello (String name)
                  {return \"Hello,\" + name + \"!\";}}";
            //类名及文件名
            String clsName = "Hello";
            //方法名
            String methodName = "sayHello";
            //当前编译器
            JavaCompiler cmp = ToolProvider.getSystemJavaCompiler();
            //Java标准文件管理器
            StandardJavaFileManager fm = cmp.getStandardFileManager(null,null,null);
            //Java文件对象
            JavaFileObject jfo = new StringJavaObject(clsName,sourceStr);
            //编译参数,类似于javac <options>中的options
            List<String> optionsList = new ArrayList<String>();
            //编译文件的存放地方,注意:此处是为Eclipse工具特设的
            optionsList.addAll(Arrays.asList("-d","./bin"));
            //要编译的单元
            List<JavaFileObject> jfos = Arrays.asList(jfo);
            //设置编译环境
            JavaCompiler.CompilationTask task = cmp.getTask(null, fm, null, optionsList,null,jfos);
            //编译成功
            if(task.call()){
                    //生成对象
                    Object obj = Class.forName(clsName).newInstance();
                    Class<? extends Object> cls = obj.getClass();
                    //调用sayHello方法
                    Method m = cls.getMethod(methodName, String.class);
                    String str = (String) m.invoke(obj, "Dynamic Compilation");
                    System.out.println(str);
            }
   }
}
//文本中的Java对象
class StringJavaObject extends SimpleJavaFileObject{
    //源代码
    private String content = "";
    //遵循Java规范的类名及文件
    public StringJavaObject(String _javaFileName,String _content){
          super(_createStringJavaObjectUri(_javaFileName),Kind.SOURCE);
          content = _content;
    }
    //产生一个URL资源路径
    private static URI _createStringJavaObjectUri(String name){
          //注意此处没有设置包名
          return URI.create("String:///" + name + Kind.SOURCE.extension);
    }
    //文本文件代码
    @Override
    public CharSequence getCharContent(boolean ignoreEncodingErrors)
                    throws IOException {
          return content;
   }
}

只要静态编译能做的事情,动态编译就能实现。

动态编译时,需要注意以下几点:
(1)在框架中谨慎使用
比如要在Struts中使用动态编译,动态实现一个类,它若继承自ActionSupport就希望它成为一个Action。能做到,但是debug很困难;再比如在Spring中,写一个动态类,要让它动态注入到Spring容器中,这是需要花费老大功夫的。
(2)不要在要求高性能的项目使用
动态编译毕竟需要一个编译过程,与静态编译相比多了一个执行环节,因此在高性能项目中不要使用动态编译。不过,如果是在工具类项目中它则可以很好地发挥其优越性,比如在Eclipse工具中写一个插件,就可以很好地使用动态编译,不用重启即可实现运行、调试功能,非常方便。
(3)动态编译要考虑安全问题
如果你在Web界面上提供了一个功能,允许上传一个Java文件然后运行,那就等于说:“我的机器没有密码,大家都来看我的隐私吧”,这是非常典型的注入漏洞,只要上传一个恶意Java程序就可以让你所有的安全工作毁于一旦。
(4)记录动态编译过程
建议记录源文件、目标文件、编译过程、执行过程等日志,不仅仅是为了诊断,还是为了安全和审计,对Java项目来说,空中编译和运行是很不让人放心的,留下这些依据可以更好地优化程序。

相关文章

  • 改善Java程序建议17

    建议 17: 慎用动态编译。 关于动态编译的代码: 只要静态编译能做的事情,动态编译就能实现。 动态编译时,需要注...

  • 改善Java程序的一些建议

    一、改善Java程序的一些建议 The reasonable man adapts himself to the ...

  • 改善Java程序建议19

    建议19 : 断言不是鸡肋 基本用法:assert <布尔表达式>assert <布尔表达式> : <错误信息>在...

  • 改善Java程序建议14

    建议14:使用序列化类的私有方法巧妙的解决部分属性持久化问题。 对于不想持久化的属性有几种方法: 对于一些属性不想...

  • 改善Java程序建议12

    建议12:避免用序列化类在构造函数中为不变量赋值 一般来说,final标识的属性是不变量,也就是说只能赋值一次,不...

  • 改善Java程序建议1

    建议1:不要在常量和变量中出现易混淆的字母 包名要小写,类名首字母要大写,常量全部大写并用下划线分隔,变量采用驼峰...

  • 改善Java程序建议2

    建议2:莫让常量蜕变成变量 RAND_CONST 这种常量的定义方式是极不可取的,常量就是常量,在编译期就必须确定...

  • 改善Java程序建议4

    建议4:避免带有变长参数的方法重载 简单折扣后的价格是:¥374.25.这个重载有点特殊:calPrice(int...

  • 改善Java程序建议3

    建议3:三元操作符的类型务必一致。 运行结果是false。s = 90, s1 = 90.0. 三元操作符类型的转...

  • 改善Java程序建议11

    建议11:实现序列化接口的类,显示声明UID 我们在编写实现Serializable接口的类的时候,IDE会提示:...

网友评论

      本文标题:改善Java程序建议17

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