Spring-02

作者: 50ef8076d671 | 来源:发表于2017-07-15 20:18 被阅读0次
注入自定义对象
上一篇文章说到了注入java内置对象的方法(Setter) 这次来说一说关于注入自定义类型对象的一些内容

Bean和注入

定义两个Bean如下所示:

public class User {
    private String name;
    private String sex;
    private Role role;
}

public class Role {
    private String role_name;
}

两个Bean里面都实现Get()和Set()方法 并且重写toString()方法

xml主配置文件内使用如下配置进行测试:

<bean id="user" class="com.test.model.User">
    <property name="name" value="wukong" />
    <property name="sex" value="男" />
    <!-- 此时name所指代的属性 不再是简单的数据类型 而是自定义的对象 -->
    <!-- 再次使用bean标签来进行实例化 -->
    <property name="role">
        <bean id="role" class="com.test.model.Role">
            <property name="role_name" value="斗战胜佛" />
        </bean>
    </property>
</bean>

在测试文件内(使用Junit)编写测试方法:

    @Test
    public void GetUserMessige(){
        ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
        //User指代xml文件内的bean标签的id
       //User.class指明User的类路径 利用框架映射帮助我们进行实例化 
        User user= ac.getBean("user",User.class);
        System.out.println(user);
    }

控制台输出:

  user [name=wukong, sex=男, role=role [role_name=斗战胜佛]]

但是这样的写法 会存在这样一个大问题:
我们在xml文件内定义了两个<bean> id分别为 user 和 role
但是我们只能获取到id为user的bean内的内容,role我们获取不到
进行如下测试:
xml配置文件内容保持上述不动 在Junit测试文件内新建测试方法 指向id为role的bean标签

@Test
public void GetRoleMessige(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
    Role role= ac.getBean("role",Role.class);
    System.out.println(role);
}

进行测试发现Junit报错:

image.png

报错内容为 : No bean named 'role' available
没有名为role的bean被定义
出现这个报错的原因很好理解
==>
我们的确是定义了两个bean
但是它们是包含与被包含的关系
role在user的内层 其作用范围在uesr内
我们发出了想要越过user来直接获取内层bean的请求
自然是获取不到

因此 我们在内层的bean上设置id 其实没有任何意义

综上 我们还有第二种解决方案
我们将role的配置单独书写 并将两个bean建立联系
代码如下:

  <bean id="user" class="com.test.model.User">
        <property name="name" value="wukong" />
        <property name="sex" value="男" />
        <property name="role">
            <!-- 使用<ref>标签来使两个独立的bean建立联系 -->
            <ref bean="role"/>
        </property>
    </bean>
    <bean id="role" class="com.test.model.Role">
        <property name="role_name" value="斗战胜佛" />
    </bean>
</beans>

存在这样一种简写方式:

  <property name="role" ref="role"/>
  <!--
        name="role"  指代类中的属性
        ref="role"  指代独立的bean的id值
  -->

可以将上面配置联系的代码进行简化书写代码

接口和注入

在开发当中我们经常会使用到接口 按照传统的开发模式 现在我们有了新的包结构 如下图:

image.png

在正常的开发模式下 我们需要在接口的实现类下进行接口回调 如下图


image.png

现在我们可以尝试通过注入的方式来完成操作
在xml文件内声明两个实现类:

    <bean  id="userDaoImpl" class="com.test.dao.impl.UserDaoImpl"/>
    <bean  id="userServiceImpl" class="com.test.service.impl.UserServiceImpl">
            <!--  
                UserDaoImpl userdaoimpl = new UserDaoImpl();
                UserDao userDao = userdaoimpl;
              -->
           <property name="userdao" ref="userDaoImpl"></property>
    </bean>

可以看到 我们通过注入方式 利用框架帮助我们完成了接口回调的操作 具体实现逻辑 看上述代码的注释部分

此时 我们可以直接通过UsereService进行测试了:

@Test
public void 测试UserService(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
    UserService userService= ac.getBean("userServiceImpl",UserService.class);
    userService.Test01();
}

注意:
在书写这段代码的时候 ac.getBean("userServiceImpl",UserService.class);
双引号括起来的地方 指向Service层bean的id
后面的类路径 需要书写接口的类路径 不是接口实现类(impl)的类路径

代码提示:其实本段代码 类似于上面所列出的User和Role的例子 不过是把实体bean转换成了接口及其实现类 一点都不复杂 千万别迷糊 hhhhh

构造函数注入方式

在前文 我们主要使用Setter注入方式 类中只有存在set方法才能够使用
下面我们来测试另一种注入方式:构造函数注入

我们更改User类里面的内容 如下:

private String name;
private String sex;
private Role role;

public User(){
    System.out.println("构造方法");
}
public void setName( String name ) {
    System.out.println("setName");
    this.name = name;
}
public void setSex( String sex ) {
    System.out.println("setSex");
    this.sex = sex;
}
public void setRole( Role role ) {
    System.out.println("setRole");
    this.role = role;
}

然后重写toString()方法
重新进行测试 结果如下:


image.png

可以看到具体的流程:
现在我们所写的bean标签的内容 是基于java默认为每一个类提供的无参的构造函数
每次加载bean标签 都会首先去寻找所引用的类中的构造方法 然后通过公开的Set()方法 往实例化好的对象里面添加值
所以我们每个类当中 都要存在一个无参的构造方法来进行控制反转

现在我们添加一个有参的构造函数:

public User( String name , String sex ) {
    super();
    this.name = name;
    this.sex = sex;
}

在xml配置文件内添加如下代码:

    <bean id="user1" class="com.test.model.User">
        <constructor-arg name="name" value="八戒"/>
        <constructor-arg name="sex" value="男"/>
    </bean>

新建bean标签做注入 <constructor-arg>标签对应类中的构造函数
因为在User类中的有参构造函数有两个参数 所以<constructor-arg>标签要写两个 分别对应两个参数 并给予初始值

此时新建测试方法:

@Test
public void 测试User构造函数(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
    User user= ac.getBean("user1",User.class);
    System.out.println(user);

控制台输出:


image.png

产生 构造方法 SetName 等结果
是因为我们同一个xml文件中配置了使用无参构造函数 使用setter方式进行注入
xml文件一被加载 内部所有的bean都会被进行实例化

别惊慌hhhh

值得注意的是 在 <constructor-arg>标签内
存在index属性 用来标识形参的位置 起始值为1
存在type属性 指明参数类型
如刚才我们对构造函数进行的操作,完整内容应为如下:

    <bean id="user1" class="com.test.model.User">
        <constructor-arg name="name" index="0" type="java.lang.String" value="八戒"/>
        <constructor-arg name="sex" index="1" type="java.lang.String" value="男"/>
    </bean>

关于构造函数注入自定义类型 类似于下面的语句就可以完成:
指明类中属性 指明引用的bean(我们在上文曾新建了一个专门用来实例化Role类型的bean)

    <constructor-arg name="role" ref="role" />

相关文章

  • Spring-02

    注入自定义对象 上一篇文章说到了注入java内置对象的方法(Setter) 这次来说一说关于注入自定义类型对象的...

  • [java]42、Spring-02

    1、Converter 1、Spring已经内置了基本的类型转换功能,比如String 转int、String 转...

网友评论

      本文标题:Spring-02

      本文链接:https://www.haomeiwen.com/subject/jlaohxtx.html