如何使用Spring自定义实现IOC和依赖注入

如何使用Spring自定义实现IOC和依赖注入

本篇内容主要讲解“如何使用Spring自定义实现IOC和依赖注入”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何使用Spring自定义实现IOC和依赖注入”吧!

目录
  • 大致思路:

    • 注解实现方式:

    • xml实现方式:

  • 1. 引入相关jar

    • 2. 定义注解类ExtService是注解类的, ExtResource是注解属性的

      • 3.定义一个借口

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

          • 5. 自定义实现ClassPathXmlApplicationContext

            • 6.测试

              • 7. 结果

                • 8. 工具类

                  大致思路:

                  注解实现方式:

                  就是用 反射机制. 获取指定的包下使用了注解的类,存储在一个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注解

                  如何使用Spring自定义实现IOC和依赖注入

                  如何使用Spring自定义实现IOC和依赖注入

                  3.定义一个借口

                  如何使用Spring自定义实现IOC和依赖注入

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

                  如何使用Spring自定义实现IOC和依赖注入

                  如何使用Spring自定义实现IOC和依赖注入

                  5. 自定义实现ClassPathXmlApplicationContext

                  如何使用Spring自定义实现IOC和依赖注入

                  6.测试

                  如何使用Spring自定义实现IOC和依赖注入

                  7. 结果

                  如何使用Spring自定义实现IOC和依赖注入

                  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();
                  				}
                  			}
                  		}
                  	}
                  }

                  到此,相信大家对“如何使用Spring自定义实现IOC和依赖注入”有了更深的了解,不妨来实际操作一番吧!这里是亿速云网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!