手写简单IOC
ReflectUtils.java (反射工具类)
package top.icss.ioc; 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.List;
import java.util.jar.JarEntry;
import java.util.jar.JarFile; /**
* @author cd
* @desc 反射工具类
* @create 2020/3/26 11:30
* @since 1.0.0
*/
public class ReflectUtils { /**
* 是否循环迭代
*/
private final static boolean recursive = true; /**
* 扫描 包下面所有Class
* @param packageName 包名称
* @param <T>
* @return
*/
public static <T> List<Class<T>> getClass(String packageName){
List<Class<T>> list = new ArrayList<>(); String packageNamePath = packageName;
packageNamePath = packageNamePath.replace(".", "/"); Enumeration<URL> resources;
try {
//定义一个枚举的集合 并进行循环来处理这个目录下的things
resources = Thread.currentThread().getContextClassLoader().getResources(packageNamePath);
//循环迭代
while (resources.hasMoreElements()){
URL url = resources.nextElement();
//得到协议的名称
String protocol = url.getProtocol();
//如果是以文件的形式保存在服务器上
if("file".equals(protocol)){
System.err.println("file类型的扫描");
String filePath = URLDecoder.decode(url.getFile(), "utf-8");
// 获取此包的目录 建立一个File
File dir = new File(filePath);
list.addAll(getClass(dir, packageName));
}
} } catch (IOException e) {
e.printStackTrace();
} return list;
} /**
* 迭代查找文件类
* @param filePath
* @param packageName
* @param <T>
* @return
*/
private static <T> List<Class<T>> getClass(File filePath, String packageName){
List<Class<T>> classes = new ArrayList<>();
if(!filePath.exists()){
return classes;
} // 如果存在 就获取包下的所有文件 包括目录
File[] files = filePath.listFiles(new FileFilter() {
//自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)
@Override
public boolean accept(File file) {
return (recursive && file.isDirectory())
|| file.getName().endsWith(".class");
}
}); for (File file : files){
// 如果是目录 则继续扫描
if(file.isDirectory()){
classes.addAll(getClass(file, packageName + "." + file.getName()));
}else {
// 如果是java类文件 去掉后面的.class 只留下类名
String fileName = file.getName();
String className = fileName.substring(0, fileName.length() - 6);
className = packageName + "." + className;
try {
//这里用forName有一些不好,会触发static方法,没有使用classLoader的load干净
//Class<T> cls = (Class<T>) Class.forName(className);
Class<T> cls = (Class<T>) Thread.currentThread().getContextClassLoader().loadClass(className);
classes.add(cls);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
} } return classes;
}
public static void main(String[] args) {
getClass("top.icss");
}
}
MyService.java MyAutowired.java (注解类)
package top.icss.ioc.annotation; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* @author cd
* @desc
* @create 2020/3/26 14:57
* @since 1.0.0
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface MyService { String value() default "";
}
package top.icss.ioc.annotation; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* @author cd
* @desc 自动注入
* @create 2020/3/26 14:59
* @since 1.0.0
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface MyAutowired { String value() default "";
}
IocContainer.java (IOC容器)
package top.icss.ioc; import top.icss.ioc.annotation.MyAutowired;
import top.icss.ioc.annotation.MyService; import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap; /**
* @author cd
* @desc ioc容器
* @create 2020/3/26 14:48
* @since 1.0.0
*/
public class IocContainer { private Set<Class<?>> clss = new LinkedHashSet<Class<?>>();
private Map<String, Object> beans = new ConcurrentHashMap<String, Object>(); /**
* 获取类上 @MyService的注解类
*
* @param packageName
*/
public void doScanner(String packageName) {
List<Class<Object>> list = ReflectUtils.getClass(packageName);
for (Class cls : list) {
boolean service = cls.isAnnotationPresent(MyService.class);
if (service) {
clss.add(cls);
}
}
} /**
* 将class中的类实例化,经key-value:类名(小写)-类对象放入ioc字段中
*/
public void doInstance() {
for (Class cls : clss) {
if (cls.isAnnotationPresent(MyService.class)) {
MyService myService = (MyService) cls.getAnnotation(MyService.class);
String beanName = "";
if(cls.isInterface()){
beanName = cls.getName();
}else {
beanName = ("".equals(myService.value().trim())) ? toLowerFirstWord(cls.getSimpleName()) : myService.value();
}
try { Object instance = cls.newInstance();
beans.put(beanName, instance); Class[] interfaces = cls.getInterfaces();
for (Class<?> i:interfaces){
beans.put(i.getName(), instance);
} } catch (Exception e) {
e.printStackTrace();
} }
}
} /**
* 自动化的依赖注入
*/
public void doAutowired(){
if(beans.isEmpty()){
return;
} try {
Set<Map.Entry<String, Object>> entries = beans.entrySet();
for (Map.Entry<String, Object> entry: entries){
Class<?> cls = entry.getValue().getClass(); Field[] fields = cls.getDeclaredFields();
//强制获取私有字段
AccessibleObject.setAccessible(fields,true);
for (Field f: fields){
if(!f.isAnnotationPresent(MyAutowired.class)){
continue;
} MyAutowired myAutowired = f.getAnnotation(MyAutowired.class);
String beanName = "";
Class icls = f.getType();
if(icls.isInterface()){
beanName = icls.getName();
}else {
beanName = ("".equals(myAutowired.value().trim())) ? toLowerFirstWord(icls.getName()) : myAutowired.value();
}
//获取当前类实例
Object obj = entry.getValue();
//容器中获取字段实例
Object value = beans.get(beanName); f.set(obj, value); }
}
} catch (Exception e) {
e.printStackTrace();
}
} /**
* 获取实例
* @param cls
* @param <T>
* @return
*/
public <T> T getBean(Class<?> cls){
MyService myService = cls.getAnnotation(MyService.class);
String beanName = "";
if(cls.isInterface()){
beanName = cls.getName();
}else {
beanName = ("".equals(myService.value().trim())) ? toLowerFirstWord(cls.getSimpleName()) : myService.value();
} return (T) beans.get(beanName);
} /**
* 将字符串首字母转换为小写
* @param name
* @return
*/
private String toLowerFirstWord(String name) {
char[] charArray = name.toCharArray();
charArray[0] += 32;
return String.valueOf(charArray);
}
}
测试
Service1.java Service2.java (接口类)
public interface Service1 { public void print1();
} public interface Service2 {
public void print2();
}
Service1Impl.java Service2Impl.java (实现类)
package top.icss.ioc.test.impl; import top.icss.ioc.annotation.MyAutowired;
import top.icss.ioc.annotation.MyService;
import top.icss.ioc.test.Service1;
import top.icss.ioc.test.Service2; /**
* @author cd
* @desc
* @create 2020/3/26 15:45
* @since 1.0.0
*/
@MyService
public class Service1Impl implements Service1 { @MyAutowired
private Service2 service2; @Override
public void print1() {
service2.print2();
}
}
package top.icss.ioc.test.impl; import top.icss.ioc.annotation.MyAutowired;
import top.icss.ioc.annotation.MyService;
import top.icss.ioc.test.Service1;
import top.icss.ioc.test.Service2; /**
* @author cd
* @desc
* @create 2020/3/26 15:45
* @since 1.0.0
*/
@MyService
public class Service2Impl implements Service2 { @MyAutowired
private Service1 service1; @Override
public void print2() {
System.out.println("print2");
}
}
IocTest.java (测试类)
public class IocTest { public static void main(String[] args) throws InterruptedException {
IocContainer ioc = new IocContainer();
ioc.doScanner("top.icss.ioc.test");
ioc.doInstance();
ioc.doAutowired(); Service2 bean = ioc.getBean(Service2Impl.class);
bean.print2(); }
}
手写简单IOC的更多相关文章
- Spring源码分析 手写简单IOC容器
Spring的两大特性就是IOC和AOP. IOC Container,控制反转容器,通过读取配置文件或注解,将对象封装成Bean存入IOC容器待用,程序需要时再从容器中取,实现控制权由程序员向程序的 ...
- 利用递归,反射,注解等,手写Spring Ioc和Di 底层(分分钟喷倒面试官)了解一下
再我们现在项目中Spring框架是目前各大公司必不可少的技术,而大家都知道去怎么使用Spring ,但是有很多人都不知道SpringIoc底层是如何工作的,而一个开发人员知道他的源码,底层工作原理,对 ...
- 我自横刀向天笑,手写Spring IOC容器,快来Look Look!
目录 IOC分析 IOC是什么 IOC能够带来什么好处 IOC容器是做什么工作的 IOC容器是否是工厂模式的实例 IOC设计实现 设计IOC需要什么 定义接口 一:Bean工厂接口 二:Bean定义的 ...
- 不使用Tomcat,手写简单的web服务
背景: 公司使用的YDB提供了http的查询数据库服务,直接通过url传入sql语句查询数据-_-||.ydb的使用参照:https://www.cnblogs.com/hd-zg/p/7115112 ...
- 手写一个IOC容器
链接:https://pan.baidu.com/s/1MhKJYamBY1ejjjhz3BKoWQ 提取码:e8on 明白什么是IOC容器: IOC(Inversion of Control,控制反 ...
- 手写简单call,apply,bind
分析一下call的使用方法:call是显示绑定this指向,然后第一个参数是你所指向的this对象,后面跟着多个参数,以逗号隔开 function sum(num1,num2){ return num ...
- 手写简单的jq雪花飘落
闲来无事,准备写个雪花飘落的效果,没有写太牛逼的特效,极大的简化了代码量,这样容易读取代码,用起来也很简单,对于那些小白简直是福利啊,简单易读易学.先直接上代码吧,然后再一一讲解,直接复制粘贴就可以拿 ...
- 利用Java手写简单的httpserver
前言: 在看完尚学堂JAVA300中讲解如何实现一个最简单的httpserver部分的视频之后, 一.前置知识 1.HTTP协议 当前互联网网页访问主要采用了B/S的模式,既一个浏览器,一个服务器,浏 ...
- 从零开始手写 spring ioc 框架,深入学习 spring 源码
IoC Ioc 是一款 spring ioc 核心功能简化实现版本,便于学习和理解原理. 创作目的 使用 spring 很长时间,对于 spring 使用非常频繁,实际上对于源码一直没有静下心来学习过 ...
随机推荐
- python猜数字游戏快速求解解决方案
#coding=utf-8 def init_set(): r10=range(10) return [(i, j, k, l) for i in r10 for j in r10 for k in ...
- 基于seo的话 一个页面里的h1标签应该控制在多少个
不能出现多个,一个页面只能出现一次,次数多了就会造成权重分散
- IP地址0.0.0.0表示什么
参考RFC文档: 0.0.0.0/8 - Addresses in this block refer to source hosts on "this"network. Addre ...
- iOS多线程开发之NSOperation
一.什么是NSOperation? NSOperation是苹果提供的一套多线程解决方案.实际上NSOperation是基于GCD更高一层的封装,但是比GCD更加的面向对象.代码可读性更高.可控性更强 ...
- View 属性
关于 View 设置属性的方式: JavaxmlstyledefStyleAttrdefStyleResTheme 关于 defStyleRes 的使用,和在 xml 中声明 style=" ...
- Hihocoder1456 Rikka with Lattice
众所周知,萌萌哒六花不擅长数学,所以勇太给了她一些数学问题做练习,其中有一道是这样的:勇太有一个$n times m$的点阵,他想要从这$n times m$个点中选出三个点 ${A,B,C}$,满足 ...
- WEB端缓存机制
WEB端缓存机制 什么是WEB缓存 Web缓存是指一个Web资源(如html页面,图片,js,数据等)存在于Web服务器和客户端(浏览器)之间的副本.缓存会根据进来的请求保存输出内容的副本:当下一个请 ...
- Java中的成员内部类
*/ * Copyright (c) 2016,烟台大学计算机与控制工程学院 * All rights reserved. * 文件名:text.java * 作者:常轩 * 微信公众号:Worldh ...
- C++走向远洋——21(项目一,三角形,类)
*/ * Copyright (c) 2016,烟台大学计算机与控制工程学院 * All rights reserved. * 文件名:sanjiaoxing.cpp * 作者:常轩 * 微信公众号: ...
- webpack中打包拷贝静态文件CopyWebpackPlugin插件
copyWebpackPlugin: 作用:用于webpack打包时拷贝文件的插件包 安装:npm install copyWebpackPlugin@版本号 使用:// copy custom st ...