基于注解+反射手敲SpringIOC容器

目录

1.什么是IOC

2.IOC入门小练习(开口小菜)

2.1 基于properties配置文件方式

 2.2练习二判断一个类和属性是否包含某一个注解以及获取注解的value

3,手敲IOC容器以及DI(正片开始)


1.什么是IOC

IOC(Inversion of Control),控制反转。

就是指将对象的创建,对象的存储(map),对象的管理(依赖查找,依赖注入)交给了spring容器。

最要要的作用就是解耦合

2.IOC入门小练习(开口小菜)

2.1 基于properties配置文件方式

properties配置文件

pojo=cn.tedu.test.Pojo

上代码 

具体的实现类

package cn.tedu.myspring_two;

import cn.tedu.test.Pojo;

import java.io.FileInputStream;
import java.util.Properties;

/**
 * 基于properties配置文件方式 获取bean
 */
public class TestSpring {
    public static void main(String[] args) {
        try {

            Properties properties = new Properties();//properties属于map集合 存方的都是key=value的形式
            properties.load(new FileInputStream("src\\cn\\tedu\\myspring_two\\spring.properties"));//读入properties文件
            String pojo = properties.getProperty("pojo");//获取key为pojo的对象
            Class<?> aClass = Class.forName(pojo);//通过反射获取字节码文件
            Object o = aClass.newInstance();//将字节码文件实例化
            Pojo o1 = (Pojo) o;//强转成为子类类型
            System.out.println(o1);
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

 2.2练习二判断一个类和属性是否包含某一个注解以及获取注解的value

上代码

自定义Aut注解

package cn.tedu.myspring_three;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Aut {
    String value()default "Aut";
}

自定义Service注解

package cn.tedu.myspring_three;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)

public @interface Service {
    String value() default "service";
}

创建测试类SpringTest

package cn.tedu.myspring_three;

import java.lang.reflect.Field;

public class SpringTest {
     public void method() throws ClassNotFoundException {
         Class<?> aClass = Class.forName("cn.tedu.myspring_three.Student");//获取字节码文件
         if (aClass.isAnnotationPresent(Service.class)) {//判断字节码是否办函Service类
             System.out.println("这个类包含Service注解");
         }
         Field[] declaredFields = aClass.getDeclaredFields();//获取私有属性
         for (Field declaredField : declaredFields) {//遍历私有属性
             if (declaredField.isAnnotationPresent(Aut.class)) {//判断属性上是否包含Aut注解
                 System.out.println(declaredField.getName()+"属性包含"+"Aut"+"注解");
             }else {
                 System.out.println(declaredField.getName() + "属性" + "不包含Aut注解");

             }
         }
     }

    public static void main(String[] args) throws ClassNotFoundException {//测试
        new SpringTest().method();
    }
}

3,手敲IOC容器以及DI(正片开始)

创建自定义注解CGB

package cn.tedu.myspring;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface CGB {
    String value() default"";
}

自定义value注解

package cn.tedu.myspring;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;


@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface value {
    String value()default "";
}

创建工具类MyTools,用来获取包里面的所有类的字节码

package cn.tedu.myspring;
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.Enumeration;
import java.util.LinkedHashSet;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * 本类是一个扫描报的工具类
 */
public class MyTools {

    public static Set<Class<?>> getClasses(String pack) {

        // 第一个class类的集合
        Set<Class<?>> classes = new LinkedHashSet<Class<?>>();
        // 是否循环迭代
        boolean recursive = true;
        // 获取包的名字 并进行替换
        String packageName = pack;
        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");
                    // 以文件的方式扫描整个包下的文件 并添加到集合中
                    findClassesInPackageByFile(packageName, filePath, recursive, classes);
                } else if ("jar".equals(protocol)) {
                    // 如果是jar包文件
                    // 定义一个JarFile
                    System.out.println("jar类型的扫描");
                    JarFile jar;
                    try {
                        // 获取jar
                        jar = ((JarURLConnection) url.openConnection()).getJarFile();
                        // 从此jar包 得到一个枚举类
                        Enumeration<JarEntry> entries = jar.entries();
                        findClassesInPackageByJar(packageName, entries, packageDirName, recursive, classes);
                    } catch (IOException e) {
                        // log.error("在扫描用户定义视图时从jar包获取文件出错");
                        e.printStackTrace();
                    }
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return classes;
    }

    private static void findClassesInPackageByJar(String packageName, Enumeration<JarEntry> entries, String packageDirName, final boolean recursive, Set<Class<?>> classes) {
        // 同样的进行循环迭代
        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) {
                            // .error("添加用户自定义视图类错误 找不到此类的.class文件");
                            e.printStackTrace();
                        }
                    }
                }
            }
        }
    }

    private static void findClassesInPackageByFile(String packageName, String packagePath, final boolean recursive, Set<Class<?>> classes) {
        // 获取此包的目录 建立一个File
        File dir = new File(packagePath);
        // 如果不存在或者 也不是目录就直接返回
        if (!dir.exists() || !dir.isDirectory()) {
            // log.warn("用户定义包名 " + packageName + " 下没有任何文件");
            return;
        }
        // 如果存在 就获取包下的所有文件 包括目录
        File[] dirfiles = dir.listFiles(new FileFilter() {
            // 自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)
            @Override
            public boolean accept(File file) {
                return (recursive && file.isDirectory()) || (file.getName().endsWith(".class"));
            }
        });
        // 循环所有文件
        for (File file : dirfiles) {
            // 如果是目录 则继续扫描
            if (file.isDirectory()) {
                findClassesInPackageByFile(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));
                    
                    classes.add(Thread.currentThread().getContextClassLoader().loadClass(packageName + '.' + className));
                } catch (ClassNotFoundException e) {
                    // log.error("添加用户自定义视图类错误 找不到此类的.class文件");
                    e.printStackTrace();
                }
            }
        }
    }

}

IOC DI具体实现逻辑

package cn.tedu.myspring;

import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;

public class MyClassPathXMLApplicationContext {

    private HashMap<String, Object> hashMap = new HashMap();
    private Set<Class> set = new HashSet();

    /**
     * 对外提供一个代参的构造器,pakge为传入的包名的路径
     * 将包内所有的字节码文件获取,筛选出包含自定义注解的字节码文件放入Set集合中
     * 为什么要采用set集合呢?因为set集合中的元素具有不可重复性,保证字节码的唯一性
     * 调用get方法将带有表示的类实例化,并对注解的值进行判断,如果注解有值,当做key存入map
     * 如果没有值首字母小写当做key存入map
     * get方法调用add()方法,实现对属性的注入
     *
     * @param pakge
     */
    public MyClassPathXMLApplicationContext(String pakge) {
        Set<Class<?>> classes = MyTools.getClasses(pakge);//获取所有的字节码文件
        Iterator<Class<?>> iterator = classes.iterator();//获取set集合的迭代器准备将set集合中的字节码文件迭代出来
        while (iterator.hasNext()) {//开始迭代
            Class<?> next = iterator.next();//获取当前指针的元素,并将指针下移
            if (next.isAnnotationPresent(CGB.class)) {//判断当前字节码文件TYPE属性上是否有自己定义的CGB注解
                set.add(next);//如果包含自定义的CGB注解调用set集合的add()方法,将字节码文件放入set集合

            }
        }

        try {
            get();//调用get方法 将对象放入map集合中
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 将带有表示的类实例化,并对注解的值进行判断,如果注解有值,当做key存入map
     * 如果没有值首字母小写当做key存入map,调用自定义的add()方法对属性进行赋值
     *
     * @throws Exception
     */
    private void get() throws Exception {
        Class next = null;
        Object o = null;
        Iterator<Class> iterator = set.iterator();//迭代包含自定义注解的字节码文件
        while (iterator.hasNext()) {
            next = iterator.next();//获取当前字节码文件,并将指针下移
            o = next.newInstance();//将字节码文件实例化

            CGB annotation = (CGB) next.getAnnotation(CGB.class);//类上的注解,并将注解强制转换成自定义注解类型
            String value = annotation.value();//获取注解的value值

            if (value.equals("")) {//对注解的值进行判断,是否为空,为空则将类名小写作为key放入map集合当中
                String simpleName = next.getSimpleName();//通过当前类的字节码文件获取当前类名
                String s = simpleName.substring(0, 1).toLowerCase() + simpleName.substring(1, simpleName.length());//将类名首字母变成小写

                hashMap.put(s, o);//放入map集合
            } else {
                hashMap.put(value, o);//如果当前注解的value不为空,直接存入map集合


            }


        }
        add(next, o);//调用add()方法实现对包含赋值注解的属性的赋值操作
    }

    /**
     * 对属性进行赋值
     *
     * @param clazz        要赋值的类的字节码文件
     * @param o            字节码实例化的对象
     * @param <T>代表字节码类的类型
     * @throws Exception
     */
    private <T> void add(Class<T> clazz, Object o) throws Exception {
        synchronized (MyClassPathXMLApplicationContext.class) {

            Field[] declaredFields = clazz.getDeclaredFields();//通过暴力反射获取所有的属性
            for (Field declaredField : declaredFields) {//对属性进行遍历操作
                if (declaredField.isAnnotationPresent(value.class)) {//如果当前属性上包含自定义的value注解

                    value annotation = declaredField.getAnnotation(value.class);//获取属性上value注解的对象
                    String value = annotation.value();//获取value注解的值
                    Class<?> type1 = declaredField.getType();//获取当前属性的类型的字节码文件
                    String s1 = String.valueOf(type1);//将字节码文件转换成字符串

                    switch (s1) {//对s1进行比较
                        case "class java.lang.Integer"://如果是Integer类型
                            Integer a = Integer.valueOf(value);//将String字符串转换成Integer类型
                            declaredField.setAccessible(true);//将私有属性设置可见
                            declaredField.set((T) o, a);//通过暴力反射获取属性     以下类似不过多注释
                            break;
                        case "class java.lang.String":
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, value);
                            break;
                        case "class java.lang.Double":
                            Double b = Double.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, b);
                            break;
                        case "class java.lang.Float":
                            Float c = Float.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, c);
                            break;
                        case "class java.lang.Character":
                            if (value.length() == 1) {
                                declaredField.setAccessible(true);
                                declaredField.set((T) o, value.toCharArray()[0]);
                            } else {
                                throw new Exception();
                            }


                            break;
                        case "class java.lang.Byte":
                            Byte e = Byte.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, e);
                            break;
                        case "class java.lang.Short":
                            Short f = Short.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, f);
                            break;
                        case "class java.lang.Long":
                            Long fg = Long.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, fg);
                            break;
                        case "class java.lang.Boolean":
                            Boolean aBoolean = Boolean.valueOf(value);
                            declaredField.setAccessible(true);
                            declaredField.set((T) o, aBoolean);
                            break;

                        default:
                            //如果类型为引用数据类型执行default里面的内容

                            declaredField.setAccessible(true);//将私有属性设置可见

                            if (value.equals("")) {//如果value注解的值为默认值执行此分支
                                String s = declaredField.getName().substring(0, 1).toLowerCase() + declaredField.getName().substring(1, declaredField.getName().length());
                                add(getBean(s).getClass(), getBean(s));//递归本方法将将引用数据类型中包含自定义注解value的属性进行赋值
                                declaredField.set((T) o, getBean(s));//对属性进行赋值
                            }else {
                                add(getBean(value).getClass(), getBean(value));//递归本方法将将引用数据类型中包含自定义注解value的属性进行赋值
                                declaredField.set((T) o, getBean(value));//对属性进行赋值

                            }
                    }


                }
            }
        }
    }

    /**
     * 提供的获取对象的方法
     *
     * @param id
     * @return
     */
    public Object getBean(String id) {
        return hashMap.get(id);
    }

}

创建Pojo类用于作为封装类

package cn.tedu.test;

import cn.tedu.myspring.CGB;
import cn.tedu.myspring.value;

@CGB

public class Pojo {
    @value("张三")
    private String name;
    @value("13")
    private Integer age;
    @value("pojo1")
    private Pojo1 pojo1;

    @Override
    public String toString() {
        return "Pojo{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", pojo1=" + pojo1 +
                '}';
    }
}

创建Pojo1 类用于作为封装类

package cn.tedu.test;

import cn.tedu.myspring.CGB;
import cn.tedu.myspring.value;

@CGB
public class Pojo1 {
    @value("中行收")
    private String name;
    @value("121")
    private Integer age;
    @value("pojo2")
    private Pojo2 pojo2;

    @Override
    public String toString() {
        return "Pojo1{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", pojo2=" + pojo2 +
                '}';
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}

创建Pojo2类用于封装类

package cn.tedu.test;

import cn.tedu.myspring.CGB;
import cn.tedu.myspring.value;
@CGB
public class Pojo2 {
    @value("李四")
    private String name;
    @value("99")
    private Integer age;

    @Override
    public String toString() {
        return "Pojo2{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

创建Test类测试数据

package cn.tedu.test;

import cn.tedu.myspring.CGB;
import cn.tedu.myspring.MyClassPathXMLApplicationContext;

@CGB
public class Test {
    public static void main(String[] args) {
        MyClassPathXMLApplicationContext m = new MyClassPathXMLApplicationContext("cn.tedu.test");
        System.out.println(m.getBean("pojo"));
    }
}

运行结果

 相信你敲完之后会对注解反射以及Spring Ioc会有新的理解

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ᥬ᭄?

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值