您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關(guān)Spring是什么意思,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
首先我們來簡單介紹下Spring。
Spring是一個開源的輕量級的 Java 開發(fā)框架, 具有控制反轉(zhuǎn)(IoC)和面向切面(AOP)兩大核心。Java Spring 框架通過聲明式方式靈活地進行事務的管理,提高開發(fā)效率和質(zhì)量。
Spring 框架不僅限于服務器端的開發(fā)。從簡單性、可測試性和松耦合的角度而言,任何 Java 應用都可以從 Spring 中受益。Spring 框架還是一個超級粘合平臺,除了自己提供功能外,還提供粘合其他技術(shù)和框架的能力。
接下來我們來詳細介紹下具體內(nèi)容。
控制反轉(zhuǎn)(IOC)
大概以前,業(yè)務邏輯層的代碼很有可能這樣寫:
public class PersonServiceBean { private PersonDao personDao = new PersonDaoBean(); public void save(Person person){ personDao.save(person); } }
從上可看出PersonDaoBean是在應用內(nèi)部創(chuàng)建及維護的。所謂控制反轉(zhuǎn)就是應用本身不負責依賴對象的創(chuàng)建及維護,依賴對象的創(chuàng)建及維護是由外部容器負責的。這樣控制權(quán)就由應用轉(zhuǎn)移到了外部容器,控制權(quán)的轉(zhuǎn)移就是所謂反轉(zhuǎn)。
依賴注入(Dependency Injection)
當我們把依賴對象交給外部容器負責創(chuàng)建,那么PersonServiceBean類可以改成如下:
public class PersonServiceBean { private PersonDao personDao ; // 通過構(gòu)造器參數(shù),讓容器把創(chuàng)建好的依賴對象注入進PersonServiceBean,當然也可以使用setter方法進行注入。 public PersonServiceBean(PersonDao personDao){ this.personDao=personDao; } public void save(Person person){ personDao.save(person); } }
所謂依賴注入就是指:在運行期,由外部容器動態(tài)地將依賴對象注入到組件中。
為何要使用Spring?
至少在我看來,在項目中引入Spring立即可以帶來下面的好處:
降低組件之間的耦合度,實現(xiàn)軟件各層之間的解耦。
可以使用容器提供的眾多服務,如:事務管理服務、消息服務等等。當我們使用容器管理事務時,開發(fā)人員就不再需要手工控制事務,也不需處理復雜的事務傳播。
容器提供單例模式支持,開發(fā)人員不再需要自己編寫實現(xiàn)代碼。
容器提供了AOP技術(shù),利用它很容易實現(xiàn)如權(quán)限攔截、運行期監(jiān)控等功能。
容器提供的眾多輔作類,使用這些類能夠加快應用的開發(fā),如: JdbcTemplate、HibernateTemplate。
Spring對于主流的應用框架提供了集成支持,如:集成Hibernate、JPA、Struts等,這樣更便于應用的開發(fā)。
使用Spring的好處
上面我們就已詳細列出了使用Spring框架帶來的好處,我們僅就第二點進行詳細說明之。當使用Spring框架時,我們可以使用容器提供的眾多服務。
試想若要是不使用Spring框架,那么使用Hibernate框架進行事務操作就應是:
Hibernate的事務操作:
public void save(){ Session session = sessionFactory.getCurrentSession(); session.beginTransaction(); Info info = new Info("Spring框架"); info.setContent("阿昀手把手教你學習Spring框架"); session.save(info); session.getTransaction().commit(); }
既不使用Spring框架,也不使用Hibernate框架,直接使用最原始的JDBC技術(shù)進行事務操作,代碼就應是:
JDBC的事務操作:
Connection conn = null; try { ...... conn.setAutoCommit(false); Statement stmt = conn.createStatement(); stmt.executeUpdate("update person where name='葉天'"); conn.commit(); ...... } catch (Exception e) { conn.rollback(); } finally { conn.close(); }
而如果使用Spring框架,那我們就不再需要手工控制事務了。另外,如果使用Spring框架,我們也不需要處理復雜的事務傳播行為了。我們舉例子來說明之。
(視頻教程推薦:java課程)
例如,有代碼:
public void payment(){ Bean1.update(); // 更新金額 Bean2.save(); // 記錄操作日志 }
public class Bean1 { public void update(){ // 注意:下面省略了一些代碼 Connection conn = null; conn.setAutoCommit(false); Statement.executeUpdate("update account set amount=? where id=?"); } }
public class Bean2 { public void save(){ // 注意:下面省略了一些代碼 Connection conn = null; conn.setAutoCommit(false); Statement.executeUpdate("insert into Log (content) values (?)"); } }
如果我們不使用Spring框架,針對下面這兩種業(yè)務需求,我們該如何做呢?
第1種可能的業(yè)務需求:要求Bean1.update()和Bean2.save()在同一個事務中執(zhí)行。
第2種可能的業(yè)務需求:要求不管Bean1.update()的事務是否成功,都需要記錄操作日志。
若要是不使用Spring框架,針對第1種可能的業(yè)務需求,我們的解決辦法用代碼來表示就是:
public void payment(){ Connection conn = null; conn.setAutoCommit(false); Bean1.update(conn); // 更新金額 Bean2.save(conn); // 記錄操作日志 // ...提交或回滾事務 }
public class Bean1 { public void update(Connection conn){ // 注意:下面省略了一些代碼 Statement.executeUpdate("update account set amount=? where id=?"); } }
public class Bean2 { public void save(Connection conn){ // 注意:下面省略了一些代碼 Statement.executeUpdate("insert into Log (content) values (?)"); } }
針對第2種可能的業(yè)務需求,我們不需要修改代碼就可完成,因為Bean1.update()開啟了一個事務,Bean2.save()同樣也開啟了一個事務,Bean1.update()開啟的事務的回滾不會影響到Bean2.save()開啟的事務。
倘若使用Spring框架,我們只需要通過聲明式的事務屬性配置就可以輕松地實現(xiàn)這兩種業(yè)務需求。
要求Bean1.update()和Bean2.save()在同一個事務中執(zhí)行。我們只須將代碼改為:
@Transactional(propagation=Propagation.Required) public void payment(){ Bean1.update(); // 更新金額 Bean2.save(); // 記錄日志 }
public class Bean1 { @Transactional(propagation=Propagation.Required) public void update(){ executeUpdate("update account set amount=? where id=?"); } }
public class Bean2 { @Transactional(propagation=Propagation.Required) public void save(){ executeUpdate("insert into Log (content) values (?)"); } }
要求不管Bean1.update()的事務是否成功,都需要記錄日志。我們只須將代碼改為:
@Transactional(propagation=Propagation.Required) public void payment(){ Bean1.update(); // 更新金額 Bean2.save(); // 記錄日志 }
public class Bean1 { @Transactional(propagation=Propagation.Required) public void update(){ executeUpdate("update account set amount=? where id=?"); } }
public class Bean2 { @Transactional(propagation=Propagation.RequiresNew) public void save(){ executeUpdate("insert into Log (content) values (?)"); } }
輕量級與重量級概念的劃分
經(jīng)常會有人問到Spring是屬于輕量級框架,還是屬于重量級框架呢?其實劃分一個應用是否屬于輕量級還是重量級,主要看它使用了多少服務。使用的服務越多,容器要為普通java對象做的工作就越多,必然會影響到應用的發(fā)布時間或者是運行性能。
對于Spring容器來說,它提供了很多服務,但這些服務并不是默認為應用打開的,應用需要某種服務,還需要指明使用該服務,如果應用使用的服務很少,如:只使用了Spring核心服務,那么我們可以認為此時應用屬于輕量級的,如果應用使用了Spring提供的大部分服務,這時應用就屬于重量級的。目前EJB容器就因為它默認為應用提供了EJB規(guī)范中所有的功能,所以它屬于重量級。
關(guān)于Spring是什么意思就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。