• java动态加载类


    /*
    在JAVA中,通过 Class.forName()方法可以动态决定加载哪一个类,这个功能对于相同接口的不同实现来讲非常有用。比如对于设计好的数据库连接池接口,我们可 以有多种的接口实现类来完成相同的功能,同时用户可以简单的通过修改配置文件来指定实际使用哪一个实现类,在源代码里面通过读取配置文件信息,并用 Class.forName(configClassName).getInstance()就可以构造一个特定的实现类实例,而不用每次修改源代码。这 样对于程序来讲只用关心接口的定义,用户只用进行配置文件的设置就完成了同一功能的不同实现的切换。
    但是如果实现类需要通 过静态方法来进行初始化的时候,动态加载的过程就会复杂一些。同样的以数据库连接池为例,通常连接池的构造函数都会被定义为私有的,而通过自定义的 getInstance()静态方法来得到唯一实例。这种情况下简单的通过Class.forName().getInstance()就无法正确构造实 例。
    幸好JAVA所提供的反射机制(Reflection)为我们提供了完整了探悉类内部结构的方法。通过反射机制,我们能够完成基本上所有的运行时决定的动作(虽然这一实现要比其他动态语言,比如PHP,的eval()的使用要复杂的多)。
    下面通过实际的例子说明如何在运行时动态访问类的静态方法。
    -------------------------------------------------------
    */
    import java.lang.reflect.*;
    publicclass myTestClass{
        privatestatic Object pLock = new Object();
        privatestatic myTestClass p_instance = null;
        private String s_configName = "";
        privateboolean b_isFromResource = true;
        publicstatic Object getInstance(String sConfigName,
                                             Boolean bIsFromResource){
            synchronized(pLock){
                if(null == p_instance){
                     p_instance =
                        new myTestClass(sConfigName,bIsFromResource);
                 }
             }
            return p_instance;
         }
        private myTestClass(String sConfigName,Boolean bIsFromResource){
             s_configName = sConfigName;
             b_isFromResource = bIsFromResource.booleanValue();
         }
        publicvoid echoInfo(){
             System.out.println("current arguments : configName=["+
                                 s_configName+"],isFromResource=["+
                                 b_isFromResource+"]");
         }
        publicstaticvoid main(String[] args) throws Exception{
            // 设置方法的传入参数的类型.
             Class[] parameterTypes = new Class[]{
                                         java.lang.String.class,
                                         java.lang.Boolean.class
                                         };
             Method mGetInstance = null;
             String className = "myTestClass";
             Class curTestClass = Class.forName(className);
            try{
                 mGetInstance = curTestClass.
                                 getMethod("getInstance",parameterTypes);
             }
            catch(NoSuchMethodException e){
                 e.printStackTrace();
                 mGetInstance = null;
             }
            if(mGetInstance != null){
                 myTestClass pObj = (myTestClass)
                                     mGetInstance.invoke(
                                        null,
                                        new Object[]{
                                            "src/myconfig.properties",
                                             Boolean.FALSE
                                         }
                                     );
                 pObj.echoInfo();
             }
            else{
                throw
                new Exception("myTest Init Failed from class" +
                                     className +
                                     System.getProperty("line.seperator","\n") +
                                    "method getInstance(String, Boolean) exists.");
             }
         }

    }

  • 相关阅读:
    BZOJ3575 [Hnoi2014]道路堵塞
    BZOJ4456/UOJ184 [Zjoi2016]旅行者
    BZOJ4455/UOJ185 [Zjoi2016]小星星
    BZOJ1036 [ZJOI2008]树的统计Count
    BZOJ2594 [Wc2006]水管局长数据加强版
    BZOJ3669/UOJ3 魔法森林(LCT)
    BZOJ1012:[JSOI2008]最大数
    洛谷【P1175】表达式的转换
    HDU4699:Editor
    BZOJ3039:玉蟾宫
  • 原文地址:https://www.cnblogs.com/chinatefl/p/1210753.html
Copyright © 2020-2023  润新知