@Autowired注入為null問題分析


 2018-11-24 22:57:33

問題說明

最近看到Spring事務,在學習過程中遇到一個很苦惱問題                                                  

搭建好Spring的啟動環境后出現了一點小問題

在啟動時候卻出現[java.lang.NullPointerException]

不過因為當時一個小小的疏忽 很low的問題 請往下看 ... 

工程結構

代碼片段

spring.xml

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4     xmlns:context="http://www.springframework.org/schema/context"
 5     xsi:schemaLocation="
 6         http://www.springframework.org/schema/beans
 7         http://www.springframework.org/schema/beans/spring-beans.xsd
 8         http://www.springframework.org/schema/context
 9         http://www.springframework.org/schema/context/spring-context.xsd">
10 
11     <!-- Spring注解掃描 -->
12     <context:component-scan base-package="com.*" />
13 
14     <!-- 1. 數據源對象: C3P0連接池 -->
15     <bean id="dataSource"
16         class="com.mchange.v2.c3p0.ComboPooledDataSource">
17         <property name="driverClass" value="org.h2.Driver"></property>
18         <property name="jdbcUrl"
19             value="jdbc:h2:tcp://192.168.190.1/~/test"></property>
20         <property name="user" value="sa"></property>
21         <property name="password" value="123"></property>
22     </bean>
23 
24     <!-- 2. JdbcTemplate工具類實例 -->
25     <bean id="jdbcTemplate"
26         class="org.springframework.jdbc.core.JdbcTemplate">
27         <property name="dataSource" ref="dataSource"></property>
28     </bean>
29 
30     <!-- 3.配置事務 -->
31     <bean id="dataSourceTransactionManager"
32         class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
33         <property name="dataSource" ref="dataSource"></property>
34     </bean>
35 
36 </beans>
Spring.xml

 

 Test.java

1 public class Test {
2     public static void main(String[] args) {
3         ClassPathXmlApplicationContext classPathXmlApplicationContext = new ClassPathXmlApplicationContext(
4                 "spring.xml");
5         ServiceIF service = (ServiceIF) classPathXmlApplicationContext.getBean("serviceImpl");
6         service.add("小王", 23);
7     }
8 }

TransactionUtil.java

 

 1 @Component("transactionUtil")
 2 public class TransactionUtil {
 3 
 4     /**
 5      * 初始化數據源
 6      */
 7     @Autowired
 8     private DataSourceTransactionManager dataSourceTransactionManager;
 9 
10     /**
11      * 開啟事務
12      * 
13      * @return
14      */
15     public TransactionStatus begin() {
16         TransactionStatus transaction = dataSourceTransactionManager.getTransaction(new DefaultTransactionDefinition());
17         System.out.println(" 開啟事務成功 ");
18         return transaction;
19     }
20 
21     /**
22      * 提交事物
23      * 
24      * @param transaction
25      */
26     public void commit(TransactionStatus transaction) {
27         dataSourceTransactionManager.commit(transaction);
28         System.out.println(" 事物提交成功 ");
29     }
30 
31     /**
32      * 回滾事務
33      * 
34      * @param transaction
35      */
36     public void rollback(TransactionStatus transaction) {
37         dataSourceTransactionManager.rollback(transaction);
38         System.err.println(" 事物進行回滾 ");
39     }
40 }
TransactionUtil.java

 

ServiceImpl.java

 1 @Service("serviceImpl")
 2 public class ServiceImpl implements ServiceIF {
 3 
 4     @Autowired
 5     TransactionUtil transactionUtil;
 6 
 7     private TransactionStatus transactionStatus = null;
 8 
 9     @Override
10     public void add(String name, Integer age) {
11         transactionStatus = transactionUtil.begin();
12         try {
13             new DaoImpl().add(name, age);
14             transactionUtil.commit(transactionStatus);
15         } catch (Exception e) {
16             System.err.println("ERROR >>> 執行出現異常 即將進行回滾操作");
17             transactionUtil.rollback(transactionStatus);
18         }
19     }
20 }

DaoImpl.java

 1 public class DaoImpl implements DaoIF{  2  3 /**  4  * 注入jdbc模板類  5 */  6  @Autowired  7 private JdbcTemplate jdbcTemplate;  8  9 /** 10  * 第一條插入語句 11 */ 12 private final String SQL_INSERT_01 = "insert into user values (?,?)"; 13 14 /** 15  * 添加sql執行 16  * 17  * @param name 18  * @param age 19 */ 20 public void add(String name, Integer age) { 21  jdbcTemplate.update(SQL_INSERT_01, name, age); 22  } 23 }

運行結果

 

 問題分析

 


解決思路

我在想 為什么會沒有注入進來呢 我明明加了@Autowired注解

后來猜到可能是Spring.xml配置的問題

看完也沒有問題 我就從Java Source一步一步看 發現....

我靠 我就猜測是不是如果用「new Object()」的方式創建實例后 其class中的Bean的注解會失效呢?

然后我嘗試在ServiceImpl.java中以注解的方式把DaoIF的實例注入到ServiceImpl,

並在DaoImpl.java的類上面添加@Repository,

把ServiceImpl.java中new DaoImpl()替換成注入的daoImpl。

 

改修代碼

ServiceImpl.java修改后

DaoImpl.java修改后


改修后調試

 


 

其實我懂得也不太多 Spring注入的流程那

首先他會把項目中target -> classes 目錄下的「.class」文件進行解析

通過Spring.xml中的「context:component-scan」進行注解掃描

如果這個路徑下的「.class」文件的類上面是否存在@Component聲明的注解

如果被此類注解修飾,Spring會把所有被注解修飾的bean進行實例化操作  供給@Autowired進行注入

(在spring注解的源碼中@Service和@Repository等等都繼承了@Component注解)

 

 

  結論 

在使用Spring的Bean容器時 千萬要確保

                配置的注解掃描路徑正確

                Jar的依賴是否存在

                是否在bean的上面加「@Service @Repository @Component … 」

                要細心 遇到異常不要緊 慢慢分析!!!

 


 

 

謝謝你這么忙還來我的看我的博客奧

如果有哪些地方寫的不好希望能夠指出!

為了變得更強 一起加油吧~

 

 

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM