手写spring 框架——第三篇(注解的方式)

简介: 手写spring 框架——第三篇(注解的方式)

(1)注解类

package com.jd.xq.spring.annotation;
/**
 * @author
 * @Date 2019-12-27 17:05
 **/
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,ElementType.METHOD})//对字段和方法使用注解
public @interface YhdResource {
    String value() default "";//注解里面只能声明属性,不能声明方法,声明属性的方式比较特殊:
}

(2)使用注解类

package com.jd.xq.spring.bean;
import com.jd.xq.spring.annotation.YhdResource;
/**
 * @author
 * @Date 2019-12-27 11:54
 **/
public class Person {
    @YhdResource(value = "xq")
    private String name;
    @YhdResource(value = "12")
    private String age;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }
}

(3)注解解析

package com.jd.xq.spring.context;
import com.alibaba.fastjson.JSONObject;
import com.jd.xq.spring.annotation.YhdResource;
import com.jd.xq.spring.bean.BeanDefinition;
import com.jd.xq.spring.bean.PropertyDefinition;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import java.lang.reflect.Field;
import java.net.URL;
import java.util.*;
/**
 * @author
 * @Date 2019-12-27 15:06
 **/
public class XqAnnotationClassPathXmlApplicationContext {
    private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();//用来存储所有的beans
    private Map<String, Object> sigletons = new HashMap<String, Object>();//用来存储实例化后的bean
    /**
     * 构造方法,用来模拟spring的行为
     *
     * @param fileName
     */
    public XqAnnotationClassPathXmlApplicationContext(String fileName) {
        try {
            //1.read xml
            this.readXml(fileName);
            //2.实例化bean
            this.instanceBeans();
            //3.注解方式
            this.annotationInject();
        } catch (Exception e) {
        }
    }
    /**
     * 根据文件名读取xml的配置文件
     */
    private void readXml(String fileName) {
        // TODO Auto-generated method stub
        //创建一个读取器
        SAXReader saxReader = new SAXReader();
        Document document = null;
        try {
            //获取要读取的配置文件的路径
            URL xmlPath = this.getClass().getClassLoader().getResource(fileName);
            //读取文件内容
            document = saxReader.read(xmlPath);
            //获取xml中的根元素
            Element rootElement = document.getRootElement();
            for (Iterator iterator = rootElement.elementIterator(); iterator.hasNext(); ) {
                Element element = (Element) iterator.next();
                String id = element.attributeValue("id");//获取bean的id属性值
                String clazz = element.attributeValue("class");//获取bean的class属性值
                BeanDefinition beanDefinition = new BeanDefinition(id, clazz);
                //获取bean的Property属性
                for (Iterator subElementIterator = element.elementIterator(); subElementIterator.hasNext(); ) {
                    Element subElement = (Element) subElementIterator.next();
                    String propertyName = subElement.attributeValue("name");
                    String propertyValue = subElement.attributeValue("value");
                    PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName, propertyValue);
                    beanDefinition.getPropertyDefinitions().add(propertyDefinition);
                }
                beanDefines.add(beanDefinition);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    /**
     * 实例化bean
     */
    private void instanceBeans() {
        if (beanDefines != null && beanDefines.size() > 0) {
            //对每个bean进行实例化
            for (BeanDefinition beanDefinition : beanDefines) {
                try {
                    //bean的class属性存在的时候才进行实例化,否则不进行实例化
                    if (beanDefinition.getClassName() != null && !beanDefinition.getClassName().equals("")) {
                        //实例化的关键操作
                        sigletons.put(beanDefinition.getId(), Class.forName(beanDefinition.getClassName()).newInstance());
                        System.out.println("id为:" + beanDefinition.getId() + "的bean实例化成功");
                    }
                } catch (Exception e) {
                    System.out.println("bean实例化失败");
                    e.printStackTrace();
                }
            }
        }
    }
    /**
     * 注解方式注入
     * <p>
     * Administer
     * 2013-9-24 下午8:08:29
     */
    private void annotationInject() throws IllegalAccessException {
        //遍历所有的bean
        for (String beanName : sigletons.keySet()) {
            Object bean = sigletons.get(beanName);//获取需要注入的bean
            if (bean != null) {
                //再对字段进行处理,即对字段上标识有注解
                Field[] fields = bean.getClass().getDeclaredFields();//取得声明的所有字段
                for (Field field : fields) {
                    //判断字段上是否存在注解,若存在
                    if (field.isAnnotationPresent(YhdResource.class)) {
                        YhdResource resource = field.getAnnotation(YhdResource.class);//取得字段上的注解
                        //字段上存在注解,并且字段上注解的name属性不为空
                        System.out.println(JSONObject.toJSONString(field));
                        System.out.println(JSONObject.toJSONString(resource));
                        field.setAccessible(true);
                        field.set(bean, resource.value());
                    }
                }
            }
        }
    }
    public Object getBean(String beanName) {
        return sigletons.get(beanName);
    }
}

(4)使用

XqAnnotationClassPathXmlApplicationContext xqAnnotationClassPathXmlApplicationContext = new XqAnnotationClassPathXmlApplicationContext("spring.xml");
        Person person = (Person) xqAnnotationClassPathXmlApplicationContext.getBean("person");
        System.out.println(person.getName());

代码地址:

https://github.com/zhugezifang/my-spring-framework



相关文章
|
9天前
|
XML Java 数据格式
SpringBoot入门(8) - 开发中还有哪些常用注解
SpringBoot入门(8) - 开发中还有哪些常用注解
28 0
|
22天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
28天前
|
Java Spring
在使用Spring的`@Value`注解注入属性值时,有一些特殊字符需要注意
【10月更文挑战第9天】在使用Spring的`@Value`注解注入属性值时,需注意一些特殊字符的正确处理方法,包括空格、引号、反斜杠、新行、制表符、逗号、大括号、$、百分号及其他特殊字符。通过适当包裹或转义,确保这些字符能被正确解析和注入。
|
1月前
|
Java API 数据库
Spring Boot框架因其简洁的配置、快速的启动特性及丰富的功能集而备受开发者青睐
本文通过在线图书管理系统案例,详细介绍如何使用Spring Boot构建RESTful API。从项目基础环境搭建、实体类与数据访问层定义,到业务逻辑实现和控制器编写,逐步展示了Spring Boot的简洁配置和强大功能。最后,通过Postman测试API,并介绍了如何添加安全性和异常处理,确保API的稳定性和安全性。
36 0
|
17天前
|
XML JSON Java
SpringBoot必须掌握的常用注解!
SpringBoot必须掌握的常用注解!
41 4
SpringBoot必须掌握的常用注解!
|
26天前
|
前端开发 Java 数据库连接
Spring 框架:Java 开发者的春天
Spring 框架是一个功能强大的开源框架,主要用于简化 Java 企业级应用的开发,由被称为“Spring 之父”的 Rod Johnson 于 2002 年提出并创立,并由Pivotal团队维护。
43 1
Spring 框架:Java 开发者的春天
|
18天前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,帮助开发者提高开发效率和应用的可维护性。
36 2
|
17天前
|
消息中间件 NoSQL Java
springboot整合常用中间件框架案例
该项目是Spring Boot集成整合案例,涵盖多种中间件的使用示例,每个案例项目使用最小依赖,便于直接应用到自己的项目中。包括MyBatis、Redis、MongoDB、MQ、ES等的整合示例。
68 1
|
18天前
|
存储 缓存 Java
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
58 2
|
18天前
|
JSON Java 数据库
SpringBoot项目使用AOP及自定义注解保存操作日志
SpringBoot项目使用AOP及自定义注解保存操作日志
34 1