淺談@Value和@Bean的執行順序問題
問題描述
使用@Autowired處理多個同種類型的bean,出現@Value和@Bean的執行順序問題。
首先使用掃描包+註解的方式註冊User類型的不同bean,分別是user、user1,註冊方式如下
package com.fanyinhang.bean; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:11 */ @NoArgsConstructor @Data @AllArgsConstructor @Component public class User { private Integer id; private String name; }
該方式得到User類型的名為user的bean
package com.fanyinhang.config; import com.fanyinhang.bean.User; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration @ComponentScan(value={"com.fanyinhang.dao"}) public class AnnotationConfig { @Bean() public User user1(){ return new User(2,"李四"); } }
UserDao配置如下:
package com.fanyinhang.dao; import com.fanyinhang.bean.User; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Repository; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:17 */ @Repository public class UserDao { @Autowired() private User user1; @Override public String toString() { return "UserDao{" + "user1=" + user1 + '}'; } }
import com.fanyinhang.config.AnnotationConfig; import com.fanyinhang.dao.UserDao; import org.junit.Test; import org.springframework.context.annotation.AnnotationConfigApplicationContext; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:18 */ public class Test3 { @Test public void testAutowired(){ AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AnnotationConfig.class); UserDao userDao = context.getBean(UserDao.class); System.out.println(userDao); } }
輸出結果如下:
UserDao{user1=User(id=2, name=李四)}
沒有加入@Value註解時是沒有問題的,但是加入瞭@Value之後
package com.fanyinhang.bean; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:11 */ @NoArgsConstructor @Data @AllArgsConstructor @Component public class User { @Value("1") private Integer id; @Value("張三") private String name; }
再次運行testWired方法後
結果輸出如下:
UserDao{user1=User(id=1, name=張三)}
為什麼會出現這種情況?
一開始,怎麼也想不通,查看網上的資料大多數是說@Bean和@Value有執行順序這一說法。
為瞭驗證這一說法,做個對比試驗
去掉瞭一個@Value(“張三”)
package com.fanyinhang.bean; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:11 */ @NoArgsConstructor @Data @AllArgsConstructor @Component public class User { @Value("1") private Integer id; //@Value("張三") private String name; }
結果輸出如下:
UserDao{user1=User(id=1, name=李四)}
問題原因
@Value和@Bean在不同文件下時,@Bean比@Value先執行。這樣就回導致@Bean註入的值失效。
解決辦法
網上說@Value和@Bean在不同文件下時,@Value比@Bean先執行,因此,我做瞭如下設置
把User.java下的@Value註解去掉,而是將@Value註解放在@bean同一文件下
package com.fanyinhang.bean; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import org.springframework.stereotype.Component; /** * @author fanyinhang * @version 1.0 * @create 2019/10/8-19:11 */ @NoArgsConstructor @Data @AllArgsConstructor @Component public class User { private Integer id; private String name; }
package com.fanyinhang.config; import com.fanyinhang.bean.User; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration @ComponentScan(value={"com.fanyinhang.dao"}) public class AnnotationConfig { @Bean() public User user1(@Value("1") Integer id,@Value("張三") String name){ return new User(2,"李四"); } }
此時再運行測試方式,輸出結果如下:
UserDao{user1=User(id=2, name=李四)}
總結
以上為個人經驗,希望能給大傢一個參考,有什麼理解不到位的地方,請大傢多多指正,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Spring使用IOC與DI實現完全註解開發
- Spring Boot 整合 Thymeleaf 實例分享
- 圖文教程教你IDEA中的Spring環境搭建+簡單入門
- Springboot中使用lombok的@Data註解方式
- springboot日志切面通用類實例詳解