当前位置: 首页 > news >正文

手写Sping IOC(基于Setter方法注入)

 什么是IOC? 控制反转:把对象的创建和对象之间的调用过程,从程序员手里转交给Spring进行管理。

1.导入依赖

<dependency>
    <groupId>org.dom4j</groupId>
    <artifactId>dom4j</artifactId>
    <version>2.1.3</version>
</dependency>
<dependency>
    <groupId>jaxen</groupId>
    <artifactId>jaxen</artifactId>
    <version>1.2.0</version>
</dependency>
<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.13.2</version>
</dependency>
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.6</version>
</dependency>

2.编写spring ioc接口

public interface ApplicationContext{

  getBean(String beanName);
}

3.实现soring ioc接口

public class ClassPathXmlApplicationContext implements ApplicationContext{
  
  private final Map<String, Object> objectMap = new HashMap<>();	

  public ClassPathXmlApplicationContext(String springConfigXml){
  	try {
            InputStream inputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream(springConfigXml);
            SAXReader reader = new SAXReader();
            Document document = reader.read(inputStream);
            List<Node> nodes = document.selectNodes("//bean");
            //读取bean标签创建bean对象,bean存入Map中
            for (Node node : nodes) {
                Element element = (Element) node;
                String id = element.attributeValue("id");
                String className = element.attributeValue("class");
                Class<?> clazz = Class.forName(className);
                Object obj = clazz.newInstance();
                objectMap.put(id, obj);
            }

            //读取bean标签下的property标签
            for (Node node : nodes) {
                Element element = (Element) node;
                String id = element.attributeValue("id");
                String className = element.attributeValue("class");
                Class<?> clazz = Class.forName(className);
                List<Element> property = element.elements("property");
                for (Element elem : property) {
                    String ref = elem.attributeValue("ref");
                    String name = elem.attributeValue("name");
                    //将property对象设置到父级bean对象
                    String setMethod = "set" + name.toUpperCase().charAt(0) + name.substring(1);
                    Method method = clazz.getDeclaredMethod(setMethod, objectMap.get(ref).getClass());
                    method.invoke(objectMap.get(id), objectMap.get(ref));
                }
            }

        } catch (Exception e) {
            e.printStackTrace();
        }
  }

  @Override
  public Object getBean(String beanId) {
    return objectMap.get(beanId);
  }
}

4.配置下xml(bean.xml)

<?xml version="1.0" encoding="UTF-8" ?>
<beans>

    <bean id="userController" class="com.xxx.yyy.controller.UserControllr">
        <property name="userService" ref="userService"/>
    </bean>

    <bean id="userService" class="com.xxx.yyy.service.impl.UserServiceImpl">
        <property name="userMapper" ref="userMapper"/>
    </bean>

    <bean id="userMapper" class="com.xxx.yyy.mapper.UserMapper"/>
</beans>

5.测试

public class SpringIocTest{

    @Test
    public void iocTest(){
      ApplicationContext context=new ClassPathXmlApplication("bean.xml");
      UserController userController=context.getBean("userController");
      userController.getUser(1);
    }
}

结果:
Connected to the target VM, address: '127.0.0.1:36341', transport: 'socket'
User(id=1, name=张三, sex=男, age=58, addr=中国)
Disconnected from the target VM, address: '127.0.0.1:36341', transport: 'socket'

Process finished with exit code 0

6.需要的类与接口

(User.class,UserController.class,UserService.interface,UserServiceImpl.class,UserMapper.class)

//User.class
@Data
public class User{

  private Integer id;

  private String name;

  private String sex;

  private Integer age;

  private String addr;
}

//UserController.class
public class UserController{

  private UserService userService;

  public void setUserService(UserServiceImpl userServiceImpl){
    this.userService=userServiceImpl;
  }

  public User getUser(Integer userId){
	return userService.getUser(userId);
  }
}

//UserService.interface
public interface UserService{
  
  User getUser(Integer userId);
}

//UserServiceImpl.class
public UserServiceImpl implements UserService{

  @Setter
  private UserMapper userMapper;

  public User getUser(Integer userId){
    return userMapper.getUser(userId);
  }
}

//UserMapper.class  (这里只做测试,就不去具体实现mybatis与Mapper的映射了)
public UserMapper{

  User getUser(Integer userId){
    User user=new User();
    user.setId(userId);
    user.setName("张三");
    user.setSex("男");
    user.setAge("58");
    user.setAddr("中国");
    return user;
  }
}

7.反射机制


相关类

  •   Class 类的类
  •   Field 属性
  •   Method 方法
  •   Constructor 构造方法

获取class

Object obj = new Object();
Class<?> clazz1 = Class.forName("com.xxx.yyy.ClassName");
Class<?> clazz2 = ClassName.class;
Class<?> clazz3 = obj.getClass();

获取构造方法

Constructor con = clazz.getConstructor()

创建对象

  1.无参构造创建

Object obj = clazz.newInstance();

 2.有参数构造创建  

Class[] parameterTypes = {String.class,int.class,char.class};
Constructor con = clazz.getConstructor(parameterTypes);
Object obj = con.newIntstance(parameterTypes); 

获取Field

getField() 与 getDeclaredField() 区别:
getField() 获取所有 public 的属性包括继承的 public 属性,
getDeclaredField() 获取当前类的所有属性,包括 private 修饰的属性,但是在访问 private 属性时,设置 setAccessible(true) 才可以访问.

Field[] fields = clazz.getDeclaredFields()
Field field = clazz.getDeclaredFiled("fieldName");
//取值
Object value = field.get(obj);


获取method

getMethod()与getDeclaredMethod()区别:
getMethod()调用所有public的方法包括继承的public方法,
getDeclaredField()调用当前类的所有方法,包括private修饰的方法,但是在调用private方法时,设置setAccessible(true)才可以调用. 

//无参  
Method[] methods=clazz.getDeclaredMethods()
Method method=clazz.getDeclaredMethod("methodName");//如getName()
method.setAccessible(true);
method.invoke(obj);//如getName(), Obejct value = method.invoke(obj)
//有参
Method method=clazz.getDeclaredMethod("methodName",methodParamTypes.class);//如User类setName(String name)
method.setAccessible(true);
method.invoke(obj, value);//如User类setName(String name)方法 obj=User,value="张三"

相关文章:

  • 一、SpringBoot前置(从0搭建Maven项目)
  • 宝藏教程:超详细一条龙教程!从零搭建React项目全家桶
  • 网络编程学习总结3
  • 欧姆龙CP1H如何进行PLC远程编程及数据采集
  • CSGO Bway电竞ETERNAL FIRE可以参加BLAST FALL,但MOUZ却错过了
  • 收获tips
  • Vue3+elementplus搭建通用管理系统实例十二:使用通用表格、表单实现对应功能
  • 抖音根据关键词取视频列表 API 返回值说明
  • 120页7万字XX云数据中心解决方案技术方案
  • 保险行业如何实现私域快速增长
  • 计算机网络的基础知识
  • 开发者测评:相比 Harbor,我选择 ACR 的三点原因
  • 通用配置
  • 基于JAVA自由教学平台计算机毕业设计源码+系统+数据库+lw文档+部署
  • Springboot2——配置文件、Junit、日志(笔记)
  • 9月CHINA-PUB-OPENDAY技术沙龙——IPHONE
  • ES6核心特性
  • JS专题之继承
  • node-glob通配符
  • SpingCloudBus整合RabbitMQ
  • 阿里研究院入选中国企业智库系统影响力榜
  • 从 Android Sample ApiDemos 中学习 android.animation API 的用法
  • 官方解决所有 npm 全局安装权限问题
  • 规范化安全开发 KOA 手脚架
  • 好的网址,关于.net 4.0 ,vs 2010
  • 警报:线上事故之CountDownLatch的威力
  • 看图轻松理解数据结构与算法系列(基于数组的栈)
  • 前端相关框架总和
  • 区块链分支循环
  • 如何用vue打造一个移动端音乐播放器
  • 《TCP IP 详解卷1:协议》阅读笔记 - 第六章
  • ​RecSys 2022 | 面向人岗匹配的双向选择偏好建模
  • #define
  • (1综述)从零开始的嵌入式图像图像处理(PI+QT+OpenCV)实战演练
  • (C++17) std算法之执行策略 execution
  • (C++20) consteval立即函数
  • (Java)【深基9.例1】选举学生会
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (转)jdk与jre的区别
  • (转)memcache、redis缓存
  • (转)负载均衡,回话保持,cookie
  • (转)如何上传第三方jar包至Maven私服让maven项目可以使用第三方jar包
  • (转)自己动手搭建Nginx+memcache+xdebug+php运行环境绿色版 For windows版
  • .mkp勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .NET Core WebAPI中封装Swagger配置
  • .NET delegate 委托 、 Event 事件
  • .NET/C# 检测电脑上安装的 .NET Framework 的版本
  • .NET/C# 利用 Walterlv.WeakEvents 高性能地定义和使用弱事件
  • .net的socket示例
  • .vollhavhelp-V-XXXXXXXX勒索病毒的最新威胁:如何恢复您的数据?
  • /*在DataTable中更新、删除数据*/
  • [].shift.call( arguments ) 和 [].slice.call( arguments )
  • [1] 平面(Plane)图形的生成算法
  • [22]. 括号生成
  • [AR Foundation] 人脸检测的流程