草庐IT

day03-Spring管理Bean-IOC-01

liyuelian 2023-04-16 原文

Spring管理Bean-IOC

1.Spring配置/管理bean介绍

Bean管理包括两方面:

  1. 创建bean对象
  2. 给bean注入属性

Bean的配置方式:

  • 基于xml文件配置方式
  • 基于注解配置方式

2.基于XML配置bean

2.1通过类型来获取bean

通过id来获取bean在Spring基本介绍中已经使用过,这里不再赘叙

  1. 案例说明:通过spring的ioc容器,获取一个bean对象(说明:获取bean的方式:按类型)

  2. 完成步骤:

    创建一个Java对象Monster.java

    在beans.xml中配置


Monster.java:

package com.li.bean;

/**
 * @author 李
 * @version 1.0
 * Javabean / Entity
 */
public class Monster {
    private Integer monsterId;
    private String name;
    private String skill;

    //无参构造器一定要有,spring底层反射创建对象时需要使用
    public Monster() {
    }

    public Monster(Integer monsterId, String name, String skill) {
        this.monsterId = monsterId;
        this.name = name;
        this.skill = skill;
    }

    public Integer getMonsterId() {
        return monsterId;
    }

    public void setMonsterId(Integer monsterId) {
        this.monsterId = monsterId;
    }

    public String getName() {
        return name;
    }

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

    public String getSkill() {
        return skill;
    }

    public void setSkill(String skill) {
        this.skill = skill;
    }

    @Override
    public String toString() {
        return "Monster{" +
                "monsterId=" + monsterId +
                ", name='" + name + '\'' +
                ", skill='" + skill + '\'' +
                '}';
    }
}

beans.xml:

<!--配置Monster,希望通过类型来获取-->
<bean class="com.li.bean.Monster" >
    <property name="monsterId" value="10086"/>
    <property name="name" value="孙悟空"/>
    <property name="skill" value="筋斗云"/>
</bean>

测试类:

//通过Bean的类型来获取对象
@Test
public void getBeanByType() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    //直接传入class对象/类型
    Monster bean = ioc.getBean(Monster.class);
    System.out.println(bean);
}

细节说明:

  1. 按照类型来获取bean,要求ioc容器中的同一个类的bean只能有一个,否则会抛出异常NoUniqueBeanDefinitionException

  2. 用类型来获取bean的应用场景:XxxAction/Servlet/Controller或者XxxService,在一个线程中只需要一个对象实例的情况(单例情况)

  3. 在容器配置文件(比如beans.xml)中给属性值,底层是通过setter方法完成的,这也是为什么需要在java对象中提供setter方法,否则会报错

2.2通过构造器配置bean

案例说明:使用spring的ioc容器,可以通过构造器来来配置bean对象


在beans.xml配置:

前提是Monster类中有对应的构造器

   <!--配置Monster对象,并指定构造器-->
    <!--
        1. constructor-arg 标签可以指定使用构造器的参数
        2. index 表示构造器的第几个参数,从0开始计算
        3. 除了通过index,还可以通过 name/ type 的方式来指定参数
        4. 类构造器可以有多个,但是不能有参数完全相同的类型和顺序的构造器,
           这就决定了 通过构造器参数type 可以唯一确定一个构造器)
    -->
    <bean id="monster03" class="com.li.bean.Monster">
        <constructor-arg value="200" index="0"/>
        <constructor-arg value="白骨精" index="1"/>
        <constructor-arg value="吸血" index="2"/>
    </bean>
    <!--通过 name 的方式来指定参数-->
    <bean id="monster04" class="com.li.bean.Monster">
        <constructor-arg value="200" name="monsterId"/>
        <constructor-arg value="白骨精" name="name"/>
        <constructor-arg value="吸血" name="skill"/>
    </bean>
    <!--通过 type 的方式来指定参数-->
    <bean id="monster05" class="com.li.bean.Monster">
        <constructor-arg value="200" type="java.lang.Integer"/>
        <constructor-arg value="白骨精" type="java.lang.String"/>
        <constructor-arg value="吸血" type="java.lang.String"/>
    </bean>

测试类:

//通过构造器来设置属性
@Test
public void setBeanConstructor() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    Monster monster03 = ioc.getBean("monster03", Monster.class);
    System.out.println("monster03=" + monster03);
}

细节说明

  1. 通过index属性来区分是第几个参数
  2. 通过type属性来区分是什么类型(按照顺序)

2.3通过p名称空间配置bean

  1. 案例说明:在spring的ioc容器,可以通过p名称空间来配置bean对象

  2. 完成步骤:在beans.xml配置,增加命名空间配置,如下,点击Create namespace declaration,成功后在配置文件头会自动添加xmlns


beans.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:p="http://www.springframework.org/schema/p"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!--通过p名称空间来配置bean
        将光标放在p,输入alt+enter,就会自动添加 xmlns
     -->
    <bean id="monster06" class="com.li.bean.Monster"
          p:monsterId="500"
          p:name="红孩儿"
          p:skill="风火轮"
    />
</beans>

测试类:

//通过p名称空间来设置属性
@Test
public void setBeanByP() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    Monster monster06 = ioc.getBean("monster06", Monster.class);
    System.out.println("monster06=" + monster06);
}

细节说明:注意要引入p名称空间

2.4引用/注入其他bean对象

  1. 案例说明:在spring的ioc容器中,可以通过 ref (reference)来实现bean对象的相互应用

  2. 完成步骤:

    (1)创建MemberDAOImpl.java,MemberServiceImpl.java

    (2)配置beans.xml


beans.xml:

<!--配置MemberDAOImpl对象-->
<bean class="com.li.dao.MemberDAOImpl" id="memberDAO"/>

<!--配置MemberServiceImpl对象
     1.ref="memberDAO" 表示
        MemberServiceImpl 对象的属性memberDAO 引用的对象是id=memberDAO的对象
     2.这就体现了spring容器的依赖注入
     3.注意在spring容器中,是作为一个整体来执行的,也就是说如果你引用到了一个bean对象,对你配置的顺序没有要求
        (在底层,对spring配置文件进行扫描,将关联的关系自动梳理,并放到beanDefinitionMap中,
         也就是说,是从beanDefinitionMap中查找关联关系的。与spring配置文件的配置顺序无关)
     4.但是建议按照顺序配置,易于阅读
  -->
<bean class="com.li.service.MemberServiceImpl" id="memberService">
    <property name="memberDAO" ref="memberDAO"/>
</bean>

MemberDAOImpl:

package com.li.dao;

/**
 * @author 李
 * @version 1.0
 * DAO 对象
 */
public class MemberDAOImpl {
    //构造器
    public MemberDAOImpl() {
        System.out.println("MemberDAOImpl 构造器...");
    }

    //方法
    public void add() {
        System.out.println("MemberDAOImpl add()方法被执行");
    }

}

MemberServiceImpl:

package com.li.service;

import com.li.dao.MemberDAOImpl;

/**
 * @author 李
 * @version 1.0
 * Service类
 */
public class MemberServiceImpl {
    private MemberDAOImpl memberDAO;

    public MemberDAOImpl getMemberDAO() {
        return memberDAO;
    }

    public void setMemberDAO(MemberDAOImpl memberDAO) {
        this.memberDAO = memberDAO;
    }

    public void add() {
        System.out.println("MemberServiceImpl add() 方法被调用...");
        memberDAO.add();
    }
}

测试类:

//通过ref 来设置 bean属性
@Test
public void setBeanByRef() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    MemberServiceImpl memberService =
            ioc.getBean("memberService", MemberServiceImpl.class);

    memberService.add();
}

细节说明:

注意:在spring容器中,是作为一个整体来执行的,也就是说如果你引用到了一个bean对象,对你配置的顺序没有要求。例如这里的例子,如果我们将Memb erDAOImpl对象的配置,放到MemberServiceImpl对象的配置之后也是可以的。

因为在底层,会对spring的配置文件进行扫描,将关联的关系自动梳理,并放到beanDefinitionMap中。

也就是说,是从beanDefinitionMap中查找关联关系的。与spring配置文件的配置顺序无关。

但是建议按照顺序配置,易于阅读

2.5引入/注入内部bean对象

  1. 案例说明:在spring的ioc容器中,可以直接配置内部bean对象

  2. 完成步骤:

    (1)创建MemberDAOImpl.java,MemberServiceImpl.java

    (2)配置beans.xml


创建MemberDAOImpl.java,MemberServiceImpl.java(见2.4)

beans.xml:

<!--配置MemberService对象(使用内部bean)-->
<bean class="com.li.service.MemberServiceImpl" id="memberService2">
    <!--自己配置一个内部bean-->
    <property name="memberDAO">
        <bean class="com.li.dao.MemberDAOImpl"/>
    </property>
</bean>

测试类:

//通过内部 bean来设置 bean属性
@Test
public void setBeanByProper() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    MemberServiceImpl memberService2 =
            ioc.getBean("memberService2", MemberServiceImpl.class);

    memberService2.add();
}

2.6引用/注入集合/数组类型

  1. 案例说明:在spring的ioc容器中,看看如何给bean对象的集合/数组类型属性赋值

  2. 完成步骤:

    (1)创建Monster.java

    (2)创建Master.java

    (3)配置beans.xml


Monster.java:见2.1

Master.java:

package com.li.bean;

import java.util.*;

/**
 * @author 李
 * @version 1.0
 * Master类
 */
public class Master {
    private String name;

    private List<Monster> monsterList;
    private Map<String, Monster> monsterMap;
    private Set<Monster> monsterSet;

    //数组
    private String[] monsterName;

    // Properties 是 Hashtable 的子类,也是k-v形式
    // 这里Properties 的key和value都是String类型
    private Properties pros;

    public String getName() {
        return name;
    }

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

    public List<Monster> getMonsterList() {
        return monsterList;
    }

    public void setMonsterList(List<Monster> monsterList) {
        this.monsterList = monsterList;
    }

    public Map<String, Monster> getMonsterMap() {
        return monsterMap;
    }

    public void setMonsterMap(Map<String, Monster> monsterMap) {
        this.monsterMap = monsterMap;
    }

    public Set<Monster> getMonsterSet() {
        return monsterSet;
    }

    public void setMonsterSet(Set<Monster> monsterSet) {
        this.monsterSet = monsterSet;
    }

    public String[] getMonsterName() {
        return monsterName;
    }

    public void setMonsterName(String[] monsterName) {
        this.monsterName = monsterName;
    }

    public Properties getPros() {
        return pros;
    }

    public void setPros(Properties pros) {
        this.pros = pros;
    }

    @Override
    public String toString() {
        return "Master{" +
                "name='" + name + '\'' +
                ", monsterList=" + monsterList +
                ", monsterMap=" + monsterMap +
                ", monsterSet=" + monsterSet +
                ", monsterName=" + Arrays.toString(monsterName) +
                ", pros=" + pros +
                '}';
    }
}

beans.xml:

<!--配置Master对象-->
<!--体会Spring容器配置特点-->
<bean class="com.li.bean.Master" id="master">
    <property name="name" value="太上老君"/>

    <!--(1)给list属性赋值-->
    <property name="monsterList">
        <list>
            <!--方式1.引用的方式-->
            <ref bean="monster01"/>
            <ref bean="monster02"/>
            <!--方式2.内部bean-->
            <bean class="com.li.bean.Monster">
                <property name="name" value="黄袍怪"/>
                <property name="monsterId" value="1024"/>
                <property name="skill" value="吃人"/>
            </bean>
        </list>
    </property>

    <!--(2)给map属性赋值-->
    <property name="monsterMap">
        <map>
            <entry>
                <key>
                    <value>monster01</value>
                </key>
                <!--也可以通过内部 bean来配置,这里使用引用-->
                <ref bean="monster01"/>
            </entry>
            <entry>
                <key>
                    <value>monster02</value>
                </key>
                <!--也可以通过内部 bean来配置,这里使用引用-->
                <ref bean="monster02"/>
            </entry>
        </map>
    </property>

    <!--(3)给set属性赋值-->
    <property name="monsterSet">
        <set>
            <!--同理,依然可以使用ref或者内部 bean的方式配置-->
            <ref bean="monster05"/>
            <bean class="com.li.bean.Monster">
                <property name="name" value="金角大王"/>
                <property name="monsterId" value="888"/>
                <property name="skill" value="有钱"/>
            </bean>
        </set>
    </property>

    <!--(4)给数组属性赋值-->
    <property name="monsterName">
        <array>
            <!--这里根据数组的类型来选择对应的标签-->
            <value>小猪妖</value>
            <value>大鹏妖</value>
            <value>老狐狸</value>
            <!--同理,依然可以使用ref或者内部 bean的方式配置-->
        </array>
    </property>

    <!--(5)给Properties属性赋值,结构 k(String)-v(String)-->
    <property name="pros">
        <props>
            <prop key="username">root</prop>
            <prop key="pwd">1234</prop>
            <prop key="ip">127.0.0.1</prop>
        </props>
    </property>
</bean>

测试类:

//给集合/数组属性进行配置值
@Test
public void setBeanByCollection() {
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans.xml");
    Master master = ioc.getBean("master", Master.class);

    System.out.println("========master的list属性=========");
    for (Monster monster : master.getMonsterList()) {
        System.out.println(monster);
    }

    System.out.println("\n========master的map属性=========");
    Iterator<Map.Entry<String, Monster>> iterator =
            master.getMonsterMap().entrySet().iterator();
    while (iterator.hasNext()) {
        System.out.println(iterator.next());
    }

    System.out.println("\n========master的set属性=========");
    for (Monster monster : master.getMonsterSet()) {
        System.out.println(monster);
    }

    System.out.println("\n========master的数组属性=========");
    for (int i = 0; i < master.getMonsterName().length; i++) {
        System.out.println(master.getMonsterName()[i]);
    }

    System.out.println("\n========master的Properties属性=========");
    Iterator<Map.Entry<Object, Object>> iterator1
            = master.getPros().entrySet().iterator();
    while (iterator1.hasNext()) {
        System.out.println(iterator1.next());
    }
}

细节说明:

  1. 主要掌握List/Map/Properties三种集合的使用

  2. Properties集合的特点:

    (1)Properties是Hashtable的子类,也是key-value形式

    (2)key是String类型,value也是String类型

有关day03-Spring管理Bean-IOC-01的更多相关文章

  1. ruby-on-rails - 带 Spring 锁的 Rails 4 控制台 - 2

    我正在使用Ruby2.1.1和Rails4.1.0.rc1。当执行railsc时,它被锁定了。使用Ctrl-C停止,我得到以下错误日志:~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`gets':Interruptfrom~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`verify_server_version'from~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.

  2. spring.profiles.active和spring.profiles.include的使用及区别说明 - 2

    转自:spring.profiles.active和spring.profiles.include的使用及区别说明下文笔者讲述spring.profiles.active和spring.profiles.include的区别简介说明,如下所示我们都知道,在日常开发中,开发|测试|生产环境都拥有不同的配置信息如:jdbc地址、ip、端口等此时为了避免每次都修改全部信息,我们则可以采用以上的属性处理此类异常spring.profiles.active属性例:配置文件,可使用以下方式定义application-${profile}.properties开发环境配置文件:application-dev

  3. ruby-on-rails - Spring 不起作用。 [未初始化常量 Spring::SID::DL] - 2

    我无法运行Spring。这是错误日志。myid-no-MacBook-Pro:myid$spring/Users/myid/.rbenv/versions/1.9.3-p484/lib/ruby/gems/1.9.1/gems/spring-0.0.10/lib/spring/sid.rb:17:in`fiddle_func':uninitializedconstantSpring::SID::DL(NameError)from/Users/myid/.rbenv/versions/1.9.3-p484/lib/ruby/gems/1.9.1/gems/spring-0.0.10/li

  4. ruby-on-rails - rails : Find tasks that were created on a certain day? - 2

    我有一个任务列表(名称、starts_at),我试图在每日View中显示它们(就像iCal)。deftodays_tasks(day)Task.find(:all,:conditions=>["starts_atbetween?and?",day.beginning,day.ending]end我不知道如何将Time.now(例如“2009-04-1210:00:00”)动态转换为一天的开始(和结束),以便进行比较。 最佳答案 deftodays_tasks(now=Time.now)Task.find(:all,:conditio

  5. 什么是0day漏洞?如何预防0day攻击? - 2

    什么是0day漏洞?0day漏洞,是指已经被发现,但是还未被公开,同时官方还没有相关补丁的漏洞;通俗的讲,就是除了黑客,没人知道他的存在,其往往具有很大的突发性、破坏性、致命性。0day漏洞之所以称为0day,正是因为其补丁永远晚于攻击。所以攻击者利用0day漏洞攻击的成功率极高,往往可以达到目的并全身而退,而防守方却一无所知,只有在漏洞公布之后,才后知后觉,却为时已晚。“后知后觉、反应迟钝”就是当前安全防护面对0day攻击的真实写照!为了方便大家理解,中科三方为大家梳理当前安全防护模式下,一个漏洞从发现到解决的三个时间节点:T0:此时漏洞即0day漏洞,是已经被发现,还未被公开,官方还没有相

  6. 【云原生】SpringCloud-Spring Boot Starter使用测试 - 2

    目录SpringBootStarter是什么?以前传统的做法使用SpringBootStarter之后starter的理念:starter的实现: 创建SpringBootStarter步骤在idea新建一个starter项目、直接执行下一步即可生成项目。 在xml中加入如下配置文件:创建proterties类来保存配置信息创建业务类:创建AutoConfiguration测试如下:SpringBootStarter是什么? SpringBootStarter是在SpringBoot组件中被提出来的一种概念、简化了很多烦琐的配置、通过引入各种SpringBootStarter包可以快速搭建出一

  7. ruby - Rails 比较 date.end_of_day.to_datetime 和 date.to_datetime.end_of_day 返回的日期对象值时返回 false - 2

    ruby1.9.3dev(2011-09-23修订版33323)[i686-linux]轨道3.0.20最近为什么在与DateTimeonRails相关的RSpecs项目上工作我发现在给定日期以下语句发出的值date.end_of_day.to_datetime和date.to_datetime.end_of_day虽然它们表示相同的日期时间,但比较时返回false。为了确认这一点,我打开了Rails控制台并尝试了以下操作1.9.3dev:053>monday=Time.now.monday=>2013-02-2500:00:00+05301.9.3dev:054>monday.cla

  8. Spring Boot集成ElasticSearach - 2

    文章目录前言一、Elasticsearch版本介绍二、客户端种类三、客户端与版本兼容性四、引入Elasticsearch依赖包五、客户端配置六、Elasticsearch使用前言ElasticSearch是Elastic公司出品的一款功能强大的搜索引擎,被广泛的应用于各大IT公司,它的代码位于https://github.com/elastic/elasticsearch,目前是一个开源项目。ElasticSearch公司的另外两个开源产品Logstash、Kibana与ElasticSearch构成了著名的ELK技术栈。。他们三个共同形成了一个强大的生态圈。简单地说,Logstash负责数据

  9. Spring Security 6.0系列【32】授权服务器篇之默认过滤器 - 2

    有道无术,术尚可求,有术无道,止于术。本系列SpringBoot版本3.0.4本系列SpringSecurity版本6.0.2本系列SpringAuthorizationServer版本1.0.2源码地址:https://gitee.com/pearl-organization/study-spring-security-demo文章目录前言1.OAuth2AuthorizationServerMetadataEndpointFilter2.OAuth2AuthorizationEndpointFilter3.OidcProviderConfigurationEndpointFilter4.N

  10. IDEA 2022 创建 Spring Boot 项目详解 - 2

    如何用IDEA2022创建并初始化一个SpringBoot项目?目录如何用IDEA2022创建并初始化一个SpringBoot项目?0. 环境说明1.  创建SpringBoot项目 2.编写初始化代码0. 环境说明IDEA2022.3.1JDK1.8SpringBoot1.  创建SpringBoot项目        打开IDEA,选择NewProject创建项目。        填写项目名称、项目构建方式、jdk版本,按需要修改项目文件路径等信息。        选择springboot版本以及需要的包,此处只选择了springweb。        此处需特别注意,若你使用的是jdk1

随机推荐