大致思路:
注解实现方式:
就是用 反射机制. 获取指定的包下使用了注解的类,存储在一个map容器, 然后获取map容器下类的属性, 利用反射给使用了注解的属性赋值对象.
xml实现方式:
使用xml解析配置文件+反射机制
此处使用注解方式
1. 引入相关jar
- <dependencies>
- <!-- 引入Spring-AOP等相关Jar -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-core</artifactId>
- <version>3.0.6.RELEASE</version>
- </dependency>
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-context</artifactId>
- <version>3.0.6.RELEASE</version>
- </dependency>
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-aop</artifactId>
- <version>3.0.6.RELEASE</version>
- </dependency>
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-orm</artifactId>
- <version>3.0.6.RELEASE</version>
- </dependency>
- <dependency>
- <groupId>org.aspectj</groupId>
- <artifactId>aspectjrt</artifactId>
- <version>1.6.1</version>
- </dependency>
- <dependency>
- <groupId>aspectj</groupId>
- <artifactId>aspectjweaver</artifactId>
- <version>1.5.3</version>
- </dependency>
- <dependency>
- <groupId>cglib</groupId>
- <artifactId>cglib</artifactId>
- <version>2.1_2</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/com.mchange/c3p0 -->
- <dependency>
- <groupId>com.mchange</groupId>
- <artifactId>c3p0</artifactId>
- <version>0.9.5.2</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
- <dependency>
- <groupId>mysql</groupId>
- <artifactId>mysql-connector-java</artifactId>
- <version>5.1.37</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/dom4j/dom4j -->
- <dependency>
- <groupId>dom4j</groupId>
- <artifactId>dom4j</artifactId>
- <version>1.6.1</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/commons-lang/commons-lang -->
- <dependency>
- <groupId>commons-lang</groupId>
- <artifactId>commons-lang</artifactId>
- <version>2.6</version>
- </dependency>
- </dependencies>
2. 定义注解类ExtService是注解类的, ExtResource是注解属性的
模拟@Service 和@Resource注解


3.定义一个借口

4. 接口和使用注解的实现类


5. 自定义实现ClassPathXmlApplicationContext

6.测试

7. 结果

8. 工具类
- package com.learn.utils;
- import java.io.File;
- import java.io.FileFilter;
- import java.io.IOException;
- import java.net.JarURLConnection;
- import java.net.URL;
- import java.net.URLDecoder;
- import java.util.ArrayList;
- import java.util.Enumeration;
- import java.util.LinkedHashSet;
- import java.util.List;
- import java.util.Set;
- import java.util.jar.JarEntry;
- import java.util.jar.JarFile;
- public class ClassUtil {
- /**
- * 取得某个接口下所有实现这个接口的类
- */
- public static List<Class> getAllClassByInterface(Class c) {
- List<Class> returnClassList = null;
- if (c.isInterface()) {
- // 获取当前的包名
- String packageName = c.getPackage().getName();
- // 获取当前包下以及子包下所以的类
- List<Class<?>> allClass = getClasses(packageName);
- if (allClass != null) {
- returnClassList = new ArrayList<Class>();
- for (Class classes : allClass) {
- // 判断是否是同一个接口
- if (c.isAssignableFrom(classes)) {
- // 本身不加入进去
- if (!c.equals(classes)) {
- returnClassList.add(classes);
- }
- }
- }
- }
- }
- return returnClassList;
- }
- /*
- * 取得某一类所在包的所有类名 不含迭代
- */
- public static String[] getPackageAllClassName(String classLocation, String packageName) {
- // 将packageName分解
- String[] packagePathSplit = packageName.split("[.]");
- String realClassLocation = classLocation;
- int packageLength = packagePathSplit.length;
- for (int i = 0; i < packageLength; i++) {
- realClassLocation = realClassLocation + File.separator + packagePathSplit[i];
- }
- File packeageDir = new File(realClassLocation);
- if (packeageDir.isDirectory()) {
- String[] allClassName = packeageDir.list();
- return allClassName;
- }
- return null;
- }
- /**
- * 从包package中获取所有的Class
- *
- * @param pack
- * @return
- */
- public static List<Class<?>> getClasses(String packageName) {
- // 第一个class类的集合
- List<Class<?>> classes = new ArrayList<Class<?>>();
- // 是否循环迭代
- boolean recursive = true;
- // 获取包的名字 并进行替换
- String packageDirName = packageName.replace('.', '/');
- // 定义一个枚举的集合 并进行循环来处理这个目录下的things
- Enumeration<URL> dirs;
- try {
- dirs = Thread.currentThread().getContextClassLoader().getResources(packageDirName);
- // 循环迭代下去
- while (dirs.hasMoreElements()) {
- // 获取下一个元素
- URL url = dirs.nextElement();
- // 得到协议的名称
- String protocol = url.getProtocol();
- // 如果是以文件的形式保存在服务器上
- if ("file".equals(protocol)) {
- // 获取包的物理路径
- String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
- // 以文件的方式扫描整个包下的文件 并添加到集合中
- findAndAddClassesInPackageByFile(packageName, filePath, recursive, classes);
- } else if ("jar".equals(protocol)) {
- // 如果是jar包文件
- // 定义一个JarFile
- JarFile jar;
- try {
- // 获取jar
- jar = ((JarURLConnection) url.openConnection()).getJarFile();
- // 从此jar包 得到一个枚举类
- Enumeration<JarEntry> entries = jar.entries();
- // 同样的进行循环迭代
- while (entries.hasMoreElements()) {
- // 获取jar里的一个实体 可以是目录 和一些jar包里的其他文件 如META-INF等文件
- JarEntry entry = entries.nextElement();
- String name = entry.getName();
- // 如果是以/开头的
- if (name.charAt(0) == '/') {
- // 获取后面的字符串
- name = name.substring(1);
- }
- // 如果前半部分和定义的包名相同
- if (name.startsWith(packageDirName)) {
- int idx = name.lastIndexOf('/');
- // 如果以"/"结尾 是一个包
- if (idx != -1) {
- // 获取包名 把"/"替换成"."
- packageName = name.substring(0, idx).replace('/', '.');
- }
- // 如果可以迭代下去 并且是一个包
- if ((idx != -1) || recursive) {
- // 如果是一个.class文件 而且不是目录
- if (name.endsWith(".class") && !entry.isDirectory()) {
- // 去掉后面的".class" 获取真正的类名
- String className = name.substring(packageName.length() + 1, name.length() - 6);
- try {
- // 添加到classes
- classes.add(Class.forName(packageName + '.' + className));
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- return classes;
- }
- /**
- * 以文件的形式来获取包下的所有Class
- *
- * @param packageName
- * @param packagePath
- * @param recursive
- * @param classes
- */
- public static void findAndAddClassesInPackageByFile(String packageName, String packagePath, final boolean recursive,
- List<Class<?>> classes) {
- // 获取此包的目录 建立一个File
- File dir = new File(packagePath);
- // 如果不存在或者 也不是目录就直接返回
- if (!dir.exists() || !dir.isDirectory()) {
- return;
- }
- // 如果存在 就获取包下的所有文件 包括目录
- File[] dirfiles = dir.listFiles(new FileFilter() {
- // 自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)
- public boolean accept(File file) {
- return (recursive && file.isDirectory()) || (file.getName().endsWith(".class"));
- }
- });
- // 循环所有文件
- for (File file : dirfiles) {
- // 如果是目录 则继续扫描
- if (file.isDirectory()) {
- findAndAddClassesInPackageByFile(packageName + "." + file.getName(), file.getAbsolutePath(), recursive,
- classes);
- } else {
- // 如果是java类文件 去掉后面的.class 只留下类名
- String className = file.getName().substring(0, file.getName().length() - 6);
- try {
- // 添加到集合中去
- classes.add(Class.forName(packageName + '.' + className));
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
以上为个人经验,希望能给大家一个参考,也希望大家多多支持w3xue。