JFinal自动扫描路由配置,支持jar包

本文介绍了一种利用自动扫描技术优化JFinal框架路径配置的方法,通过自动扫描类和路径扫描,实现了对jar包和类路径下所有路由配置信息的高效获取与配置,简化了架构师的工作流程。

JFinal原有的路径配置方式是很简洁的,我们可以方便的通过两种方式对路径进行配置。1.通过route.add(String ControlKey,Class<?> clz) 2.另一种可以把路径信息的配置拿到单独的一个类中,该类必须继承Routes,然后再通过route.add(Routes rt)方式配置路径映射。

奈何现状需求很别扭,需要在相应的插件信息中读取路径信息,考虑插件的定制性,必须做到可插拔,插件以jar包形式放到主体工程中,因此,相应的路径信息配置不能写死,通过自动扫描的方式可以方便的实现可定制。

一、自动扫描类(@绝望的八皮 提供的优化,感谢!)

 

package net.zfsy.db.plugin;

import java.io.File;
import java.io.FilenameFilter;
import java.io.IOException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

import com.jfinal.core.Controller;

public class ClassSearcher {
	static URL classPathUrl = ClassSearcher.class.getResource("/");
	static String lib = new File(classPathUrl.getFile()).getParent() + "/lib/";

	/**
	 * 递归查找文件
	 * 
	 * @param baseDirName
	 *            查找的文件夹路径
	 * @param targetFileName
	 *            需要查找的文件名
	 */
	private static List<String> findFiles(String baseDirName,
			String targetFileName) {
		/**
		 * 算法简述: 从某个给定的需查找的文件夹出发,搜索该文件夹的所有子文件夹及文件,
		 * 若为文件,则进行匹配,匹配成功则加入结果集,若为子文件夹,则进队列。 队列不空,重复上述操作,队列为空,程序结束,返回结果。
		 */
		List<String> classFiles = new ArrayList<String>();
		String tempName = null;
		// 判断目录是否存在
		File baseDir = new File(baseDirName);
		if (!baseDir.exists() || !baseDir.isDirectory()) {
			System.err.println("search error:" + baseDirName + "is not a dir!");
		} else {
			String[] filelist = baseDir.list();
			for (int i = 0; i < filelist.length; i++) {
				File readfile = new File(baseDirName + File.separator
						+ filelist[i]);
				if (!readfile.isDirectory()) {
					tempName = readfile.getName();
					if (ClassSearcher.wildcardMatch(targetFileName, tempName)) {
						String classname;
						String tem = readfile.getAbsoluteFile().toString()
								.toString().replaceAll("\\\\", "/");
						classname = tem.substring(tem.indexOf("/classes")
								+ "/classes".length(), tem.indexOf(".class"));
						if (classname.startsWith("/")) {
							classname = classname.substring(classname
									.indexOf("/") + 1);
						}
						classname = className(classname, "/classes");
						classFiles.add(classname);
					}
				} else if (readfile.isDirectory()) {
					classFiles.addAll(findFiles(baseDirName + File.separator
							+ filelist[i], targetFileName));
				}
			}
		}
		return classFiles;
	}

	/**
	 * 查找jar包中的class
	 * 
	 * @param baseDirName
	 *            jar路径
	 * @param includeJars
	 * @param jarFileURL
	 *            jar文件地址 <a href="http://my.oschina.net/u/556800"
	 *            target="_blank" rel="nofollow">@return</a>
	 */
	public static List<String> findjarFiles(String baseDirName,
			final List<String> includeJars) {
		List<String> classFiles = new ArrayList<String>();
		try {
			// 判断目录是否存在
			File baseDir = new File(baseDirName);
			if (!baseDir.exists() || !baseDir.isDirectory()) {
				System.out.println("file serach error:" + baseDirName
						+ "is not a dir!");
			} else {
				String[] filelist = baseDir.list(new FilenameFilter() {
					@Override
					public boolean accept(File dir, String name) {
						return includeJars.contains(name);
					}
				});
				for (int i = 0; i < filelist.length; i++) {
					JarFile localJarFile = new JarFile(new File(baseDirName
							+ File.separator + filelist[i]));
					Enumeration<JarEntry> entries = localJarFile.entries();
					while (entries.hasMoreElements()) {
						JarEntry jarEntry = entries.nextElement();
						String entryName = jarEntry.getName();
						if (!jarEntry.isDirectory()
								&& entryName.endsWith(".class")) {
							String className = entryName.replaceAll("/", ".")
									.substring(0, entryName.length() - 6);
							classFiles.add(className);
						}
					}
					localJarFile.close();
				}
			}

		} catch (IOException e) {
			e.printStackTrace();
		}
		return classFiles;

	}

	public static List<Class> findInClasspathAndInJars(Class clazz,
			List<String> includeJars) {
		List<String> classFileList = findFiles(classPathUrl.getFile(),
				"*.class");
		classFileList.addAll(findjarFiles(lib, includeJars));
		return extraction(clazz, classFileList);
	}

	private static List<Class> extraction(Class clazz,
			List<String> classFileList) {
		List<Class> classList = new ArrayList<Class>();
		for (String classFile : classFileList) {
			try {
				Class<?> classInFile = Class.forName(classFile);
				if (classInFile.getSuperclass() == clazz) {
					classList.add(classInFile);
				}
			} catch (ClassNotFoundException e) {
				e.printStackTrace();
			}
		}

		return classList;
	}

	private static String className(String classFile, String pre) {
		String objStr = classFile.replaceAll("\\\\", "/");
		return objStr.replaceAll("/", ".");
	}

	/**
	 * 通配符匹配
	 * 
	 * @param pattern
	 *            通配符模式
	 * @param str
	 *            待匹配的字符串 <a href="http://my.oschina.net/u/556800"
	 *            target="_blank" rel="nofollow">@return</a>
	 *            匹配成功则返回true,否则返回false
	 */
	private static boolean wildcardMatch(String pattern, String str) {
		int patternLength = pattern.length();
		int strLength = str.length();
		int strIndex = 0;
		char ch;
		for (int patternIndex = 0; patternIndex < patternLength; patternIndex++) {
			ch = pattern.charAt(patternIndex);
			if (ch == '*') {
				// 通配符星号*表示可以匹配任意多个字符
				while (strIndex < strLength) {
					if (wildcardMatch(pattern.substring(patternIndex + 1),
							str.substring(strIndex))) {
						return true;
					}
					strIndex++;
				}
			} else if (ch == '?') {
				// 通配符问号?表示匹配任意一个字符
				strIndex++;
				if (strIndex > strLength) {
					// 表示str中已经没有字符匹配?了。
					return false;
				}
			} else {
				if ((strIndex >= strLength) || (ch != str.charAt(strIndex))) {
					return false;
				}
				strIndex++;
			}
		}
		return (strIndex == strLength);
	}

	public static List<Class> findInClasspath(Class<Controller> clazz) {
		List<String> classFileList = findFiles(classPathUrl.getFile(),
				"*.class");
		return extraction(clazz, classFileList);
	}

	public static List<Class> findInJar(Class clazz, List<String> includeJars) {
		List<String> classFileList = findjarFiles(lib, includeJars);
		return extraction(clazz, classFileList);
	}
}
二、路径扫描

package net.zfsy.routes.plugin;

import java.util.ArrayList; import java.util.List;

import net.zfsy.db.plugin.ClassSearcher;

import org.slf4j.Logger; import org.slf4j.LoggerFactory;

import com.jfinal.config.Routes;

 

 

public class RoutsAutoBind{  private static Logger logger = LoggerFactory.getLogger(RoutsAutoBind.class);  private static  String[] includeJars;    @SuppressWarnings("static-access")  public RoutsAutoBind(String[] includeJars){   this.includeJars = includeJars;  }  @SuppressWarnings("rawtypes")  public   static List<Class> getClassList(){   List<String> jars = new ArrayList<String>();   for(String jar : includeJars){    jars.add(jar);   }   return ClassSearcher.findInClasspathAndInJars(Routes.class, jars);    }  @SuppressWarnings("rawtypes")  public static void regist(Routes route){     List<Class> clzz = getClassList(); //获取jar中routes的class   for (Class clz : clzz) {    try {     route.add((Routes) clz.newInstance());     logger.debug("routes regist:");    } catch (InstantiationException e) {     e.printStackTrace();    } catch (IllegalAccessException e) {     e.printStackTrace();    }   }  }    public static void main(String[] args){

 } }


三、全局config的引用

 

/*
	 * 配置访问路由
	 * 
	 * @see com.jfinal.config.JFinalConfig#configRoute(com.jfinal.config.Routes)
	 */
	@SuppressWarnings("static-access")
	@Override
	public void configRoute(Routes route) {
		RoutsAutoBind rd = new RoutsAutoBind(PLUGINJARS);// 相应的jar包名称
		rd.regist(route);
	}

这样就可以扫描到jar包和classpath下所有路由配置的映射信息了。

转载于:https://my.oschina.net/mousai/blog/81991

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值