• Spring2


    Spring2

    配置文件

    通过配置 bean 标签来完成对象的管理。

    id :对象名。

    class :对象的模版类(所有交给 IoC 容器来管理的类必须有⽆参构造函数,因为 Spring 底 层是通过反射机制来创建对象,调⽤的是⽆参构造)

    对象的成员变量通过 property 标签完成赋值。

    name :成员变量名。

    value :成员变量值(基本数据类型,String 可以直接赋值,如果是其他引⽤类型,不能通 过 value 赋值)

    ref :将 IoC 中的另外⼀个 bean 赋给当前的成员变量(DI)

    <bean id="student" class="com.southwind.entity.Student">
        <property name="id" value="1"></property>
        <property name="name" value="张三"></property>
        <property name="age" value="22"></property>
        <property name="address" ref="address"></property>
    </bean>
    <bean id="address" class="com.southwind.entity.Address">
        <property name="id" value="1"></property>
        <property name="name" value="科技路"></property>
    </bean>
    

    IOC底层原理

    读取配置⽂件,解析 XML。

    通过反射机制实例化配置⽂件中所配置所有的 bean。

    import org.dom4j.Document;
    import org.dom4j.DocumentException;
    import org.dom4j.Element;
    import org.dom4j.io.SAXReader;
    import java.lang.reflect.Constructor;
    import java.lang.reflect.Field;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    import java.util.HashMap;
    import java.util.Iterator;
    import java.util.Map;
    public class ClassPathXmlApplicationContext implements ApplicationContext {
    private Map<String,Object> ioc = new HashMap<String, Object>();
    public ClassPathXmlApplicationContext(String path){
    try {
    SAXReader reader = new SAXReader();
    Document document = reader.read("./src/main/resources/"+path);
    Element root = document.getRootElement();
    Iterator<Element> iterator = root.elementIterator();
        while(iterator.hasNext()){
        Element element = iterator.next();
        String id = element.attributeValue("id");
        String className = element.attributeValue("class");
        //通过反射机制创建对象
        Class clazz = Class.forName(className);
        //获取⽆参构造函数,创建⽬标对象
        Constructor constructor = clazz.getConstructor();
        Object object = constructor.newInstance();
        //给⽬标对象赋值
        Iterator<Element> beanIter = element.elementIterator();
            while(beanIter.hasNext()){
            Element property = beanIter.next();
            String name = property.attributeValue("name");
            String valueStr = property.attributeValue("value");
            String ref = property.attributeValue("ref");
            if(ref == null){
            String methodName =
            "set"+name.substring(0,1).toUpperCase()+name.substring(1);
            Field field = clazz.getDeclaredField(name);
            Method method =
            clazz.getDeclaredMethod(methodName,field.getType());
            //根据成员变量的数据类型将 value 进⾏转换
            Object value = null;
            if(field.getType().getName() == "long"){
            value = Long.parseLong(valueStr);
            }
            if(field.getType().getName() == "java.lang.String"){
            value = valueStr;
            }
            if(field.getType().getName() == "int"){
            value = Integer.parseInt(valueStr);
            }
            method.invoke(object,value);
            }
            ioc.put(id,object);
            }
            }
            } catch (DocumentException e) {
            e.printStackTrace();
            } catch (ClassNotFoundException e){
            e.printStackTrace();
            } catch (NoSuchMethodException e){
            e.printStackTrace();
            } catch (InstantiationException e){
            e.printStackTrace();
            } catch (IllegalAccessException e){
            e.printStackTrace();
            } catch (InvocationTargetException e){
            e.printStackTrace();
            } catch (NoSuchFieldException e){
            e.printStackTrace();
            }
            }
            public Object getBean(String id) {
            return ioc.get(id);
            }
    }
    

    通过运⾏时类获取 bean

    ApplicationContext applicationContext = new
    ClassPathXmlApplicationContext("spring.xml");
    Student student = (Student) applicationContext.getBean(Student.class);
    System.out.println(student);
    

    这种⽅式存在⼀个问题,配置⽂件中⼀个数据类型的对象只能有⼀个实例,否则会抛出异常,因为没有 唯⼀的 bean。

    通过有参构造创建 bean

    在实体类中创建对应的有参构造函数。

    配置⽂件

    <bean id="student3" class="com.southwind.entity.Student">
        <constructor-arg name="id" value="3"></constructor-arg>
        <constructor-arg name="name" value="⼩明"></constructor-arg>
        <constructor-arg name="age" value="18"></constructor-arg>
        <constructor-arg name="address" ref="address"></constructor-arg>
    </bean>
    
    <bean id="student3" class="com.southwind.entity.Student">
        <constructor-arg index="0" value="3"></constructor-arg>
        <constructor-arg index="2" value="18"></constructor-arg>
        <constructor-arg index="1" value="⼩明"></constructor-arg>
        <constructor-arg index="3" ref="address"></constructor-arg>
    <bean>
    

    给 bean 注⼊集合

    <bean id="student" class="com.southwind.entity.Student">
        <property name="id" value="2"></property>
        <property name="name" value="李四"></property>
        <property name="age" value="33"></property>
        <property name="addresses">
            <list>
                <ref bean="address"></ref>
                <ref bean="address2"></ref>
            </list>
        </property>
    </bean>
    <bean id="address" class="com.southwind.entity.Address">
        <property name="id" value="1"></property>
        <property name="name" value="科技路"></property>
    </bean>
    <bean id="address2" class="com.southwind.entity.Address">
        <property name="id" value="2"></property>
        <property name="name" value="⾼新区"></property>
    </bean>
    
  • 相关阅读:
    STM8s在利用库配置端口的小问题
    ABAP调试
    READ TABLE 的用法
    人在低谷
    力扣 两数之和
    未来选择
    选择
    室友问题该如何解决呢?
    力扣 两数之和
    谈谈自己
  • 原文地址:https://www.cnblogs.com/wind-and-sky/p/14213455.html
Copyright © 2020-2023  润新知