分享
 
 
 

一个在程序中实现plugin机制的简单例子

王朝java/jsp·作者佚名  2006-01-30
窄屏简体版  字體: |||超大  

所有的plugin模块都应该实现一个ModuleInterface接口:

public interface ModuleInterface

{

???public boolean handles(Object key);

}

这意味着所有的模块都有一个handles方法,这个方法提供了一个挑选模块的机制,根据传入的对象识别模块,例如传入一个String,根据这个String判断这是不是一个适用的模块然后返回true或false。

?

每个模块都应该在jar文件的meta-inf目录下的元数据文件中描述自己。例如在一个module.mf中:

module.name=Module1

module.type=TestInterface

module.class=TestModuleClass

module.type指明了这个模块的种类(TestInterface是一个接口或一个类,指明这个模块是哪一种模块,例如它是过滤器、协议、界面或功能扩展等等),它必须implements了ModuleInterface接口,module.class是该模块具体的类。在这个例子中module.type是应用程序的一部分,而module.class就是各个作为plugin的jar中的一部分了。

?

ModuleScanner用于搜索一个或多个指定目录,查找所有模块的jar文件,并从jar文件中获得模块的描述信息,然后用模块的文件引用和描述信息构造一个ModuleSpec对象,这样每一个ModuleSpec对象就对应一个模块,ModuleScanner维护着一个列表,保存了所有的ModuleSpec对象。

import java.io.*;

import java.net.*;

import java.util.*;

import java.util.zip.*;

import java.util.jar.*;

?

public class ModuleScanner

{

? protected List moduleSpecList = new ArrayList();

?

? public void scan(File[] paths) throws IOException{

????? for(int i=0; i

??????? scan(paths[i]);

????? }

? }

?

? public void scan(File path) throws IOException{

??? File[] list = path.listFiles();

??? for(int i =0; i

????? File file = list[i];

????? if(file.isFile() && file.getName().toLowerCase().endsWith(".jar")){

??????? System.out.println(file.getName());

??????? Properties props = readModuleManifest(file);

??????? if(props == null) continue;

??????? ModuleSpec spec = new ModuleSpec( file.toURL(),props);

??????? moduleSpecList.add(spec);

??????? System.out.println(spec);

??? ??}

??? }

? }

?

? protected Properties readModuleManifest(File file) throws IOException{

??? JarFile jar = new JarFile(file);

??? JarEntry entry = findModuleEntry(jar,"meta-inf/module.mf");

??? if(entry == null) return null;

??? InputStream input = jar.getInputStream(entry);

??? Properties props = new Properties();

??? props.load(input);

??? input.close();

??? return props;

? }

?

? protected JarEntry findModuleEntry(JarFile jar,String expect){

??? Enumeration enum = jar.entries();

??? while(enum.hasMoreElements()){

????? JarEntry entry = (JarEntry)enum.nextElement();

????? String name = entry.getName();

????? if(name.equalsIgnoreCase(expect)){

??????? return entry;

????? }

??? }

??? return null;

? }

?

? public ModuleSpec[] getModuleSpecs(){

??? int size = moduleSpecList.size();

??? ModuleSpec[] specs = new ModuleSpec[size];

??? for(int i =0; i

????? specs[i] = (ModuleSpec)moduleSpecList.get(i);

??? }

??? return specs;??

? }

}

?

ModuleSpec类根据模块信息创建模块实例,它通过两个参数构造,一个是jar文件的File引用,一个是包含了module.mf中数据的Properties。然后根据这两个参数提供的信息通过ModuleClassLoader装载类创建实例。

?

ModuleClassLoader从jar文件中装载类,它从标准的URLClassLoader派生。

?

ModuleRegistry封装了所有的模块的ModuleSpec对象,为用户和一组ModuleSpec对象提供了一个高效的映射关系。

?

最后在ModuleFactory类中,ModuleScanner收集模块,将其注册到ModuleRegistry中,在getInstance中遍历从ModuleRegistry中获得的模块实例,调用每一个模块实例的handles方法来寻找合适的模块。

import java.io.*;

import java.net.*;

import java.util.*;

?

public class ModuleFactory

{

? protected static final ModuleFactory factory = new ModuleFactory();

?

? protected ModuleRegistry registry = new ModuleRegistry();

? protected ModuleScanner scanner = new ModuleScanner();

?

? public void loadModules(File path) throws IOException{

??? scanner.scan(path);

??? register(scanner.getModuleSpecs());

? }

?

? public void loadModules(File[] paths) throws IOException{

??? scanner.scan(paths);

??? register(scanner.getModuleSpecs());

? }

?

? public void register(ModuleSpec spec){

??? registry.registerModule(spec);

? }

?

? public void register(ModuleSpec[] specs){

??? for(int i =0; i

????? registry.registerModule(specs[i]);

??? }

? }

?

? public Object getInstance(Class type, Object key){

??? List modules = registry.getModulesForType(type);

??? int count = modules.size();

??? for(int i=0; i

????? ModuleSpec spec = (ModuleSpec)modules.get(i);

????? ModuleInterface module = spec.getInstance();

????? if(module.handles(key)) return module;

??? }

??? return null;

? }

?

? public static ModuleFactory getInstance(){

??? return factory;

? }

?

?

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有