dom4j 读取xml配置文件,根据配置文件利用反射创建对象

news2024/11/24 18:33:52

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example</groupId>
    <artifactId>mySpring</artifactId>
    <version>1.0</version>

    <!--打包方式-->
    <packaging>jar</packaging>

    <dependencies>

        <!--用于解析 xml 文件-->
        <dependency>
            <groupId>org.dom4j</groupId>
            <artifactId>dom4j</artifactId>
            <version>2.1.3</version>
        </dependency>

        <!--Java XPath 解析器
            XPath 是一种用于在 XML 文档中选择和查询数据的语言,可以认为它是一种针对 XML 文档的查询语言。
            Java 提供了许多库和工具来解析 XML,其中一个常见的用例是通过 XPath 查询指定的节点或数据,
            而 jaxen 提供了一种方便、灵活、易于使用、可扩展和全面的 API 来实现 XPath 查询-->
        <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>
            <scope>test</scope>
        </dependency>

        

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.1</version>
                <configuration>
                    <source>17</source>
                    <target>17</target>
                </configuration>
            </plugin>
        </plugins>
    </build>



</project>

package com.example.demo.bean;

public class User {

    private String name;

    private int age;

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public User() {
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
package com.example.demo.dao;

public class UserDao {

    public void insert(){
        System.out.println("mysql is inserting data!");
    }

}

package com.example.demo.service;

import com.example.demo.dao.UserDao;

public class UserService {

    private UserDao userDao;

    public void save(){
        userDao.insert();
    }

    public UserService(UserDao userDao) {
        this.userDao = userDao;
    }

    public UserService() {
    }

    public UserDao getUserDao() {
        return userDao;
    }

    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }
}

package com.example.spring;

public interface ApplicationContext {


    //根据 bean 的名称获取 bean 对象
    Object getBean(String beanName);
}

package com.example.spring;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;

import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class ClassPathXmlApplicationContext implements ApplicationContext {

    //存放单例bean对象
    private Map<String,Object> singletonObjects = new HashMap<>();

    //configLocation 配置文件的路径
    public ClassPathXmlApplicationContext(String configLocation){
        //解析 xml 配置文件,根据配置文件实例化bean,将bean对象存放到 singletonObjects
        /*org.dom4j.io.SAXReader 的作用是使用SAX解析器读取XML文件并将其转换为DOM对象模型。
            它可以遍历XML文档的节点树,访问和操作各个节点及其属性值,并支持XPath表达式查询*/
        SAXReader reader = new SAXReader();
        //获取类路径下资源文件的输入流
        InputStream in = ClassLoader.getSystemClassLoader().getResourceAsStream(configLocation);

        try {
            //读取配置文件, 返回DOM对象模型
            Document document = reader.read(in);
            /*selectNodes() 方法接收一个XPath表达式作为参数,返回所有匹配该表达式的节点集合(List<Node>)。这些节点可以是元素节点、属性节点或者文本节点等。*/
            List<Node> beans = document.selectNodes("//bean");
            //遍历 所有 bean 节点
            beans.forEach(
                    (bean) ->System.out.println(bean)
                    );

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

    }

    @Override
    public Object getBean(String beanName) {
        return null;
    }


}

myspring.xml 

<?xml version="1.0" encoding="UTF-8"?>
<beans>
    <bean id="user" class="com.example.demo.bean.User">
        <property name="name" value="罗小黑"></property>
        <property name="age" value="5"></property>
    </bean>

    <bean id="userDao" class="com.example.demo.dao.UserDao"></bean>

    <bean id="userService" class="com.example.demo.service.UserService">
        <property name="userDao" ref="userDao"></property>
    </bean>


</beans>

test

package demo;

import com.example.spring.ApplicationContext;
import com.example.spring.ClassPathXmlApplicationContext;
import org.junit.Test;

public class MySpring {

    @Test
    public void Test01(){
        ApplicationContext context = new ClassPathXmlApplicationContext("myspring.xml");
    }
}

result:

"C:\Program Files\Java\jdk-17\bin\java.exe" -ea -Didea.test.cyclic.buffer.size=1048576 -Didea.launcher.port=50276 "-Didea.launcher.bin.path=C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\bin" -Dfile.encoding=UTF-8 -classpath "C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\lib\idea_rt.jar;C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\plugins\junit\lib\junit-rt.jar;C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\plugins\junit\lib\junit5-rt.jar;C:\Minecloud\IDEA_workspace\spring_learn\mySpring\target\test-classes;C:\Minecloud\IDEA_workspace\spring_learn\mySpring\target\classes;C:\Users\29162\.m2\repository\org\dom4j\dom4j\2.1.3\dom4j-2.1.3.jar;C:\Users\29162\.m2\repository\jaxen\jaxen\1.2.0\jaxen-1.2.0.jar;C:\Users\29162\.m2\repository\junit\junit\4.13.2\junit-4.13.2.jar;C:\Users\29162\.m2\repository\org\hamcrest\hamcrest-core\1.3\hamcrest-core-1.3.jar" com.intellij.rt.execution.application.AppMainV2 com.intellij.rt.execution.junit.JUnitStarter -ideVersion5 -junit4 demo.MySpring,Test01
org.dom4j.tree.DefaultElement@11758f2a [Element: <bean attributes: [org.dom4j.tree.DefaultAttribute@33723e30 [Attribute: name id value "user"], org.dom4j.tree.DefaultAttribute@64f6106c [Attribute: name class value "com.example.demo.bean.User"]]/>]
org.dom4j.tree.DefaultElement@69b0fd6f [Element: <bean attributes: [org.dom4j.tree.DefaultAttribute@553a3d88 [Attribute: name id value "userDao"], org.dom4j.tree.DefaultAttribute@7a30d1e6 [Attribute: name class value "com.example.demo.dao"]]/>]
org.dom4j.tree.DefaultElement@4a87761d [Element: <bean attributes: [org.dom4j.tree.DefaultAttribute@5891e32e [Attribute: name id value "uerService"], org.dom4j.tree.DefaultAttribute@cb0ed20 [Attribute: name class value "com.example.demo.service"]]/>]

Process finished with exit code 0
 

加入 log4j2 日志框架

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example</groupId>
    <artifactId>mySpring</artifactId>
    <version>1.0</version>

    <!--打包方式-->
    <packaging>jar</packaging>

    <dependencies>

        <!--用于解析 xml 文件-->
        <dependency>
            <groupId>org.dom4j</groupId>
            <artifactId>dom4j</artifactId>
            <version>2.1.3</version>
        </dependency>

        <!--Java XPath 解析器
            XPath 是一种用于在 XML 文档中选择和查询数据的语言,可以认为它是一种针对 XML 文档的查询语言。
            Java 提供了许多库和工具来解析 XML,其中一个常见的用例是通过 XPath 查询指定的节点或数据,
            而 jaxen 提供了一种方便、灵活、易于使用、可扩展和全面的 API 来实现 XPath 查询-->
        <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>
            <scope>test</scope>
        </dependency>

        <!--log4j-slf4j2-impl 的作用是将应用程序中使用 SLF4J 接口的日志记录请求转发到 Log4j2 实现。
        这样做可以方便地在应用程序中使用 SLF4J 的统一日志接口,同时还能利用 Log4j2 强大的日志级别、过滤器和输出选项等功能。-->
        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-core</artifactId>
            <version>2.19.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-slf4j2-impl</artifactId>
            <version>2.19.0</version>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.1</version>
                <configuration>
                    <source>17</source>
                    <target>17</target>
                </configuration>
            </plugin>
        </plugins>
    </build>



</project>

log4j2 的配置文件

log4j2.xml

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

<configuration>

    <loggers>
        <!--
            level指定日志级别,从低到高的优先级:
                ALL < TRACE < DEBUG < INFO < WARN < ERROR < FATAL < OFF
        -->
        <root level="DEBUG">
            <appender-ref ref="myspringlog"/>
        </root>
    </loggers>

    <appenders>
        <!--输出日志信息到控制台-->
        <console name="myspringlog" target="SYSTEM_OUT">
            <!--控制日志输出的格式-->
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss SSS} [%t] %-3level %logger{1024} - %msg%n"/>
        </console>
    </appenders>

</configuration>
package com.example.spring;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class ClassPathXmlApplicationContext implements ApplicationContext {

    /*创建一个名为 "ClassPathXmlApplicationContext" 的日志记录器对象并赋值给 logger 变量。
        通过在程序中使用该 logger 对象输出日志信息,可以方便地追踪代码执行过程中产生的日志事件,从而帮助开发者进行调试和排查错误。*/
    private static final Logger logger = LoggerFactory.getLogger(ClassPathXmlApplicationContext.class);

    //存放单例bean对象
    private Map<String,Object> singletonObjects = new HashMap<>();

    //configLocation 配置文件的路径
    public ClassPathXmlApplicationContext(String configLocation){
        //解析 xml 配置文件,根据配置文件实例化bean,将bean对象存放到 singletonObjects
        /*org.dom4j.io.SAXReader 的作用是使用SAX解析器读取XML文件并将其转换为DOM对象模型。
            它可以遍历XML文档的节点树,访问和操作各个节点及其属性值,并支持XPath表达式查询*/
        SAXReader reader = new SAXReader();
        //获取类路径下资源文件的输入流
        InputStream in = ClassLoader.getSystemClassLoader().getResourceAsStream(configLocation);

        try {
            //读取配置文件, 返回DOM对象模型
            Document document = reader.read(in);
            /*selectNodes() 方法接收一个XPath表达式作为参数,返回所有匹配该表达式的节点集合(List<Node>)。这些节点可以是元素节点、属性节点或者文本节点等。*/
            List<Node> beans = document.selectNodes("//bean");
            //遍历 所有 bean 节点
            beans.forEach(
                    (bean) ->
                    {
                        //向下转型,为了使用Element的方法
                       Element beanElement = (Element)bean;
                       //获取 bean 的 id
                       String id = beanElement.attributeValue("id");
                       //获取 className
                       String className = beanElement.attributeValue("class");
                       //输出日志消息 是在INFO级别 level 记录消息,表示消息是普通的信息性质
                       logger.info("beanName=" + id);
                       logger.info("className" + className);
                    }
                    );

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

    }

    @Override
    public Object getBean(String beanName) {
        return null;
    }


}

test:

package com.example;

import com.example.spring.ClassPathXmlApplicationContext;

public class Myspring {

    public static void main(String[] args) {
        new ClassPathXmlApplicationContext("myspring.xml");
    }
}

结果: 

2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=user
2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - classNamecom.example.demo.bean.User
2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=userDao
2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - classNamecom.example.demo.dao
2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=uerService
2023-05-20 09:37:18 114 [main] INFO com.example.spring.ClassPathXmlApplicationContext - classNamecom.example.demo.service

Process finished with exit code 0
 

反射创建对象并加入map中 曝光

package com.example.spring;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.InputStream;
import java.lang.reflect.Constructor;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class ClassPathXmlApplicationContext implements ApplicationContext {

    /*创建一个名为 "ClassPathXmlApplicationContext" 的日志记录器对象并赋值给 logger 变量。
        通过在程序中使用该 logger 对象输出日志信息,可以方便地追踪代码执行过程中产生的日志事件,从而帮助开发者进行调试和排查错误。*/
    private static final Logger logger = LoggerFactory.getLogger(ClassPathXmlApplicationContext.class);

    //存放单例bean对象
    private Map<String,Object> singletonObjects = new HashMap<>();

    //configLocation 配置文件的路径
    public ClassPathXmlApplicationContext(String configLocation){
        //解析 xml 配置文件,根据配置文件实例化bean,将bean对象存放到 singletonObjects
        /*org.dom4j.io.SAXReader 的作用是使用SAX解析器读取XML文件并将其转换为DOM对象模型。
            它可以遍历XML文档的节点树,访问和操作各个节点及其属性值,并支持XPath表达式查询*/
        SAXReader reader = new SAXReader();
        //获取类路径下资源文件的输入流
        InputStream in = ClassLoader.getSystemClassLoader().getResourceAsStream(configLocation);

        try {
            //读取配置文件, 返回DOM对象模型
            Document document = reader.read(in);
            /*selectNodes() 方法接收一个XPath表达式作为参数,返回所有匹配该表达式的节点集合(List<Node>)。这些节点可以是元素节点、属性节点或者文本节点等。*/
            List<Node> beans = document.selectNodes("//bean");
            //遍历 所有 bean 节点
            beans.forEach(
                    (bean) ->
                    {
                        //向下转型,为了使用Element的方法
                       Element beanElement = (Element)bean;
                       //获取 bean 的 id
                       String id = beanElement.attributeValue("id");
                       //获取 className
                       String className = beanElement.attributeValue("class");
                       //输出日志消息 是在INFO级别 level 记录消息,表示消息是普通的信息性质
                       logger.info("beanName=" + id);
                       logger.info("className=" + className);


                       try {
                           //加载类,得到类 对象
                           Class<?> clazz = Class.forName(className);
                           //反射 获取无参构造方法
                           Constructor<?> constructor = clazz.getDeclaredConstructor();
                           //利用无参构造方法创建对象
                           Object o = constructor.newInstance();
                           //将对象加入map集合 曝光
                           singletonObjects.put(id,o);

                           logger.info(singletonObjects.toString());

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

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

    }

    @Override
    public Object getBean(String beanName) {
        return null;
    }


}

 结果:

"C:\Program Files\Java\jdk-17\bin\java.exe" -Didea.launcher.port=59706 "-Didea.launcher.bin.path=C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\bin" -Dfile.encoding=UTF-8 -classpath "C:\Minecloud\IDEA_workspace\spring_learn\mySpring\target\classes;C:\Users\29162\.m2\repository\org\dom4j\dom4j\2.1.3\dom4j-2.1.3.jar;C:\Users\29162\.m2\repository\jaxen\jaxen\1.2.0\jaxen-1.2.0.jar;C:\Users\29162\.m2\repository\org\apache\logging\log4j\log4j-core\2.19.0\log4j-core-2.19.0.jar;C:\Users\29162\.m2\repository\org\apache\logging\log4j\log4j-api\2.19.0\log4j-api-2.19.0.jar;C:\Users\29162\.m2\repository\org\apache\logging\log4j\log4j-slf4j2-impl\2.19.0\log4j-slf4j2-impl-2.19.0.jar;C:\Users\29162\.m2\repository\org\slf4j\slf4j-api\2.0.0\slf4j-api-2.0.0.jar;C:\Minecloud\IDEA_2019\IntelliJ IDEA 2019.1\lib\idea_rt.jar" com.intellij.rt.execution.application.AppMainV2 com.example.Myspring
2023-05-20 10:25:51 298 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=user
2023-05-20 10:25:51 298 [main] INFO com.example.spring.ClassPathXmlApplicationContext - className=com.example.demo.bean.User
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - {user=User{name='null', age=0}}
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=userDao
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - className=com.example.demo.dao.UserDao
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - {user=User{name='null', age=0}, userDao=com.example.demo.dao.UserDao@3e78b6a5}
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - beanName=uerService
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - className=com.example.demo.service.UserService
2023-05-20 10:25:51 314 [main] INFO com.example.spring.ClassPathXmlApplicationContext - {uerService=com.example.demo.service.UserService@41f69e84, user=User{name='null', age=0}, userDao=com.example.demo.dao.UserDao@3e78b6a5}

Process finished with exit code 0
 

调用set方法注入属性

package com.example.spring;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.InputStream;
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.List;
import java.util.Map;


public class ClassPathXmlApplicationContext implements ApplicationContext {

    /*创建一个名为 "ClassPathXmlApplicationContext" 的日志记录器对象并赋值给 logger 变量。
        通过在程序中使用该 logger 对象输出日志信息,可以方便地追踪代码执行过程中产生的日志事件,从而帮助开发者进行调试和排查错误。*/
    private static final Logger logger = LoggerFactory.getLogger(ClassPathXmlApplicationContext.class);

    //存放单例bean对象
    private Map<String,Object> singletonObjects = new HashMap<>();

    //configLocation 配置文件的路径
    public ClassPathXmlApplicationContext(String configLocation){
        //解析 xml 配置文件,根据配置文件实例化bean,将bean对象存放到 singletonObjects
        /*org.dom4j.io.SAXReader 的作用是使用SAX解析器读取XML文件并将其转换为DOM对象模型。
            它可以遍历XML文档的节点树,访问和操作各个节点及其属性值,并支持XPath表达式查询*/
        SAXReader reader = new SAXReader();
        //获取类路径下资源文件的输入流
        InputStream in = ClassLoader.getSystemClassLoader().getResourceAsStream(configLocation);

        try {
            //读取配置文件, 返回DOM对象模型
            Document document = reader.read(in);
            /*selectNodes() 方法接收一个XPath表达式作为参数,返回所有匹配该表达式的节点集合(List<Node>)。这些节点可以是元素节点、属性节点或者文本节点等。*/
            List<Node> beans = document.selectNodes("//bean");
            //遍历 所有 bean 节点
            beans.forEach(
                    (bean) ->
                    {
                        //向下转型,为了使用Element的方法
                       Element beanElement = (Element)bean;
                       //获取 bean 的 id
                       String id = beanElement.attributeValue("id");
                       //获取 className
                       String className = beanElement.attributeValue("class");
                       //输出日志消息 是在INFO级别 level 记录消息,表示消息是普通的信息性质
                       logger.info("beanName=" + id);
                       logger.info("className=" + className);


                       try {
                           //加载类,得到类 对象
                           Class<?> clazz = Class.forName(className);
                           //反射 获取无参构造方法
                           Constructor<?> constructor = clazz.getDeclaredConstructor();
                           //利用无参构造方法创建对象
                           Object o = constructor.newInstance();
                           //将对象加入map集合 曝光
                           singletonObjects.put(id,o);

                           logger.info(singletonObjects.toString());

                       } catch (Exception e) {
                           e.printStackTrace();
                       }
                    }
                    );
            //再次遍历bean标签,给属性赋值
            beans.forEach(
                    (bean) ->
                    {
                        try {
                            //向下转型,为了使用Element的方法
                            Element beanElement = (Element) bean;
                            //获取 bean 的 id
                            String id = beanElement.attributeValue("id");
                            //获取 className
                            String className = beanElement.attributeValue("class");
                            //加载类
                            Class<?> clazz = Class.forName(className);
                            //获取该标签下的所有 property 标签
                            List<Element> properties = beanElement.elements("property");
                            //遍历所有的 property 标签
                            properties.forEach(
                                    property ->
                                    {
                                        try {
                                            //获取属性名
                                            String propertyName = property.attributeValue("name");
                                            logger.info("propertyName=" + propertyName);
                                            //获取 value 属性的值
                                            String value = property.attributeValue("value");
                                            //获取 ref 的值
                                            String ref = property.attributeValue("ref");
                                            //通过 propertyName 获取 set 方法名
                                            String methodName = "set" + propertyName.toUpperCase().charAt(0) + propertyName.substring(1);
                                            logger.info("methodName=" + methodName);
                                            //反射获取属性
                                            Field field = clazz.getDeclaredField(propertyName);
                                            //反射获取 set 方法
                                            Method method = clazz.getDeclaredMethod(methodName,field.getType());
                                            //value 转换类型后的值
                                            Object actualValue = null;
                                            if(value != null) {
                                                if( String.class != field.getType() ){
                                                    //根据属性的类名将 value 转换为 相应的类型
                                                    actualValue = convertValue( value, field.getType() );
                                                    //调用 set 方法
                                                    method.invoke( singletonObjects.get(id), actualValue );
                                                }else{
                                                    method.invoke( singletonObjects.get(id), value );
                                                }

                                            }else if(ref != null){
                                                //调用 set 方法
                                                method.invoke( singletonObjects.get(id), singletonObjects.get(ref) );
                                            }
                                        }catch (Exception e){
                                            e.printStackTrace();
                                        }
                                    }
                            );
                        }
                        catch (Exception e){
                            e.printStackTrace();
                        }
                    }


            );

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

    }

    @Override
    public Object getBean(String beanName) {
        return this.singletonObjects.get(beanName);
    }

    //将 value转换为 type 类型
    public static Object convertValue(String value, Class<?> type) {
        // Check if the type is a primitive type
        if (type.isPrimitive()) {
            // Get the corresponding wrapper class
            type = getWrapperClass(type);
        }

        try {
            // Get the valueOf method of the wrapper class
            Method valueOfMethod = type.getMethod("valueOf", String.class);
            // Invoke the valueOf method to convert the value to the desired type
            return valueOfMethod.invoke(null, value);
        } catch (Exception e) {
            throw new RuntimeException("Error converting value", e);
        }
    }

    private static Class<?> getWrapperClass(Class<?> primitiveType) {
        if (primitiveType == boolean.class) return Boolean.class;
        if (primitiveType == byte.class) return Byte.class;
        if (primitiveType == char.class) return Character.class;
        if (primitiveType == short.class) return Short.class;
        if (primitiveType == int.class) return Integer.class;
        if (primitiveType == long.class) return Long.class;
        if (primitiveType == float.class) return Float.class;
        if (primitiveType == double.class) return Double.class;
        throw new IllegalArgumentException("Not a primitive type");
    }





}

 test

package com.example;

import com.example.demo.bean.User;
import com.example.demo.service.UserService;
import com.example.spring.ApplicationContext;
import com.example.spring.ClassPathXmlApplicationContext;

public class Myspring {

    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("myspring.xml");
        User user = (User) context.getBean("user");
        System.out.println( user.toString() );
        UserService userService = (UserService)context.getBean("userService");
        userService.getUserDao().insert();
    }
}

结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/549230.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

maven打包命令打出的可执行的jar包和可依赖的jar包的区别

目录 引出问题: 介绍打包插件 下面开始打包: 解压比较这两个jar包的区别: 引出问题: 当我建了一个maven的springboot项目A写了一个工具类,我把A项目打包成jar包去给B项目用,结果 B项目报错找不到这个jar包. 百度后发现原来jar包分为可执行jar包和可被依赖的jar包 介绍打包…

二、医院设置管理

文章目录 一、医院设置管理1、项目开发流程1.1 定义路由模块1.2 定义api模块1.3 定义页面组件脚本1.4 定义页面组件模板1.5 测试数据通信1.6 跨域处理 2、分页查询2.1 定义页面组件脚本2.2 定义页面组件模板2.3 表单查询 3、删除3.1 定义api模块3.2 定义页面组件模板3.3 定义页…

Midjourney|文心一格prompt教程[进阶篇]:Midjourney Prompt 高级参数、各版本差异、官方提供常见问题

Midjourney|文心一格prompt教程[进阶篇]&#xff1a;Midjourney Prompt 高级参数、各版本差异、官方提供常见问题 1.Midjourney Prompt 高级参数 Quality 图片质量是另一个我比较常用的属性&#xff0c;首先需要注意这个参数并不影响分辨率&#xff0c;并不改变分辨率&#x…

【C++】实现一个日期计算器

&#x1f307;个人主页&#xff1a;平凡的小苏 &#x1f4da;学习格言&#xff1a;命运给你一个低的起点&#xff0c;是想看你精彩的翻盘&#xff0c;而不是让你自甘堕落&#xff0c;脚下的路虽然难走&#xff0c;但我还能走&#xff0c;比起向阳而生&#xff0c;我更想尝试逆风…

线程池学习

一、线程池的7个核心参数说明&#xff1a; corePoolSize&#xff1a;核心线程数 maximumPoolSize&#xff1a;最大线程数 keepAliveTime&#xff1a;最大空闲时间 unit&#xff1a;最大空闲时间单位 workQueue&#xff1a;任务队列 threadFactory&#xff1a;表示生成线程…

1.Hyperledger Fabric架构介绍

&#xff08;1&#xff09;Hyperledger定义&#xff1a; Hyperledger是一个开放源代码的区块链项目合作组织&#xff0c;旨在推动跨行业的企业级区块链解决方案的发展。该项目由Linux基金会于2015年发起&#xff0c;致力于建立一个可靠、安全和可扩展的区块链框架和工具集。Hy…

堆结构 - 大根堆、小根堆

在开发语言中&#xff0c;heap在使用层次的名字叫PriorityQueue&#xff08;优先级队列&#xff09;&#xff0c;PriorityQueue数据结构的名字就叫做堆&#xff0c;底层就是用堆结构实现的。 完全二叉树 空树也算是完全二叉树每一层都是满的也算是完全二叉树如果层不满&#…

魔改车钥匙实现远程控车:(番外)在macOS上安装使用MicroPython

前言 哈哈&#xff0c;各位可能会奇怪为啥上一篇文章还在说怎么在 ESP32C3 上安装 Arduino&#xff0c;现在怎么又变成了安装 MIcroPython。 其实是因为上次写 Arduino 还是我高中时候的事了&#xff0c;已经不太会了。 虽然 MIcroPython 我从来没有接触过&#xff0c;但是 …

Microsoft Office 2003的安装

哈喽&#xff0c;大家好。今天一起学习的是office2003的安装&#xff0c;这个老版本的office可是XP操作系统的老搭档了&#xff0c;有兴趣的小伙伴也可以来一起试试手。 一、测试演示参数 演示操作系统&#xff1a;Windows XP 不建议win7及以上操作系统使用 系统类型&#xff…

Springboot 搭建WebService客户端+服务端

WebService简介 Web Service技术&#xff0c; 能使得运行在不同机器上的不同应用无须借助附加的、专门的第三方软件或硬件&#xff0c; 就可相互交换数据或集成。依据Web Service规范实施的应用之间&#xff0c; 无论它们所使用的语言、 平台或内部协议是什么&#xff0c; 都可…

软件设计和架构设计

软件设计和架构设计 1.软件设计 1.1设计 设计是从架构 构件 接口以及系统其他特征定义的过程。 软件设计的结果必须描述系统的架构&#xff0c;系统如何分解和组织构件。 描述构件间的接口。 描述构件必须详细到可进一步构造的程度。 设计是把分析模型转换成设计模型的过…

三个帮助你整理信息的桌面 WiKi

如果你想在桌面上感受 wiki&#xff0c;而不用做那些复杂的工作&#xff0c;这很容易做到。这有一些轻量级 wiki&#xff0c;可以帮助你组织你的信息、跟踪你的任务、管理你的笔记等等。 这个词时&#xff0c;可能会想到 MediaWiki 或 DokuWiki 这样的例子。它们开源、好用、强…

Go 并发之channel(通道)

一、前言 作为 Go 语言最有特色的数据类型&#xff0c;通道&#xff08;channel&#xff09;完全可以与 goroutine&#xff08;也可称为 go 程&#xff09;并驾齐驱&#xff0c;共同代表 Go 语言独有的并发编程模式和编程哲学。 通道&#xff08;channel&#xff09;可以利用…

TOGAF架构开发方法—G阶段:实施治理

本章提供了对实现的体系结构监督。 一、目标 G阶段的目标是&#xff1a; 通过实施项目确保符合目标架构为解决方案和任何实施驱动的架构更改请求执行适当的架构管理功能 二、 输入 本节定义阶段 G 的输入。 1 、企业外部参考物质 架构参考资料 2、 非架构输入 架构工作请…

K8s之污点、容忍度与Pod重启策略详解

文章目录 一、污点-Taint二、容忍度-Tolerations二、Pod重启策略1、Pod常见状态2、Pod重启策略 一、污点-Taint 在 Kubernetes 中&#xff0c;污点&#xff08;Taint&#xff09;是一种标记&#xff0c;用于标识一个Node节点上的某些资源或条件不可用或不可接受。当一个节点被…

基于springboot的社区疫情防控平台

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SpringBoot 前端&#xff1a;HTML、Vue 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#xff1a;是 前言 基于springboot…

模板类与友元

目录 分类 一、非模板友元&#xff1a;友元函数不是模板函数&#xff0c;而是利用模板类参数生成的函数&#xff1b; 代码 分析 运行结果 二、约束模板友元&#xff1a;模板类实例化时&#xff0c;每个实例化的类对应一个友元函数&#xff1b;并且这个模板友元适用多种类模…

AtCoder Beginner Contest 302(A-D)

TOYOTA MOTOR CORPORATION Programming Contest 2023#2 (AtCoder Beginner Contest 302) Contest Duration: 2023-05-20(Sat) 20:00 - 2023-05-20(Sat) 21:40 (local time) (100 minutes) 暴搜场&#xff0c;1个小时出了4道&#xff0c;以为很有机会&#xff0c;结果E交了十发没…

栈和队列OJ题:LeetCode--232.用栈实现队列

朋友们、伙计们&#xff0c;我们又见面了&#xff0c;今天给大家带来的是LeetCode--232.用栈实现队列 数 据 结 构 专 栏&#xff1a;数据结构 个 人 主 页 &#xff1a;stackY、 LeetCode 专 栏 &#xff1a;LeetCode刷题训练营 LeetCode--232.用栈实现队列&#xff…

使用 compose 封装一个通用的关于页面库

前言 现在很多 APP 都会有一个关于页面&#xff0c;用于放置一些必要的信息&#xff0c;例如&#xff1a;版本号、版权信息等。有时也会用于展示设置、帮助、反馈等功能的入口。 通常&#xff0c;我们都会自己挨个创建不同的 item &#xff0c;略显繁琐。 所以我就在想&…