Spring中我们用到的功能实现:基于注解的Ioc自动装配

Easter79
• 阅读 510

我们要完成自动装配,那么就要有一个存放bean对象的容器,然后要有装配的注解,那么哪些类该被存到容器呢,在spring中我们使用过@Service、@Resource等,看下面的代码,你也可以做到。

来看看这是一个简单的容器接口

/**
 * 容器接口
 * @author:rex
 * @create_time:2014-6-26
 * @version:V1.0
 */
public interface Container {

    Object getBean(String name, BeanType beanType);
    
    Object getBean(Class<?> type, BeanType beanType);
    
    Set<?> getBeanNames();
    
    Collection<?> getBeans();
    
    boolean hasBean(Class<?> clazz);
    
    boolean hasBean(String name);
    
    void registBean(Class<?> clazz);

    void initWired();
    
}

这个容器提供了基础的存取方法,分别是获取bean对象和注册、是否包含bean,还有一个初始化的方法。

接下来我们来为容器做一个基本的实现。

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

import com.biezhi.ioc.BeanType;
import com.biezhi.ioc.Container;
import com.biezhi.ioc.anntation.Autowired;

/**
 * 默认的bean容器实现
 * @author:rex
 * @create_time:2014-6-26
 * @version:V1.0
 */
public class DefaultContainerImpl implements Container {

    //存放bean的容器
    private final Map<String, Object> beansMap = new HashMap<String, Object>();
    
    public DefaultContainerImpl() {
        //初始化加载bean
        ContainerLoader c = new ContainerLoader(this);
        c.init();
    }
    
    @Override
    public Object getBean(String name, BeanType beanType) {
        try {
            if(beanType == BeanType.NEW)
                return Class.forName(name).newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return beansMap.get(name);
    }
    
    @Override
    public Object getBean(Class<?> type, BeanType beanType) {
        try {
            if(beanType == BeanType.NEW)
                return type.newInstance();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        Iterator<Object> it = this.beansMap.values().iterator();
        while(it.hasNext()){
            Object obj = it.next();
            if(type.isAssignableFrom(obj.getClass())){
                return obj;
            }
        }
        return null;
    }

    @Override
    public Set<?> getBeanNames(){
        return beansMap.keySet();
    }
    
    @Override
    public Collection<?> getBeans(){
        return beansMap.values();
    }

    @Override
    public boolean hasBean(Class<?> clz) {
        if(null != this.getBean(clz, null)){
            return true;
        }
        return false;
    }
    
    @Override
    public boolean hasBean(String name){
        if(null != this.getBean(name, null)){
            return true;
        }
        return false;
    }

    /**
     * 注册一个bean对象到容器里
     */
    @Override
    public void registBean(Class<?> clazz){
        String name = clazz.getCanonicalName();
        try {
            if(!Modifier.isAbstract(clazz.getModifiers()) && 
                !Modifier.isInterface(clazz.getModifiers())){
                Object obj = clazz.newInstance();
                beansMap.put(name, obj);
            }
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } 
    }
    
    /**
     * 初始化注入
     */
    @Override
    public void initWired(){
        Iterator<Object> it = this.beansMap.values().iterator();
        try {
            while(it.hasNext()){
                Object obj = it.next();
                Field[] fields = obj.getClass().getDeclaredFields();
                for(Field field : fields){
                    Autowired autowired = 
                        field.getAnnotation(Autowired.class);
                    if(null != autowired){
                        //要注入的字段
                        Object wiredField = 
                            this.getBean(field.getType(), null);
                        if(null == wiredField){
                                   throw new RuntimeException("Unable to load "+field.getType().getCanonicalName()+"!");
                        }
                        boolean accessible = field.isAccessible();
                        field.setAccessible(true);
                        field.set(obj, wiredField);
                        field.setAccessible(accessible);
                    }
                }
            }
        } catch (SecurityException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } 
    }
}

    在构造器里将扫描到的类加载到容器里,然后提供注册bean和获取bean的方法。

import java.io.File;
import java.io.FileFilter;
import java.util.HashSet;
import java.util.Set;

import com.biezhi.ioc.Container;
import com.biezhi.ioc.anntation.Service;
import com.biezhi.ioc.util.ClassHelper;

/**
 * 加载容器bean
 * @author:rex
 * @create_time:2014-6-26
 * @version:V1.0
 */
public class ContainerLoader {
    
    private Container container;
    
    public ContainerLoader(Container container) {
        this.container = container;
    }
    
    public void init(){
        //加载要扫描的包,这里可以使用配置文件,我们就默认扫描所有类
        Set<String> packages = getPackages();
        for(String pack : packages){
            scanPack(pack);
        }
        //初始化注入
        container.initWired();
    }
    
    private void scanPack(String pack){
        Set<Class<?>> classes = ClassHelper.scanPackage(pack);
        for(Class<?> clazz : classes){
            // 这里我只把带有@Service注解的存进去了,你也可以存其他的或者全部
            Service service = clazz.getAnnotation(Service.class);
            if(null != service){
                //将扫描到的对象保存到容器中
                container.registBean(clazz);
            }
        }
    }
    
    /**
     * 获取当前classes的包名称
     * @author:rex  
     * @return
     */
    private Set<String> getPackages(){
        Set<String> packages = new HashSet<String>();
        String appPath = ContainerLoader.class.getResource("/").getPath();
        File classDir = new File(appPath);
        // 如果存在 就获取包下的所有文件 包括目录
        File[] dirfiles = classDir.listFiles(new FileFilter() {
            public boolean accept(File file) {
                return file.isDirectory();
            }
        });
        for(File f : dirfiles){
            packages.add(f.getName());
        }
        return packages;
    }
}

    这个类是加载需要的类文件。还有几个代码文件没有贴出来,想看代码的等会打包自己看。

    接下来我们看看这个测试,

@Service
public class A {

    String name = "菊花";
    
    public void say(){
        System.out.println("hello, I,m rex !");
    }
}

@Service
public class B {

    @Autowired
    private A a;
    
    private String qq = "3838438";
    
    public void hehe(){
        a.say();
        System.out.println("请问您是" + a.name + "吗?");
    }
    
    public String getQq(){
        return this.qq;
    }
}

public class Test {

    public static void main(String[] args) {
        Container c = new DefaultContainerImpl();
        c.initWired();
        //System.out.println(c.getBeanNames());
        B b = (B) c.getBean(B.class, BeanType.SINGLE);
        b.hehe();
        System.out.println(b.getQq());
        System.out.println("==================");
        B b2 = (B) c.getBean(B.class, BeanType.NEW);
        b2.hehe();
    }
}

    运行结果:

hello, I,m rex !
请问您是菊花吗?
3838438
==================
Exception in thread "main" java.lang.NullPointerException
    at com.biezhi.ioc.test.B.hehe(B.java:15)
    at com.biezhi.ioc.test.Test.main(Test.java:18)

好了,这样就基本完成了一个简单的ioc自动装配。有喜欢的朋友可以参考代码。点击下载

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Wesley13 Wesley13
2年前
SSM的基本知识点
\SpringSpring框架是一个容器,它的主要作用是创建和管理对象。Spring框架实现了解耦。Spring框架通过DI实现了IoC。关于Spring框架,必须掌握和理解的:1. Spring的基本概念与特性;2. 常用的注解:通用的4个注解,自动装配的注解;3.
Stella981 Stella981
2年前
Spring 的 IOC 容器
一.BeanFactory 1\.在spring中,最基本的IOC容器接口是BeanFactory这个接口为具体的IOC容器的实现做了最基本的功能规定。  2\.在BeanFactory只对IOC容器的基本行为做了定义,根本不关心你的bean是怎样定义怎样加载的;XmlBeanFactory就是针对最
Wesley13 Wesley13
2年前
Spring常用注解
使用注解来构造IoC容器用注解来向Spring容器注册Bean。需要在applicationContext.xml中注册<context:componentscanbasepackage”pagkage1\,pagkage2,…,pagkageN\”/。如:在basepackage指明一个包
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
2年前
Spring scope属性详解
scope用来声明IOC容器中的对象应该处的限定场景或者说该对象的存活空间,即在IOC容器在对象进入相应的scope之前,生成并装配这些对象,在该对象不再处于这些scope的限定之后,容器通常会销毁这些对象。打个比方吧!我们都是处在社会(容器)中,如果把中学教师作为一个类定义,那么当容器初始化这些类之后,中学教师只能局限在中学这个场景中,中学,就可以
Easter79 Easter79
2年前
Springboot启动扩展点超详细总结,再也不怕面试官问了
1.背景Spring的核心思想就是容器,当容器refresh的时候,外部看上去风平浪静,其实内部则是一片惊涛骇浪,汪洋一片。Springboot更是封装了Spring,遵循约定大于配置,加上自动装配的机制。很多时候我们只要引用了一个依赖,几乎是零配置就能完成一个功能的装配。我非常喜欢这种自动装配的机制,所以在自己开发中间件和公共依赖工具的时
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
Easter79
Easter79
Lv1
今生可爱与温柔,每一样都不能少。
文章
2.8k
粉丝
5
获赞
1.2k