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的更多相关文章

  1. Spring源码分析 手写简单IOC容器

    Spring的两大特性就是IOC和AOP. IOC Container,控制反转容器,通过读取配置文件或注解,将对象封装成Bean存入IOC容器待用,程序需要时再从容器中取,实现控制权由程序员向程序的 ...

  2. 利用递归,反射,注解等,手写Spring Ioc和Di 底层(分分钟喷倒面试官)了解一下

    再我们现在项目中Spring框架是目前各大公司必不可少的技术,而大家都知道去怎么使用Spring ,但是有很多人都不知道SpringIoc底层是如何工作的,而一个开发人员知道他的源码,底层工作原理,对 ...

  3. 我自横刀向天笑,手写Spring IOC容器,快来Look Look!

    目录 IOC分析 IOC是什么 IOC能够带来什么好处 IOC容器是做什么工作的 IOC容器是否是工厂模式的实例 IOC设计实现 设计IOC需要什么 定义接口 一:Bean工厂接口 二:Bean定义的 ...

  4. 不使用Tomcat,手写简单的web服务

    背景: 公司使用的YDB提供了http的查询数据库服务,直接通过url传入sql语句查询数据-_-||.ydb的使用参照:https://www.cnblogs.com/hd-zg/p/7115112 ...

  5. 手写一个IOC容器

    链接:https://pan.baidu.com/s/1MhKJYamBY1ejjjhz3BKoWQ 提取码:e8on 明白什么是IOC容器: IOC(Inversion of Control,控制反 ...

  6. 手写简单call,apply,bind

    分析一下call的使用方法:call是显示绑定this指向,然后第一个参数是你所指向的this对象,后面跟着多个参数,以逗号隔开 function sum(num1,num2){ return num ...

  7. 手写简单的jq雪花飘落

    闲来无事,准备写个雪花飘落的效果,没有写太牛逼的特效,极大的简化了代码量,这样容易读取代码,用起来也很简单,对于那些小白简直是福利啊,简单易读易学.先直接上代码吧,然后再一一讲解,直接复制粘贴就可以拿 ...

  8. 利用Java手写简单的httpserver

    前言: 在看完尚学堂JAVA300中讲解如何实现一个最简单的httpserver部分的视频之后, 一.前置知识 1.HTTP协议 当前互联网网页访问主要采用了B/S的模式,既一个浏览器,一个服务器,浏 ...

  9. 从零开始手写 spring ioc 框架,深入学习 spring 源码

    IoC Ioc 是一款 spring ioc 核心功能简化实现版本,便于学习和理解原理. 创作目的 使用 spring 很长时间,对于 spring 使用非常频繁,实际上对于源码一直没有静下心来学习过 ...

随机推荐

  1. springboot oauth 鉴权之——授权码authorization_code鉴权

    近期一直在研究鉴权方面的各种案例,这几天有空,写一波总结及经验. 第一步:什么是 OAuth鉴权 OAuth2是工业标准的授权协议.OAuth2取代了在2006创建的原始OAuthTM协议所做的工作. ...

  2. 每日一译系列-模块化css怎么玩(译文)

    原文链接:How Css Modules Work 原文作者是Preact的作者 这是一篇关于如何使用Css Modules的快速介绍,使用到的工具是Webpack吊炸的css-loader 首先,我 ...

  3. 用RecyclerView做一个小清新的Gallery效果 - Ryan Lee的博客

    一.简介 RecyclerView现在已经是越来越强大,且不说已经被大家用到滚瓜烂熟的代替ListView的基础功能,现在RecyclerView还可以取代ViewPager实现Banner效果,当然 ...

  4. 一位资深程序员大牛推荐的Java技术学习路线图

    Web应用,最常见的研发语言是Java和PHP. 后端服务,最常见的研发语言是Java和C/C++. 大数据,最常见的研发语言是Java和Python. 可以说,Java是现阶段中国互联网公司中,覆盖 ...

  5. DIV的失去焦点(blur)实现

    用防抖实现DIV鼠标移出消失   由于div标签本身不支持onblur事件,所以对于点击一个按钮弹出的div,我们想要当这个div失去焦点的时候,让它消失不能使用的onblur来实现.  但是可以利用 ...

  6. 关于nw的简单应用

    最近使用到了桌面开发应用nw.js.进行简单的介绍一下,基本用法 nwjs实际上是基于node js的,支持node js的所有api 中文官网https://nwjs.org.cn/ 第一步.在官网 ...

  7. iview中select搜索

    https://www.jianshu.com/p/1c40d7cc440e https://www.cnblogs.com/yun1108/p/10967735.html https://blog. ...

  8. Flutter保持页面状态AutomaticKeepAliveClientMixin

    使用bottomNavigationBar切换底部tab,再切换回来就会丢失之前的状态(重新渲染列表,丢失滚动条位置). 解决方法 使用 AutomaticKeepAliveClientMixin 重 ...

  9. ios background task

    今天要实现一个需求,当用户触摸HOME键,将应用切换到后台时,启动自动备份的任务.这涉及到ios的后台任务处理,本文简单总结一下 首先,ios app有5种状态,分别是:not running, in ...

  10. 2020ubuntu1804server编译安装redis5笔记(二)配置redis

    前一篇笔记记录了ubuntu1804server编译安装redis5,接下来要配置redis5了 网址:https://www.cnblogs.com/qumogu/p/12435694.html 第 ...