2018-11-24 22:57:33 問題說明 最近看到Spring事務,在學習過程中遇到一個很苦惱問題 搭建好Spring的啟動環境後出現了一點小問題 在啟動時候卻出現[java.lang.NullPointerException] 不過因為當時一個小小的疏忽 很low的問題 請往下看 ... ...
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 … 」
要細心 遇到異常不要緊 慢慢分析!!!
謝謝你這麼忙還來我的看我的博客奧
如果有哪些地方寫的不好希望能夠指出!
為了變得更強 一起加油吧~