轉(zhuǎn)自:http://lihaiyan.javaeye.com/blog/127796
業(yè)務(wù)層
1、業(yè)務(wù)層接口
"面向接口而非面向類編程"是Spring不遺余力所推薦的編程原則,這條原則也已經(jīng)為大部開發(fā)者所接受;此外,JDK的動(dòng)態(tài)代理只對(duì)接口有效,否則必須使用CGLIB生成目標(biāo)類的子類。我們依從于Spring的倡導(dǎo)為業(yè)務(wù)類定義一個(gè)接口:
代碼 7 業(yè)務(wù)層操作接口
1. public interface FileService
2. {
3. void save(FileActionForm fileForm);//將提交的上傳文件保存到數(shù)據(jù)表中
4. List getAllFile();//得到T_FILE所示記錄
5. void write(OutputStream os,String fileId);//將某個(gè)文件的文件數(shù)據(jù)寫出到輸出流中
6. String getFileName(String fileId);//獲取文件名
7. } |
其中save(FileActionForm fileForm)方法,將封裝在fileForm中的上傳文件保存到數(shù)據(jù)庫(kù)中,這里我們使用FileActionForm作為方法入?yún)ⅲ現(xiàn)ileActionForm是Web層的表單數(shù)據(jù)對(duì)象,它封裝了提交表單的數(shù)據(jù)。將FileActionForm直接作為業(yè)務(wù)層的接口入?yún)ⅲ喈?dāng)于將Web層傳播到業(yè)務(wù)層中去,即將業(yè)務(wù)層綁定在特定的Web層實(shí)現(xiàn)技術(shù)中,按照分層模型學(xué)院派的觀點(diǎn),這是一種反模塊化的設(shè)計(jì),但在"一般"的業(yè)務(wù)系統(tǒng)并無需提供多種UI界面,系統(tǒng)Web層將來切換到另一種實(shí)現(xiàn)技術(shù)的可能性也微乎其微,所以筆者覺得沒有必要為了這個(gè)業(yè)務(wù)層完全獨(dú)立于調(diào)用層的過高目標(biāo)而去搞一個(gè)額外的隔離層,浪費(fèi)了原材料不說,還將系統(tǒng)搞得過于復(fù)雜,相比于其它原則,"簡(jiǎn)單"始終是最大的一條原則。
getAllFile()負(fù)責(zé)獲取T_FILE表所有記錄,以便在網(wǎng)頁上顯示出來。
而getFileName(String fileId)和write(OutputStream os,String fileId)則用于下載某個(gè)特定的文件。具體的調(diào)用是將Web層將response.getOutputStream()傳給write(OutputStream os,String fileId)接口,業(yè)務(wù)層直接將文件數(shù)據(jù)輸出到這個(gè)響應(yīng)流中。具體實(shí)現(xiàn)請(qǐng)參見錯(cuò)誤!未找到引用源。節(jié)下載文件部分。
2、業(yè)務(wù)層接口實(shí)現(xiàn)類
FileService的實(shí)現(xiàn)類為FileServiceImpl,其中save(FileActionForm fileForm)的實(shí)現(xiàn)如下所示:
代碼 8 業(yè)務(wù)接口實(shí)現(xiàn)類之save()
1. …
2. public class FileServiceImpl
3. implements FileService
4. {
5. private TfileDAO tfileDAO;
6. public void save(FileActionForm fileForm)
7. {
8. Tfile tfile = new Tfile();
9. try
10. {
11. tfile.setFileContent(fileForm.getFileContent().getFileData());
12. }
13. catch (FileNotFoundException ex)
14. {
15. throw new RuntimeException(ex);
16. }
17. catch (IOException ex)
18. {
19. throw new RuntimeException(ex);
20. }
21. tfile.setFileName(fileForm.getFileContent().getFileName());
22. tfile.setRemark(fileForm.getRemark());
23. tfileDAO.save(tfile);
24. }
25. …
26. } |
在save(FileActionForm fileForm)方法里,完成兩個(gè)步驟:
其一,象在水桶間倒水一樣,將FileActionForm對(duì)象中的數(shù)據(jù)倒入到Tfile對(duì)象中;
其二,調(diào)用TfileDAO保存數(shù)據(jù)。
需要特別注意的是代碼的第11行,F(xiàn)ileActionForm的fileContent屬性為org.apache.struts.upload.FormFile類型,F(xiàn)ormFile提供了一個(gè)方便的方法getFileData(),即可獲取文件的二進(jìn)制數(shù)據(jù)。通過解讀FormFile接口實(shí)現(xiàn)類DiskFile的原碼,我們可能知道FormFile本身并不緩存文件的數(shù)據(jù),只有實(shí)際調(diào)用getFileData()時(shí),才從磁盤文件輸入流中獲取數(shù)據(jù)。由于FormFile使用流讀取方式獲取數(shù)據(jù),本身沒有緩存文件的所有數(shù)據(jù),所以對(duì)于上傳超大體積的文件,也是沒有問題的;但是,由于數(shù)據(jù)持久層的Tfile使用byte[]來緩存文件的數(shù)據(jù),所以并不適合處理超大體積的文件(如100M),對(duì)于超大體積的文件,依然需要使用java.sql.Blob類型以常規(guī)流操作的方式來處理。
此外,通過FileForm的getFileName()方法就可以獲得上傳文件的文件名,如第21行代碼所示。
write(OutputStream os,String fileId)方法的實(shí)現(xiàn),如代碼 9所示:
代碼 9 業(yè)務(wù)接口實(shí)現(xiàn)類之write()
1. …
2. public class FileServiceImpl
3. implements FileService
4. {
5.
6. public void write(OutputStream os, String fileId)
7. {
8. Tfile tfile = tfileDAO.findByFildId(fileId);
9. try
10. {
11. os.write(tfile.getFileContent());
12. os.flush();
13. }
14. catch (IOException ex)
15. {
16. throw new RuntimeException(ex);
17. }
18. }
19. …
20. } |
write(OutputStream os,String fileId)也簡(jiǎn)單地分為兩個(gè)操作步驟,首先,根據(jù)fileId加載表記錄,然后將fileContent寫入到輸出流中。
3、Spring事務(wù)配置
下面,我們來看如何在Spring配置文件中為FileService配置聲明性的事務(wù)
1. <beans>
2. …
3. <bean id="transactionManager"
4. class="org.springframework.orm.hibernate3.HibernateTransactionManager">
5. <property name="sessionFactory" ref="sessionFactory"/>
6. </bean>
7. <!-- 事務(wù)處理的AOP配置 //-->
8. <bean id="txProxyTemplate" abstract="true"
9. class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
10. <property name="transactionManager" ref="transactionManager"/>
11. <property name="transactionAttributes">
12. <props>
13. <prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
14. <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>
15. <prop key="save">PROPAGATION_REQUIRED</prop>
16. <prop key="write">PROPAGATION_REQUIRED,readOnly</prop>
17. </props>
18. </property>
19. </bean>
20. <bean id="fileService" parent="txProxyTemplate">
21. <property name="target">
22. <bean class="sshfile.service.FileServiceImpl">
23. <property name="tfileDAO" ref="tfileDAO"/>
24. </bean>
25. </property>
26. </bean>
27. </beans> |
Spring的事務(wù)配置包括兩個(gè)部分:
其一,定義事務(wù)管理器transactionManager,使用HibernateTransactionManager實(shí)現(xiàn)事務(wù)管理;
其二,對(duì)各個(gè)業(yè)務(wù)接口進(jìn)行定義,其實(shí)txProxyTemplate和fileService是父子節(jié)點(diǎn)的關(guān)系,本來可以將txProxyTemplate定義的內(nèi)容合并到fileService中一起定義,由于我們的系統(tǒng)僅有一個(gè)業(yè)務(wù)接口需要定義,所以將其定義的一部分抽象到父節(jié)點(diǎn)txProxyTemplate中意義確實(shí)不大,但是對(duì)于真實(shí)的系統(tǒng),往往擁有為數(shù)眾多的業(yè)務(wù)接口需要定義,將這些業(yè)務(wù)接口定義內(nèi)容的共同部分抽取到一個(gè)父節(jié)點(diǎn)中,然后在子節(jié)點(diǎn)中通過parent進(jìn)行關(guān)聯(lián),就可以大大簡(jiǎn)化業(yè)務(wù)接口的配置了。
父節(jié)點(diǎn)txProxyTemplate注入了事務(wù)管理器,此外還定義了業(yè)務(wù)接口事務(wù)管理的方法(允許通過通配符的方式進(jìn)行匹配聲明,如前兩個(gè)接口方法),有些接口方法僅對(duì)數(shù)據(jù)進(jìn)行讀操作,而另一些接口方法需要涉及到數(shù)據(jù)的更改。對(duì)于前者,可以通過readOnly標(biāo)識(shí)出來,這樣有利于操作性能的提高,需要注意的是由于父類節(jié)點(diǎn)定義的Bean僅是子節(jié)點(diǎn)配置信息的抽象,并不能具體實(shí)現(xiàn)化一個(gè)Bean對(duì)象,所以需要特別標(biāo)注為abstract="true",如第8行所示。
fileService作為一個(gè)目標(biāo)類被注入到事務(wù)代理器中,而fileService實(shí)現(xiàn)類所需要的tfileDAO實(shí)例,通過引用3.2節(jié)中定義的tfileDAO Bean注入。
posted on 2008-04-10 15:01
阿偉 閱讀(241)
評(píng)論(0) 編輯 收藏 所屬分類:
框架整合