序言
有时候,项目中会用到java动态加载指定的类或者jar包反射调用其方法来达到模块的分离,使各个功能之间耦合性大大降低,更加的模块化,代码利用率更高。模式中的代理模式就用到java的这一机制。下边就让我们通过代码来看看如何实现此功能。
代码详细
package loadjarclass;
import java.io.File;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import org.junit.Test;
public class LoadJarClassTest {
@Test
public void testLoadClass() throws Exception{
/*动态加载指定类*/
File file=new File("D:/test");//类路径(包文件上一层)
URL url=file.toURI().toURL();
ClassLoader loader=new URLClassLoader(new URL[]{url});//创建类加载器
//import com.sun.org.apache.bcel.internal.util.ClassLoader;
//ClassLoader classLoader = new ClassLoader(new String[]{""});//类路径
Class<?> cls=loader.loadClass("loadjarclass.TestTest");//加载指定类,注意一定要带上类的包名
Object obj=cls.newInstance();//初始化一个实例
Method method=cls.getMethod("printString",String.class,String.class);//方法名和对应的参数类型
Object o=method.invoke(obj,"chen","leixing");//调用得到的上边的方法method
System.out.println(String.valueOf(o));//输出"chenleixing"
/*动态加载指定jar包调用其中某个类的方法*/
file=new File("D:/test/commons-lang3.jar");//jar包的路径
url=file.toURI().toURL();
loader=new URLClassLoader(new URL[]{url});//创建类加载器
cls=loader.loadClass("org.apache.commons.lang3.StringUtils");//加载指定类,注意一定要带上类的包名
method=cls.getMethod("center",String.class,int.class,String.class);//方法名和对应的各个参数的类型
o=method.invoke(null,"chen",Integer.valueOf(10),"0");//调用得到的上边的方法method(静态方法,第一个参数可以为null)
System.out.println(String.valueOf(o));//输出"000chen000","chen"字符串两边各加3个"0"字符串
}
}
如果对StringsUtils工具类感兴趣的朋友,可以参考我的博文:StringUtils方法全集介绍,commons-lang中常用方法
反射和代理技术知识,可参考我的的博文:Java的静态代理、动态代理,CGLib的动态代理,使用动态代理基于AOP的AspectJ框架—比较与详解,Java反射机制深入研究