【Spring框架全系列】方法注解@Bean的使用

一、如何使用方法注解

🌃1、前面介绍过,类注解是写到类上面的,那么方法注解是写到对应的方法上的;

    @Bean(name = "user1")     public User getUser1() {        User user = new User();        user.setId(1);        user.setName("张三");        return user;    }

2、方法注解的话是不能够单独使用的,如果我们只给一个方法注解@Bean的话,看是否能运行;

 ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml"); User user = context.getBean(User.class);//只使用类属性 System.out.println(user);

运行结果:

如何解决?我们只需要在方法注解的类中添加一个类注解即可;

再次运行:

注意Bean 的命名规则,当没有设置 name 属性时,那么 bean 默认的名称就是方法名,当设置了 name 属性之后,只能通过重命名的 name 属性对应的值来获取,也就是说重命名之后,再使用方法名就获取不到 bean 对象;

比如已经给bean重命名了,我们在启动类App中通过方法名来获取Bean对象;

看运行结果:

二、同一类型的对象注入多次的问题

1、在UserBeans类中写两个对象注解;

@Componentpublic class UserBeans {    @Bean(name = "user1") // 【注意事项:只使用一个 @Bean 是无法将对象存储到容器中的】    public User getUser1() {        User user = new User();        user.setId(1);        user.setName("张三");        return user;    }     @Bean(name = "user2") // 【注意事项:只使用一个 @Bean 是无法将对象存储到容器中的】    public User getUser2() {        User user = new User();        user.setId(2);        user.setName("李四");        return user;    }

2、通过启动类来获取对象;

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");User user = context.getBean(User.class);//只使用类属性System.out.println(user);

运行结果:

翻译过来的意思就是:没有可用的“com.User”类型的合格bean:应为单个匹配bean,但找到2:user1,user2;

那么,如何解决呢?这里我们提供了三种解决方案;

1、精准的描述bean的名称(将注入的名称写对)

比如我们在启动类中使用id+class的形式来取对象;

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");User user = context.getBean("user1",User.class);//只使用类属性System.out.println(user);

运行结果:没有问题;

2、使用@Resource设置name的方式来重命名注入的对象;

1、这里我们通过UserController类来进行演示;注意我们添加的注解@Resource,下面新写了一个对象user1;

package com; import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;import org.springframework.stereotype.Controller; import javax.annotation.Resource; @Controllerpublic class UserController {    @Resource    private User user1;    public void sayHi(){        System.out.println("User"+user1);    }}

启动类App中代码:

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");UserController userController = context.getBean(UserController.class);userController.sayHi();

运行结果:

3、使用@AutoWired+@Qualifier来筛选bean对象;

UserController代码:

package com; import org.springframework.beans.factory.annotation.Autowired;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;import org.springframework.stereotype.Controller; import javax.annotation.Resource; @Controllerpublic class UserController {    @Autowired    @Qualifier(value = "user2")    private User user2;    public void sayHi(){        System.out.println("User"+user2);    }}

运行结果:

对象注入的三种方法

1、属性注入

属性注⼊是使⽤ @Autowired 实现的,将 Service 类注⼊到 Controller 类中。

package com;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Controller;import javax.annotation.Resource;/** * 根据属性实现 bean 对象的注入 */@Controllerpublic class UserController2 {    // 对象注入方式1:属性注入    @Autowired    private UserService userService;    public void sayHi() {        userService.sayHi();    } }

启动类App:

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");UserController2 userController2 = context.getBean(UserController2.class);userController2.sayHi();

运行结果:

2、构造方法注入

package com; import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Controller; /** * 使用构造方法实现 bean 注入 */@Controllerpublic class UserController3 {     private UserService userService;     // 构造方法注入(官广推荐写法)    @Autowired    public UserController3(UserService userService) {//        userService = new UserService(); // 传统的写法        this.userService = userService;    }        public void sayHi() {        userService.sayHi();    } }

注:如果当前类只存在一个构造方法,那么@Autowired注解可以省略;

启动类App:

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");UserController3 userController3 = context.getBean(UserController3.class);userController3.sayHi();

运行结果:

3、Setter方法注入

package com; import org.springframework.stereotype.Controller; import javax.annotation.Resource; /** * 使用 Setter 实现 bean 注入 */@Controllerpublic class UserController4 {     private UserService userService;     @Resource    public void setUserService(UserService userService) {        this.userService = userService;    }     public void sayHi() {        userService.sayHi();    } }

启动类App:

ApplicationContext context= new ClassPathXmlApplicationContext("spring-config.xml");UserController4 userController4 = context.getBean(UserController4.class);userController4.sayHi();

运行结果:

属性注入和构造方法注入以及 Setter 注入的区别是什么?

🚈1、属性注入:特点写法简单。但是通用性不好,它只能运行在 IoC 容器下,如果是非 IOC 容器就会出现问题。

🚅2、Setter 注入:早期 Spring 版本推荐的写法,Setter 注入通用性没有构造方法注入通用。

🚄3、构造方法注入: 通用性更好、它能确保在使用注入对象之前,此注入对象一定初始化过了。当构造方法注入参数过多时,此时开发者就要检查自己所写的代码是否符合单一

设计原则

的规范了,此注入方式也是 spring后期版本中推荐的注入方式。

注意:@Resource注解只支持属性注入和Setter注入,不支持构造方法注入;

@Resource VS @Autowired?

🎑出身不同: @Resource 来着与 JDK (Java 亲儿子) ,@Autowired 是 Spring 框架提供的。

🏚2、用法不同: @Autowired 支持属性注入、构造方法注入和 Setter 注入,而 @Resource 不支持构造方法注入。

🏫3、支持的参数不同: @Resource 支持更多的参数设置,比如 name、type 设置;@Autowired 只支持required 参数设置。

💡💡OK,今天的学习内容就到这里啦,至此,Spring更简单的读取和存储对象就已经更新完毕了,下篇博客我们会继续更新"bean的作用域和生命周期",欢迎阅读,感谢订阅!!

全部评论

相关推荐

三年之期已到我的offer快到碗里来:9硕都比不上9本
点赞 评论 收藏
分享
我也曾抱有希望:说的好直白
点赞 评论 收藏
分享
1 收藏 评论
分享
牛客网
牛客企业服务