Spring 实现对象实例化的几种方式

本文深入解析Spring框架中的依赖注入技术,包括构造器注入、setter注入、p-namespace和c-namespace的使用方法及实例,帮助读者理解不同注入方式的特点与适用场景。

构造器注入

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 无参数构造方法 默认-->
    <bean id="userService" class="guhong.test1.UserService"/>

    <!-- 有参构造:下标赋值 -->
    <bean id="userService2" class="guhong.test1.UserService2">
        <constructor-arg index="0" value="小西瓜"/>
        <constructor-arg index="1" value="23"/>
    </bean>

    <!-- 有参构造:类型赋值 参数类型必须各不相同 不推荐-->
    <bean id="userService3" class="guhong.test1.UserService3">
        <constructor-arg type="java.lang.String" value="刷该"/>
        <constructor-arg type="int" value="33"/>
    </bean>

    <!-- 有参构造:名称赋值 -->
    <bean id="userService4" class="guhong.test1.UserService4">
        <constructor-arg name="age" value="444"/>
        <constructor-arg name="name" value="name"/>
    </bean>

</beans>

 

setter注入

 

    public class Student {
    
        private String name;
        private Address address;
        private String[] books;
        private List<String> hobbys;
        private Map<String, String> cards;
        private Set<String> games;
        private String wife;
        private Properties info;
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Address getAddress() {
            return address;
        }
    
        public void setAddress(Address address) {
            this.address = address;
        }
    
        public String[] getBooks() {
            return books;
        }
    
        public void setBooks(String[] books) {
            this.books = books;
        }
    
        public List<String> getHobbys() {
            return hobbys;
        }
    
        public void setHobbys(List<String> hobbys) {
            this.hobbys = hobbys;
        }
    
        public Map<String, String> getCards() {
            return cards;
        }
    
        public void setCards(Map<String, String> cards) {
            this.cards = cards;
        }
    
        public Set<String> getGames() {
            return games;
        }
    
        public void setGames(Set<String> games) {
            this.games = games;
        }
    
        public String getWife() {
            return wife;
        }
    
        public void setWife(String wife) {
            this.wife = wife;
        }
    
        public Properties getInfo() {
            return info;
        }
    
        public void setInfo(Properties info) {
            this.info = info;
        }
    
        @Override
        public String toString() {
            return "Student{" +
                    "name='" + name + '\'' +
                    ", address=" + address +
                    ", books=" + Arrays.toString(books) +
                    ", hobbys=" + hobbys +
                    ", cards=" + cards +
                    ", games=" + games +
                    ", wife='" + wife + '\'' +
                    ", info=" + info +
                    '}';
        }
    }

XML

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">

<bean id="student" class="demo.guhong.test04.Student">
    <!-- 普通值注入 value -->
    <property name="name" value="guhong"/>
    <!-- bean注入 ref -->
    <property name="address" ref="address"/>

    <!-- 数组 -->
    <property name="books">
        <array>
            <value>西游记</value>
            <value>水浒传</value>
            <value>红楼梦</value>
            <value>三国演义</value>
        </array>
    </property>

    <!-- 列表 -->
    <property name="hobbys">
        <list>
            <value>篮球</value>
            <value>足球</value>
            <value>羽毛球</value>
            <value>乒乓球</value>
        </list>
    </property>

    <!-- map -->
    <property name="cards">
        <map>
            <entry key="公交卡" value="122343" />
            <entry key="学生卡" value="1dweesdsf22343" />
            <entry key="老人卡" value="43fdgdereg" />
        </map>
    </property>

    <!-- set -->
    <property name="games">
        <set>
            <value>CF</value>
            <value>LOL</value>
            <value>cs1.5</value>
        </set>
    </property>

    <!-- null -->
    <property name="wife">
        <null/>
    </property>

    <property name="info">
        <props>
            <prop key="学号">12323232</prop>
            <prop key="年龄">16</prop>
            <prop key="性别">male</prop>
            <prop key="姓名">guhong</prop>
            <prop key="userName">guhong</prop>
            <prop key="password">232323</prop>
        </props>
    </property>

</bean>

    <bean id="address" class="demo.guhong.test04.Address">
        <property name="address" value="亚特兰蒂斯"/>
    </bean>

</beans>

 

p-namespace:

主要是加入: xmlns:p="http://www.springframework.org/schema/p"

如果在XML中加入上述URL,显示红色的话,请看这里

可以直接注入属性的值,使用的的无参数的构造方式
public class User {

    private String name;
    private int age;

    public User(){
        System.out.println("User 无参构造");
    }

    public User(String name, int age) {
        System.out.println("User 有参数构造");
        this.name = name;
        this.age = age;
    }

    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 +
                '}';
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:p="http://www.springframework.org/schema/p"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
      http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

    <bean id="user" class="demo.guhong.test04.User" p:name="sdd" p:age="12" />

</beans>

 

c-namespace

加入:xmlns:c="http://www.springframework.org/schema/c"

使用的有参数的构造方法

public class Car {

    private String name;

    private double price;

    public Car(){
        System.out.println("Car 无参构造");
    }

    public Car(String name, double price) {
        System.out.println(" Car 有参数构造");
        this.name = name;
        this.price = price;
    }

    public String getName() {
        return name;
    }

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

    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }


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

XML

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:c="http://www.springframework.org/schema/c"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- c命名空间注入,通过构造器注入,construct args -->
    <bean id="car" class="demo.guhong.test04.Car" c:name="sdsd" c:price="12"/>

</beans>

 

p-namespace 和c-namesapce的结果测试

我在 User 和Car中都显示增加了无参构造和有参构造

public User(){
    System.out.println("User 无参构造");
}

public User(String name, int age) {
    System.out.println("User 有参数构造");
    this.name = name;
    this.age = age;
}

---------------------------------------------------------------

public Car(){
    System.out.println("Car 无参构造");
}

public Car(String name, double price) {
    System.out.println(" Car 有参数构造");
    this.name = name;
    this.price = price;
}

测试结果如下:

 

User 无参构造
 Car 有参数构造
User{name='sdd', age=12}
Car{name='sdsd', price=12.0}

Process finished with exit code 0

 

 

参考:狂神说

Spring框架中,Bean的实例化方式决定了对象如何被创建并纳入Spring容器管理。以下是主要的Bean实例化方式及其特点: --- ### **1. 通过构造器实例化(默认方式)** **原理**:Spring通过调用类的无参构造器创建Bean实例。 **适用场景**:大多数普通Java类。 **配置示例**: ```xml <!-- XML配置 --> <bean id="userService" class="com.example.UserService"/> ``` ```java // Java配置(@Bean) @Configuration public class AppConfig { @Bean public UserService userService() { return new UserService(); // 显式调用构造器 } } ``` **关键点**: - 若类没有无参构造器,会抛出`NoSuchMethodException`。 - 支持通过`<constructor-arg>`或`@ConstructorArgs`注入参数。 --- ### **2. 通过静态工厂方法实例化** **原理**:通过调用类的静态工厂方法(而非构造器)创建实例。 **适用场景**:需要复杂初始化逻辑或单例控制的类(如历史遗留代码)。 **配置示例**: ```xml <bean id="calendar" class="java.util.Calendar" factory-method="getInstance"/> ``` ```java // Java配置 @Bean public Calendar calendar() { return Calendar.getInstance(); // 调用静态工厂方法 } ``` **关键点**: - 工厂方法必须是`static`的。 - 适用于无法直接实例化的类(如工具类)。 --- ### **3. 通过实例工厂方法实例化** **原理**:通过另一个Bean的实例方法创建目标Bean。 **适用场景**:需要依赖其他Bean的工厂类。 **配置示例**: ```xml <!-- 定义工厂Bean --> <bean id="userFactory" class="com.example.UserFactory"/> <!-- 通过工厂方法创建Bean --> <bean id="adminUser" factory-bean="userFactory" factory-method="createAdmin"/> ``` ```java // Java配置 @Bean public UserFactory userFactory() { return new UserFactory(); } @Bean public User adminUser() { return userFactory().createAdmin(); // 调用实例方法 } ``` **关键点**: - 需先定义工厂Bean(`userFactory`)。 - 适用于需要动态生成对象的场景。 --- ### **4. 通过`FactoryBean`接口实例化** **原理**:实现`FactoryBean`接口的类由Spring调用其`getObject()`方法创建Bean。 **适用场景**:需要完全控制Bean创建过程(如代理、连接池等)。 **配置示例**: ```java public class MyFactoryBean implements FactoryBean<MyBean> { @Override public MyBean getObject() { return new MyBean(); // 自定义创建逻辑 } @Override public Class<?> getObjectType() { return MyBean.class; } } ``` ```xml <bean id="myBean" class="com.example.MyFactoryBean"/> ``` **关键点**: - 获取Bean时实际返回的是`getObject()`的结果。 - 常用于创建复杂对象(如`SqlSessionFactoryBean`)。 --- ### **5. 通过`@Bean`注解(Java配置)** **原理**:在`@Configuration`类中通过方法返回Bean实例。 **适用场景**:基于Java的配置(替代XML)。 **配置示例**: ```java @Configuration public class AppConfig { @Bean public UserService userService() { return new UserServiceImpl(); // 可包含复杂逻辑 } } ``` **关键点**: - 方法名默认作为Bean的`id`。 - 支持依赖注入(通过方法参数)。 --- ### **6. 通过组件扫描(`@Component`及其派生注解)** **原理**:Spring自动扫描标记了`@Component`、`@Service`、`@Repository`、`@Controller`的类并实例化。 **适用场景**:现代Spring应用的主流方式。 **配置示例**: ```java @Service // 或@Component, @Repository, @Controller public class UserServiceImpl implements UserService { // 类实现 } ``` ```java @Configuration @ComponentScan("com.example") // 指定扫描包 public class AppConfig {} ``` **关键点**: - 需配合`@ComponentScan`使用。 - 默认Bean的`id`为类名首字母小写(如`userServiceImpl`)。 --- ### **7. 通过`Supplier`接口(Spring 5+)** **原理**:通过`java.util.function.Supplier`延迟实例化Bean。 **适用场景**:需要懒加载或动态生成对象的场景。 **配置示例**: ```java @Bean public User user(Supplier<User> userSupplier) { return userSupplier.get(); // 调用Supplier的get()方法 } ``` 或直接使用`@Bean`的`initMethod`: ```java @Bean(initMethod = "customInit") public User user() { return new User(); } ``` --- ### **8. 通过`ServiceLoader`(SPI机制)** **原理**:利用Java的`ServiceLoader`加载实现类(需在`META-INF/services/`下配置)。 **适用场景**:模块化或插件化架构。 **配置示例**: 1. 创建接口`com.example.MyService`。 2. 在`META-INF/services/com.example.MyService`文件中写入实现类全限定名。 3. Spring通过`ServiceLoader`自动注册Bean。 --- ### **总结对比** | 方式 | 核心机制 | 适用场景 | |---------------------|-----------------------------------|----------------------------| | 构造器 | 调用无参构造器 | 普通Java类 | | 静态工厂方法 | 调用类的静态方法 | 工具类、单例 | | 实例工厂方法 | 调用其他Bean的实例方法 | 依赖工厂类的场景 | | `FactoryBean` | 实现`getObject()`方法 | 复杂对象创建(如代理) | | `@Bean` | 方法返回实例 | Java配置 | | 组件扫描 | 自动扫描标记注解的类 | 现代Spring应用 | | `Supplier` | 通过函数式接口延迟实例化 | 懒加载、动态生成 | | `ServiceLoader` | 利用Java SPI机制 | 模块化架构 | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值