<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆 - 3, 文章 - 152, 評(píng)論 - 17, 引用 - 0

          數(shù)據(jù)加載中……

          實(shí)例介紹Struts+Spring+Hibernate開(kāi)發(fā)

          一 介紹 本文并不想介紹Struts,Spring,Hibernate的原理系統(tǒng)架構(gòu)等,本文地目的是通過(guò)一個(gè)較復(fù)雜地實(shí)例介紹如何整合Struts,Spring,Hibernate,網(wǎng)上現(xiàn)有的例子雖然也能達(dá)到目的,但功能都比較單一,復(fù)雜的例子時(shí)會(huì)有意想不到的麻煩。本文對(duì)讀者假設(shè)已經(jīng)具備了以上框架的基礎(chǔ)知識(shí)。以及那些已經(jīng)了解Struts,Spring,Hibernate的基本概念,但是還沒(méi)有親身在較復(fù)雜的項(xiàng)目中體驗(yàn)Struts+Spring+Hibernate的開(kāi)發(fā)人員。 1 Struts 雖然不打算過(guò)多介紹Struts的原理,但是大概介紹一下還是有必要的。Struts本身就是 MVC 在這里負(fù)責(zé)將用戶數(shù)據(jù)傳人業(yè)務(wù)層,以及 將業(yè)務(wù)層處理的結(jié)果返回給用戶,此系統(tǒng)屬于較簡(jiǎn)單WEB應(yīng)用,采用了OpenSessionInView模式處理LazyLoad問(wèn)題,這樣我們可以在用戶視圖中使用 get,set方法來(lái)方便地獲取關(guān)聯(lián)對(duì)象。為了處理龐大的Action和ActionForm問(wèn)題,在此我門準(zhǔn)備使用DynaActionForm (DynaValidatorForm)和DispatchAction以及 動(dòng)態(tài)驗(yàn)證框架 來(lái)解決。及使用Tile來(lái)解決框架問(wèn)題 。使用自定義標(biāo)簽處理分頁(yè)和身份驗(yàn)證問(wèn)題。 2 Spring Spring Framework最得以出名的是與Hibernate的無(wú)縫鏈接,雖然Spring 對(duì)Hibernate提供了90%以上的封裝,使我們不必去關(guān)心Session 的建立,關(guān)閉,以及事務(wù)使我們能夠?qū)P牡年P(guān)注業(yè)務(wù)邏輯。但是一些特殊情況如 有時(shí)需要Query以及Criteria 對(duì)象,分頁(yè)等,Spring不能給我們提供支持,總不能每次都在你的DAO上寫個(gè)HibernateCallBackup()吧?Spring的作用不是把Hibernate再封裝一層,而是讓你接觸不到Hibernate的API,而是幫助你管理好Session和Transaction。 在這里解決方法是:首先 寫一個(gè)IBase 的接口,和一個(gè)BaseDao的實(shí)現(xiàn)。在實(shí)現(xiàn)中仿照HibernateTemplate,將其功能一一實(shí)現(xiàn),同時(shí)考慮到Spring 未能支持的地方,我們不得已只好自己來(lái)管理Session,因此加入public Session openSession(),public Query getQuery(String sql),public Criteria getCriteria(Class clazz),以及分頁(yè)的方法。 然后為每一個(gè)Entity 都建立繼承于以上類的IEntity,與EntityDao。這里可以根據(jù)需求對(duì)Entity加入特殊的方法實(shí)現(xiàn),如 在 StudentsDao.java 中加入類似用戶身份驗(yàn)證等。以上就是數(shù)據(jù)訪問(wèn)層。接下來(lái)在Service層中通過(guò)對(duì)dao的引用完成業(yè)務(wù)邏輯方法。在下面的例子中我們分別為學(xué)生模塊,教師模塊,管理員模塊構(gòu)建Service層,StudentsServiceImpl,TeachersServiceImpl,AdminServiceImpl。 3 Hibernate 有了Spring的封裝,我們要對(duì)Hibernate做的就是正確實(shí)現(xiàn)對(duì)象關(guān)系的映射。由于此處處于系統(tǒng)的最底層,準(zhǔn)確無(wú)誤的實(shí)現(xiàn)對(duì)象之間的關(guān)聯(lián)關(guān)系映射將起著至關(guān)重要的作用。 總之,理解了Struts,Spring,Hibernate地原理以及之間的關(guān)系之后,剩下的工作就如同在以Spring為核心的Struts為表現(xiàn)的框架中堆積木。 下圖可以更好的幫助我們理解Struts,Spring,Hibernate之間的關(guān)系。 pic1.JPG 二 案例簡(jiǎn)述: 設(shè)計(jì)思路主要源于 大學(xué)選修課,該系統(tǒng)可以方便處理學(xué)生在課程選報(bào),學(xué)分查詢,成績(jī)查詢,以及 成績(jī)發(fā)布等。 系統(tǒng)以班級(jí)為核心,一門課程可以對(duì)應(yīng)多個(gè)班級(jí),一名教師也可以帶不同的班級(jí),學(xué)生可以選報(bào)不同課程所對(duì)應(yīng)的班級(jí),班級(jí)自身有目前人數(shù),和最大人數(shù),以及上課時(shí)間,上課地點(diǎn)的屬性。 學(xué)生在選報(bào)班級(jí)之后,班級(jí)的人數(shù)會(huì)自動(dòng)加一,直到等于最大人數(shù)時(shí),其他學(xué)生將會(huì)有人數(shù)已滿的錯(cuò)誤提示。同理如果學(xué)生選擇了同一課程的不同班級(jí),也將收到錯(cuò)誤提示。學(xué)生有密碼,系別,學(xué)分,地址,電話等屬性。 教師在系統(tǒng)中主要負(fù)責(zé)成績(jī)發(fā)布,教師可以對(duì)其所帶的班級(jí)的學(xué)生的成績(jī)修改,系統(tǒng)會(huì)以成績(jī)是否大于等于60來(lái)判斷學(xué)生是否通過(guò)考試,如果通過(guò)會(huì)將該課程的學(xué)分累加到學(xué)生學(xué)分,同樣如果教師二次修改了成績(jī),而且小于60,系統(tǒng)會(huì)在學(xué)生學(xué)分上扣掉該課程的分?jǐn)?shù)。 課程在系統(tǒng)中具體體現(xiàn)為班級(jí),自身帶有學(xué)分屬性。 系有編號(hào),名稱的屬性,同時(shí)可以作為聯(lián)系教師,課程,學(xué)生的橋梁。 功能模塊 l 身份驗(yàn)證模塊: 根據(jù)用戶名,密碼,用戶類別 轉(zhuǎn)發(fā)用戶到不同的模塊。 l 學(xué)生模塊: 查看課程,查看班級(jí),選報(bào)課程,查看己選課程,成績(jī)查詢。 l 教師模塊: 錄入成績(jī) l 管理員模塊:對(duì)學(xué)生,教師,課程,班級(jí),系 增,刪,查,改。 三 具體實(shí)踐 代碼下載 http://m.tkk7.com/Files/limq/StudentManger.rar 1 對(duì)象關(guān)系映射: 首先,將庫(kù)表映射為數(shù)據(jù)模型(SQL在源碼中查看),轉(zhuǎn)換后的數(shù)據(jù)模型如下圖: pic2.jpg 由此我們可以看出一下關(guān)聯(lián)關(guān)系: 1 Students 和 Contact(聯(lián)系方式)一對(duì)一關(guān)系。 2 Students 和 History(選課歷史) 一對(duì)多關(guān)系 3 Students 和 Classes 多對(duì)多關(guān)系。 4 Classes 和 Classes_info 一對(duì)多關(guān)系。 5 Classes 和 Teachers 多對(duì)一關(guān)系。 6 Classes 和 Courses 多對(duì)一關(guān)系。 7 Course 和 Department(系) 多對(duì)一關(guān)系。 8 Teachers 和 Department 多對(duì)一關(guān)系。 9 Students 和 Department 多對(duì)一關(guān)系。 在Hibernate中將以上關(guān)系一一映射,如Students 和 History 一對(duì)多關(guān)系 Students.cfg.xm.: 1 6 7 9 set> 10 同樣在History.cfg.xml中加入: 1 4 many-to-one> 5 用過(guò)MyEclipse開(kāi)發(fā)Hibernate的就知道,MyEclipse會(huì)幫助我們生成持久對(duì)象和抽象對(duì)象,我們要在 Students.java 中加入對(duì)History的引用 private Set history=new HashSet(); public Set getHistory() { return history; } public void setHistory(Set history) { this.history = history; } 同時(shí),在AbstractHistory.java 中刪除student_id 以及對(duì)應(yīng)的get,set 方法,History.java 中加入 private Students student; public Students getStudent() { return student; } public void setStudent(Students student) { this.student = student; } 具體內(nèi)容請(qǐng)查看 源代碼。 2 DAO 數(shù)據(jù)訪問(wèn)層 首先,編寫IBaseDao與BaseDao,其中IBaseDao代碼如下: 1 package limq.hibernate.dao; 2 3 import java.util.Collection; 4 import java.util.List; 5 import net.sf.hibernate.Criteria; 6 import net.sf.hibernate.Query; 7 import net.sf.hibernate.Session; 8 import limq.exception.DaoException; 9 10 public interface IBaseDao { 11 12 public Session openSession(); 13 14 public int getTotalCount( String hql) throws Exception; 15 16 public Query getQuery(String sql) throws Exception; 17 18 public Criteria getCriteria(Class clazz) throws Exception; 19 20 public int getTotalPage(int totalCount,int pageSize); 21 22 public void create(Object entity); 23 24 public void update(Object entity); 25 26 public void delete(Object entity) throws DaoException; 27 28 public void deleteAll(Class clazz) throws DaoException; 29 30 public void deleteAll(Collection entities) throws DaoException; 31 32 public Object loadByKey(Class clazz, String keyName, Object keyValue); 33 34 public List find(String queryString) throws DaoException; 35 36 public List find(String queryString, Object param) throws DaoException; 37 38 public List find(String queryString, Object[] params) throws DaoException; 39 40 } 41 BaseDao繼承org.springframework.orm.hibernate.support.HibernateDaoSupport 實(shí)現(xiàn)以上的 定義的方法 如: 1 public void create(Object entity) { 2 try { 3 getHibernateTemplate().save(entity); 4 5 } catch (Exception e) { 6 log.error("保存 " + entity.getClass().getName() + " 實(shí)例到數(shù)據(jù)庫(kù)失敗", e); 7 8 } 9 } 10 /** 11 * 獲得session 12 */ 13 public Session openSession() { 14 return SessionFactoryUtils.getSession(getSessionFactory(), false); 15 } 16 17 /** 18 * 獲得Query對(duì)象 19 */ 20 public Query getQuery(String sql) throws Exception{ 21 Session session = this.openSession(); 22 Query query = session.createQuery(sql); 23 return query; 24 } 25 /** 26 * 獲得Criteria對(duì)象 27 */ 28 public Criteria getCriteria(Class clazz) throws Exception{ 29 30 Session session=this.openSession(); 31 Criteria criteria = session.createCriteria(clazz); 32 return criteria; 33 } 34 可以看到,這里即充分利用了Spring對(duì)Hibernate的支持,還彌補(bǔ)了Spring的不足。最后分別為每個(gè)持久對(duì)象建立Interface,以及DAO,使其分別繼承IBaseDao與BaseDao。 如IDepartment,DepartmentDao 1 public interface IDepartment extends IBaseDao {} 2 3 public class DepartmentDao extends BaseDao implements IBaseDao {} 4 3 Service 層 在這里需要認(rèn)真思考每個(gè)業(yè)務(wù)邏輯所能用到的持久層對(duì)象和DAO,還要完成配置Spring框架, 首先我一起看看applications-service.xml 1 xml version="1.0" encoding="UTF-8"?> 2 DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" 3 "http://www.springframework.org/dtd/spring-beans.dtd"> 4 5 6 7 com.mysql.jdbc.Drivervalue> 8 property> 9 10 jdbc:mysql://localhost:3306/Studentvalue> 11 property> 12 13 rootvalue> 14 property> 15 16 value> 17 property> 18 bean> 19 20 21 22 property> 23 24 25 limq/hibernate/vo/Admins.hbm.xmlvalue> 26 limq/hibernate/vo/Classes.hbm.xmlvalue> 27 limq/hibernate/vo/Courses.hbm.xmlvalue> 28 limq/hibernate/vo/Students.hbm.xmlvalue> 29 limq/hibernate/vo/ClassesInfo.hbm.xmlvalue> 30 limq/hibernate/vo/Contact.hbm.xmlvalue> 31 limq/hibernate/vo/Department.hbm.xmlvalue> 32 limq/hibernate/vo/History.hbm.xmlvalue> 33 limq/hibernate/vo/Teachers.hbm.xmlvalue> 34 list> 35 property> 36 37 38 net.sf.hibernate.dialect.MySQLDialectprop> 39 trueprop> 40 props> 41 property> 42 bean> 43 44 45 46 property> 47 bean> 48 49 50 51 52 property> 53 bean> 54 55 56 57 property> 58 bean> 59 60 61 62 property> 63 bean> 64 65 66 67 property> 68 bean> 69 70 71 72 property> 73 bean> 74 75 76 77 property> 78 bean> 79 80 81 82 property> 83 bean> 84 85 86 limq.hibernate.dao.IStudentsvalue> 87 property> 88 89 90 hibernateInterceptorvalue> 91 studentDaoTargetvalue> 92 list> 93 property> 94 bean> 95 96 97 limq.hibernate.dao.ITeachersvalue> 98 property> 99 100 101 hibernateInterceptorvalue> 102 teacherDaoTargetvalue> 103 list> 104 property> 105 bean> 106 107 108 limq.hibernate.dao.ICoursesvalue> 109 property> 110 111 112 hibernateInterceptorvalue> 113 courseDaoTargetvalue> 114 list> 115 property> 116 bean> 117 118 119 limq.hibernate.dao.IClassesvalue> 120 property> 121 122 123 hibernateInterceptorvalue> 124 classDaoTargetvalue> 125 list> 126 property> 127 bean> 128 129 130 limq.hibernate.dao.IDepartmentvalue> 131 property> 132 133 134 hibernateInterceptorvalue> 135 departmentDaoTargetvalue> 136 list> 137 property> 138 bean> 139 140 141 limq.hibernate.dao.IAdminvalue> 142 property> 143 144 145 hibernateInterceptorvalue> 146 adminDaoTargetvalue> 147 list> 148 property> 149 bean> 150 151 152 153 154 property> 155 156 157 property> 158 159 160 property> 161 162 163 property> 164 bean> 165 166 167 168 property> 169 170 171 property> 172 173 174 property> 175 176 177 property> 178 bean> 179 180 181 182 property> 183 184 185 property> 186 187 188 property> 189 190 191 property> 192 193 194 property> 195 196 197 property> 198 bean> 199 200 201 202 203 property> 204 205 206 property> 207 208 209 PROPAGATION_SUPPORTSprop> 210 PROPAGATION_REQUIREDprop> 211 props> 212 property> 213 bean> 214 215 216 217 property> 218 219 220 property> 221 222 223 PROPAGATION_SUPPORTSprop> 224 PROPAGATION_REQUIREDprop> 225 props> 226 property> 227 bean> 228 229 230 231 property> 232 233 234 property> 235 236 237 PROPAGATION_SUPPORTSprop> 238 PROPAGATION_REQUIREDprop> 239 props> 240 property> 241 bean> 242 beans> 243 以StudentsServiceImpl以為例,下圖演示了如何利用Spring的Ioc與Hibernate的結(jié)合。 可以看到分別將studentDao,classDao,coursesDao,departmentDao,注入studentManager. pic4.JPG 1 IStudentsService.java 2 public interface IStudentsService { 3 4 public boolean validate(String username,String pasword); 5 public Classes[] getClassesFromCourse(Courses courses); 6 public Department getDepFromID(Integer id); 7 public Courses getCourseFromID(Integer id); 8 public Classes getClassFromID(Integer id); 9 public Students getStudetFromName(String name); 10 public boolean ifEnrolSameCourse(Classes clazz,Students stu); 11 public void selectClasses(Students stu, Classes clazz,Date date); 12 public boolean ifMoreThanCap(Classes clazz); 13 public void updateSudent(Students stu,Contact contact); 14 public HashMap getCourse(PageInfo pageinfo) throws Exception; 15 public HashMap getStudentHistory(PageInfo pageinfo,String stu_name) throws Exception; 16 17 } 18 19 實(shí)現(xiàn)StudentsServiceImpl.java 20 public class StudentsServiceImpl implements IStudentsService { 21 22 private Logger log = Logger.getLogger(this.getClass()); 23 24 private IStudents studentsDao; 25 26 private ICourses coursesDao; 27 28 private IClasses classesDao; 29 30 private IDepartment departmentsdao; 31 32 /** 33 * 驗(yàn)證用戶名密碼 34 * 35 * @param username 36 * 用戶名 37 * @param password 38 * 密碼 39 */ 40 41 public boolean validate(String username, String password) { 42 43 String password2 = studentsDao.getPasswordFromUsername(username); 44 if (password.equals(password2)) 45 return true; 46 else 47 return false; 48 49 } 50 51 /** 52 * 查找所有課程 53 * 54 */ 55 public Courses[] getAllCourses() { 56 57 List list = null; 58 try { 59 60 list = coursesDao.find("select c from Courses as c "); 61 } catch (Exception e) { 62 } 63 64 return (Courses[]) list.toArray(new Courses[0]); 65 } 66 67 /** 68 * 分頁(yè)顯示所有課程 69 * 70 * @param pageinfo 71 */ 72 public HashMap getCourse(PageInfo pageinfo) throws Exception { 73 74 HashMap hp = new HashMap(); 75 String hsql = "select c from Courses as c order by c.id"; 76 Query query = coursesDao.getQuery(hsql); 77 int totalCount = pageinfo.getTatalCount(); 78 int totalPage = pageinfo.getTotalpage(); 79 int start = pageinfo.getStart(); 80 totalCount = totalCount == -1 ? coursesDao.getTotalCount(hsql) 81 : totalCount; 82 totalPage = totalPage == -1 ? coursesDao.getTotalPage(totalCount, 83 pageinfo.getPageSize()) : totalPage; 84 query.setFirstResult(start); 85 query.setMaxResults(pageinfo.getPageSize()); 86 List list = query.list(); 87 hp.put("courses", (Courses[]) list.toArray(new Courses[0])); 88 hp.put("totalCount", new Integer(totalCount)); 89 hp.put("totalPage", new Integer(totalPage)); 90 return hp; 91 } 92 /** 93 * 分頁(yè)顯示所有選課歷史 94 * @param pageinfo 95 * @param stu_name 96 */ 97 public HashMap getStudentHistory(PageInfo pageinfo, String stu_name) 98 throws Exception { 99 HashMap hp = new HashMap(); 100 Students stu = this.getStudetFromName(stu_name); 101 Integer stu_id = stu.getId(); 102 Criteria criteria = coursesDao.getCriteria(History.class); 103 criteria.createCriteria("student").add(Expression.eq("name", stu_name)); 104 int totalCount = pageinfo.getTatalCount(); 105 int totalPage = pageinfo.getTotalpage(); 106 int start = pageinfo.getStart(); 107 totalCount = totalCount == -1 ? criteria.list().size() : totalCount; 108 totalPage = totalPage == -1 ? studentsDao.getTotalPage(totalCount, 109 pageinfo.getPageSize()) : totalPage; 110 criteria.setFirstResult(start); 111 criteria.setMaxResults(pageinfo.getPageSize()); 112 criteria.addOrder(Order.asc("id")); 113 List list = criteria.list(); 114 hp.put("history", (History[]) list.toArray(new History[0])); 115 hp.put("totalCount", new Integer(totalCount)); 116 hp.put("totalPage", new Integer(totalPage)); 117 return hp; 118 } 119 /** 120 * 根據(jù)課程查找班級(jí) 121 * @param course 122 * 課程實(shí)體 123 * @return 返回該課程下所有班級(jí) 124 */ 125 public Classes[] getClassesFromCourse(Courses course) { 126 return coursesDao.getClassesFromCourse(course); 127 } 128 129 /** 130 * 根據(jù)主鍵查找系 131 * @param id 132 * 主鍵ID 133 */ 134 public Department getDepFromID(Integer id) { 135 return (Department) departmentsdao 136 .loadByKey(Department.class, "id", id); 137 } 138 139 /** 140 * 根據(jù)主鍵查找課程 141 * @param id 142 * 主鍵ID 143 */ 144 public Courses getCourseFromID(Integer id) { 145 return (Courses) coursesDao.loadByKey(Courses.class, "id", id); 146 } 147 /** 148 * 根據(jù)主鍵查找班級(jí) 149 * @param id 150 * 主鍵ID 151 */ 152 public Classes getClassFromID(Integer id) { 153 return (Classes) classesDao.loadByKey(Classes.class, "id", id); 154 } 155 156 /** 157 * 根據(jù)姓名查找學(xué)生 158 * @param name 159 */ 160 public Students getStudetFromName(String name) { 161 return (Students) studentsDao.loadByKey(Students.class, "name", name); 162 } 163 164 /** 165 * 檢查學(xué)生是否選報(bào)了同一課程的班級(jí) 166 * @param clazz 167 * 所選報(bào)的班級(jí) 168 * @param stu 169 * 學(xué)生實(shí)體 170 * @return true 該生選報(bào)同一課程的班級(jí) 171 * @return false 沒(méi)有報(bào)過(guò)該課程的班級(jí),可以選報(bào) 172 * 173 */ 174 public boolean ifEnrolSameCourse(Classes clazz, Students stu) { 175 176 Courses cour = clazz.getCourse(); 177 178 Classes[] classes = (Classes[]) stu.getClasses() 179 .toArray(new Classes[0]); 180 for (int i = 0; i < classes.length; i++) { 181 182 Courses c1 = classes[i].getCourse(); 183 184 if (c1.getId().equals(cour.getId())) 185 return true; 186 } 187 return false; 188 } 189 190 /** 191 * 檢查課程的目前人數(shù) 192 * @param clazz 193 * 檢查班級(jí)人數(shù)是否已滿 194 * @param clazz 195 * 班級(jí)實(shí)體 196 * @return true 班級(jí)人數(shù)已滿 197 * @return false 班級(jí)人數(shù)未滿 198 * 199 */ 200 public boolean ifMoreThanCap(Classes clazz) { 201 Integer capacity = clazz.getCapacity(); 202 Integer maxcapacity = clazz.getMaxcapacity(); 203 if (capacity.intValue() < maxcapacity.intValue()) { 204 clazz.setCapacity(Integer.valueOf(capacity.intValue() + 1)); 205 //classesDao.update(clazz); 206 return false; 207 } else 208 return true; 209 210 } 211 212 /** 213 * 數(shù)據(jù)庫(kù)插入選擇班級(jí)的記錄 214 * @param stu 215 * 學(xué)生 216 * @param clazz 217 * 所選擇的班級(jí) 218 */ 219 public void selectClasses(Students stu, Classes clazz, Date date) 220 { 221 stu.getClasses().add(clazz); 222 clazz.getStudents().add(stu); 223 History his = new History(); 224 his.setEnrolTime(date); 225 his.setStudent(stu); 226 his.setClasses(clazz); 227 his.setScore(clazz.getCourse().getScore()); 228 his.setMarking(new Double(0)); 229 try{ 230 String cour_name=new String(clazz.getCourse().getName().getBytes("GBK")); 231 his.setCourseName(cour_name); 232 }catch( java.io.UnsupportedEncodingException e){e.getStackTrace();} 233 stu.getHistory().add(his); 234 } 235 236 public void updateSudent(Students stu,Contact contact){ 237 238 studentsDao.update(stu); 239 studentsDao.update(contact); 240 241 } 242 public IStudents getStudentsDao() { 243 return studentsDao; 244 } 245 public void setStudentsDao(IStudents studentsDao) { 246 this.studentsDao = studentsDao; 247 } 248 public IClasses getClassesDao() { 249 return classesDao; 250 } 251 public void setClassesDao(IClasses classesDao) { 252 this.classesDao = classesDao; 253 } 254 public ICourses getCoursesDao() { 255 return coursesDao; 256 } 257 public void setCoursesDao(ICourses coursesDao) { 258 this.coursesDao = coursesDao; 259 } 260 public IDepartment getDepartmentsdao() { 261 return departmentsdao; 262 } 263 public void setDepartmentsdao(IDepartment departmentdao) { 264 this.departmentsdao = departmentdao; 265 } 266 } 267 268 4 UI層 這里我們選擇Struts,首先配置 web.xml 1 xml version="1.0" encoding="UTF-8"?> 2 3 4 contextConfigLocationparam-name> 5 /WEB-INF/classes/applications-service.xmlparam-value> 6 context-param> 7 8 log4jConfigLocationparam-name> 9 /WEB-INF/log4j.propertiesparam-value> 10 context-param> 11 12 hibernateFilterfilter-name> 13 org.springframework.orm.hibernate.support.OpenSessionInViewFilterfilter-class> 14 filter> 15 16 hibernateFilterfilter-name> 17 /*url-pattern> 18 filter-mapping> 19 20 Set Character Encodingfilter-name> 21 limq.struts.SetCharacterEncodingFilterfilter-class> 22 filter> 23 24 Set Character Encodingfilter-name> 25 /*url-pattern> 26 filter-mapping> 27 28 SpringContextServletservlet-name> 29 org.springframework.web.context.ContextLoaderServletservlet-class> 30 1load-on-startup> 31 servlet> 32 33 actionservlet-name> 34 org.apache.struts.action.ActionServletservlet-class> 35 36 configparam-name> 37 /WEB-INF/struts-config.xmlparam-value> 38 init-param> 39 40 debugparam-name> 41 3param-value> 42 init-param> 43 44 detailparam-name> 45 3param-value> 46 init-param> 47 0load-on-startup> 48 servlet> 49 50 actionservlet-name> 51 *.dourl-pattern> 52 servlet-mapping> 53 web-app> 54 55 其中注意這幾句, 1 2 hibernateFilterfilter-name> 3 org.springframework.orm.hibernate.support.OpenSessionInViewFilterfilter-class> 4 filter> 5 6 hibernateFilterfilter-name> 7 /*url-pattern> 8 filter-mapping> 9 由于我們使用了lazy = "true",如果想在UI層使用實(shí)體對(duì)象關(guān)聯(lián)來(lái)獲得其他對(duì)象時(shí)就會(huì)有這樣的提示: org.hibernate.LazyInitializationException: failed to lazily initialize a collection Spring 中引入了 OpenSessionInView模式可以處理以上問(wèn)題,即在web.xml中加入以上代碼。 接下來(lái)建立抽象BaseAction,和 BaseDispatchAction,其中后者與前者相似目的為減少Action的數(shù)量 1 abstract class BaseAction extends Action { 2 3 private IStudentsService studentsService; 4 private ITeachersService teachersSerivce; 5 private IAdminService adminService; 6 public void setServlet(ActionServlet actionServlet) { 7 super.setServlet(actionServlet); 8 ServletContext servletContext = actionServlet.getServletContext(); 9 WebApplicationContext wac = WebApplicationContextUtils 10 .getRequiredWebApplicationContext(servletContext); 11 12 this.studentsService = (IStudentsService) wac.getBean("studentManager"); 13 this.adminService = (IAdminService) wac.getBean("adminManager"); 14 this.teachersSerivce = (ITeachersService) wac.getBean("teacherManager"); 15 } 16 public IStudentsService getStudentsService() { 17 return studentsService; 18 } 19 public ITeachersService getTeachersSerivce() { 20 return teachersSerivce; 21 } 22 public void setTeachersSerivce(ITeachersService teachersSerivce) { 23 this.teachersSerivce = teachersSerivce; 24 } 25 public IAdminService getAdminService() { 26 return adminService; 27 } 28 public void setAdminService(IAdminService adminService) { 29 this.adminService = adminService; 30 } 31 } 32 BaseDispatchAction與之類似,請(qǐng)查看源碼。其他Action都從這兩個(gè)類繼承。 以下就以查看課程下的班級(jí)為例演示Struts與Spring的使用: 1 CoursesAction.java 2 /** 3 * 查看課程下的班級(jí) 4 */ 5 public ActionForward viewClassFromCourse(ActionMapping mapping, 6 ActionForm form, HttpServletRequest request, 7 HttpServletResponse response) throws Exception { 8 Integer cour_id = Integer.valueOf((request.getParameter("cour_id"))); 9 Courses cour = super.getStudentsService().getCourseFromID(cour_id); 10 Classes[] clazz =(Classes[])cour.getClasses().toArray(new Classes[0]); 11 request.setAttribute("clazz", clazz); 12 return mapping.findForward("success"); 13 } 14 這里從上一個(gè)頁(yè)面獲得課程編號(hào) cour_id, 然后通過(guò)StudentsServiceImpl中的 1 public Courses getCourseFromID(Integer id) { 2 return (Courses) coursesDao.loadByKey(Courses.class, "id", id); 3 } 4 方法查到Courses實(shí)例,利用Courses和Classes的關(guān)聯(lián)關(guān)系得到Classes[],在將其放入 Request. 通過(guò)mapping.findForward("success"),轉(zhuǎn)發(fā)到 select_course_Content.jsp CustomRequestProcessor.java 介紹 1 public class CustomRequestProcessor extends RequestProcessor { 2 protected boolean processPreprocess(HttpServletRequest request, 3 HttpServletResponse response) { 4 boolean continueProcessing = true; 5 HttpSession session = request.getSession(); 6 String uri =request.getRequestURI(); 7 if ( session == null || session.getAttribute("userName") == null ) { 8 continueProcessing = false; 9 if(uri.endsWith("login.do")) return true; 10 try{ 11 response.sendRedirect("/StudentManger/login.jsp" ); 12 }catch( Exception ex ){ 13 log.error( "Problem sending redirect from processPreprocess()" ); 14 } 15 } 16 return continueProcessing; 17 } 18 } 19 為了驗(yàn)證用戶操作權(quán)限,這里擴(kuò)展了Struts 的RequestProcessor來(lái)判斷Session如果Session和userName都不空則程序繼續(xù),否則重定向到login.jsp。要想擴(kuò)展RequestProcessor類,需在Struts的配置文件中加入 1 6 呵呵,當(dāng)然在正規(guī)使用時(shí)僅僅這樣驗(yàn)證是不夠的。歡迎你把自己修改方法告訴我。 4分頁(yè)處理: 下面重點(diǎn)討論一下Hibernate的分頁(yè)處理方式。 Hibernate 中處理查詢主要有 Query ,Criteria,分別以 HSQL或編程方式實(shí)現(xiàn), 本例對(duì)這兩種方法都有相關(guān)處理。由于在Spring中無(wú)法直接使用Query和Criteria對(duì)象 所以只有先從Spring那里借一個(gè)Session,等使用完了在還給Sping處理。讀者應(yīng)該還記得在BaseDao中有這樣的語(yǔ)句方便我們獲取Session及其他對(duì)象: 1 public Query getQuery(String sql) throws Exception{ 2 Session session = this.openSession(); 3 Query query = session.createQuery(sql); 4 return query; 5 } 6 7 public Criteria getCriteria(Class clazz) throws Exception{ 8 9 Session session=this.openSession(); 10 Criteria criteria = session.createCriteria(clazz); 11 return criteria; 12 } 13 Service層以查詢所有課程與學(xué)生選課記錄為例處理Query與Criteria: 1 StudentsServiceImpl.java 2 public HashMap getCourse(PageInfo pageinfo) throws Exception { 3 4 HashMap hp = new HashMap(); 5 String hsql = "select c from Courses as c order by c.id"; 6 Query query = coursesDao.getQuery(hsql); 7 int totalCount = pageinfo.getTatalCount(); 8 int totalPage = pageinfo.getTotalpage(); 9 int start = pageinfo.getStart(); 10 totalCount = totalCount == -1 ? coursesDao.getTotalCount(hsql) 11 : totalCount; 12 totalPage = totalPage == -1 ? coursesDao.getTotalPage(totalCount, 13 pageinfo.getPageSize()) : totalPage; 14 query.setFirstResult(start); 15 query.setMaxResults(pageinfo.getPageSize()); 16 List list = query.list(); 17 hp.put("courses", (Courses[]) list.toArray(new Courses[0])); 18 hp.put("totalCount", new Integer(totalCount)); 19 hp.put("totalPage", new Integer(totalPage)); 20 return hp; 21 } 22 23 public HashMap getStudentHistory(PageInfo pageinfo, String stu_name) 24 throws Exception { 25 HashMap hp = new HashMap(); 26 Students stu = this.getStudetFromName(stu_name); 27 Integer stu_id = stu.getId(); 28 Criteria criteria = coursesDao.getCriteria(History.class); 29 criteria.createCriteria("student").add(Expression.eq("name", stu_name)); 30 int totalCount = pageinfo.getTatalCount(); 31 int totalPage = pageinfo.getTotalpage(); 32 int start = pageinfo.getStart(); 33 totalCount = totalCount == -1 ? criteria.list().size() : totalCount; 34 totalPage = totalPage == -1 ? studentsDao.getTotalPage(totalCount, 35 pageinfo.getPageSize()) : totalPage; 36 criteria.setFirstResult(start); 37 criteria.setMaxResults(pageinfo.getPageSize()); 38 criteria.addOrder(Order.asc("id")); 39 List list = criteria.list(); 40 hp.put("history", (History[]) list.toArray(new History[0])); 41 hp.put("totalCount", new Integer(totalCount)); 42 hp.put("totalPage", new Integer(totalPage)); 43 return hp; 44 } 45 PageIngfo.java 46 public class PageInfo { 47 48 int pageNo=0; 49 int totalpage=-1; 50 int tatalCount=-1; 51 int pageSize=0; 52 int start=0; 53 54 可以看到getCourse和getStudentHistory有很多相似之處,Hibernate為Query和Criteria提供了針對(duì)不同數(shù)據(jù)庫(kù)的解決分頁(yè)方法, Quey需要我們寫HSQL, Criteria不但可以應(yīng)付帶有條件的查詢,還不用我們自己寫HSQL,PageInfo是含有分頁(yè)信息的普通java類。 再看看Struts是如何調(diào)用getStudentHistory 的, 1 PageAction.java 2 public class PageAction extends BaseDispatchAction{ 3 public ActionForward execute(ActionMapping mapping, 4 ActionForm form, 5 HttpServletRequest request, 6 HttpServletResponse response) 7 throws Exception { 8 String pageNo=request.getParameter("pageNo"); 9 String totalcount=request.getParameter("totalcount"); 10 String totalpage=request.getParameter("totalpage"); 11 int pagesize=2;//每頁(yè)的大小 12 PageInfo page =new PageInfo(); 13 page.setPageSize(pagesize); 14 HashMap hp=null; 15 History[] historys = null; 16 String stu_name=null; 17 HttpSession session = request.getSession(); 18 stu_name = (String) session.getAttribute("userName"); 19 if(pageNo == null || totalcount == null || totalpage==null){ 20 //第一次發(fā)送請(qǐng)求 21 page.setPageNo(1); 22 hp=super.getStudentsService().getStudentHistory(page,stu_name); 23 page.setTatalCount(((Integer)hp.get("totalCount")).intValue()); 24 page.setTotalpage(((Integer)hp.get("totalPage")).intValue()); 25 }else{ 26 page.setPageNo(Integer.parseInt(pageNo)); 27 page.setTatalCount(Integer.parseInt(totalcount)); 28 page.setTotalpage(Integer.parseInt(totalpage)); 29 hp=super.getStudentsService().getStudentHistory(page,stu_name); 30 31 } 32 historys =(History[]) hp.get("history"); 33 request.setAttribute("history",historys); 34 request.setAttribute("pageinfo",page); 35 return mapping.findForward("success"); 36 } 37 } 38 在stu_his_Content.jsp中避免代碼重復(fù)使用了自定義標(biāo)志來(lái)處理分頁(yè) 1 <%@ page contentType="text/html;charset=UTF-8" language="java" %> 2 <%@ taglib uri="/WEB-INF/struts-bean.tld" prefix="bean" %> 3 <%@ taglib uri="/WEB-INF/struts-html.tld" prefix="html" %> 4 <%@ taglib uri="/WEB-INF/struts-logic.tld" prefix="logic" %> 5 <%@ page import="limq.hibernate.vo.*"%> 6 <%@ taglib uri="/WEB-INF/MyTag.tld" prefix="mytag"%> 7 <%@ page import="limq.common.*"%> 8 9 10 <% 11 PageInfo pageinfo =(PageInfo) request.getAttribute("pageinfo"); 12 History[] historys = (History[])request.getAttribute("history"); 13 %> 14 15 16 27
          td> 17 td> 18 td> 19 td> 20 td> 21 tr> 22 <% 23 for(int i=0;i 26
          <%=his.getClasses().getId()%>td> 28 <%=his.getCourseName()%>td> 29 <%=his.getEnrolTime()%>td> 30 <%=his.getScore()%>td> 31 <%=his.getMarking()%>td> 32 tr> 33 <% 34 } 35 %> 36 table> 37 38 body> 39 html:html> 40 標(biāo)志處理類如下: 1 PageTag.java 2 3 public class PageTag extends SimpleTagSupport { 4 5 private PageInfo pageinfo = null; 6 private String action = null; 7 8 public String getAction() { 9 return action;} 10 public void setAction(String action) { 11 this.action = action; 12 } 13 public PageInfo getPageinfo() { 14 return pageinfo; 15 } 16 public void setPageinfo(PageInfo pageinfo) { 17 this.pageinfo = pageinfo; 18 } 19 20 public void doTag() throws JspException, IOException { 21 JspWriter out = getJspContext().getOut(); 22 23 int totalpage = pageinfo.getTotalpage(); 24 int totalcount = pageinfo.getTatalCount(); 25 int pageNo = pageinfo.getPageNo(); 26 int addPageNo = pageNo + 1; 27 int minPageNo = pageNo - 1; 28 29 out.println("< SPAN>"400\" align=\"center\" cellPadding=\"0\" cellSpacing=\"0\"> "); 30 out.print(""); 31 out.println("共" + totalcount + "條," + totalpage + "頁(yè),當(dāng)前" 32 + pageNo + "頁(yè) "); 66 67 } 68 69 } 70 5 中文亂碼問(wèn)題: 1 數(shù)據(jù)庫(kù):MYSQL 4.1 (或以上版本)4.1直接支持Unicode,以下版本支持的不好。 2 驅(qū)動(dòng): MySQL JDBC Driver的3.0.16(或以上版本) 3 在數(shù)據(jù)庫(kù)中做如下設(shè)定 pic5.JPG 4 在建立表時(shí)同樣加上ENGINE=MyISAM DEFAULT CHARSET=gbk 1CREATE TABLE `students` ( 2 `id` int(20) NOT NULL default '0', 3 `name` varchar(20) NOT NULL default '', 4 `department_id` int(11) default NULL, 5 `password` varchar(20) default NULL, 6 `score` double(15,3) default NULL, 7 PRIMARY KEY (`id`) 8) ENGINE=MyISAM DEFAULT CHARSET=gbk 9 5 配置hibernate.cfg.xml 1 jdbc:mysql://localhost:3306/Studentproperty> 2 net.sf.hibernate.dialect.MySQLDialectproperty> 3 property> 4 com.mysql.jdbc.Driverproperty> 5 robbin: MySQL JDBC Driver的3.0.16也是一個(gè)分水嶺,3.0.16版本會(huì)取數(shù)據(jù)庫(kù)本身的編碼,然后按照該編碼轉(zhuǎn)換,這種方式和Oracle的JDBC Driver是一樣的。例如你的數(shù)據(jù)庫(kù)是GBK編碼的話,JDBC Driver就會(huì)把數(shù)據(jù)庫(kù)里面的取出來(lái)的字符串按照GBK往unicode轉(zhuǎn)換,送給JVM。因此正確的設(shè)置數(shù)據(jù)庫(kù)本身的編碼就尤為重要。 MySQL JDBC Driver3.0.16以下的版本則不然,它不會(huì)那么智能的根據(jù)數(shù)據(jù)庫(kù)編碼來(lái)確定如何轉(zhuǎn)換,它總是默認(rèn)使用ISO8859-1,因此你必須使用 characterEncoding=GBK來(lái)強(qiáng)制他把數(shù)據(jù)庫(kù)中取出來(lái)的字符串按照GBK來(lái)往unicode轉(zhuǎn)換。 因此,使用什么數(shù)據(jù)庫(kù)版本,不管是3.x,還是4.0.x還是4.1.x,其實(shí)對(duì)我們來(lái)說(shuō)不重要,重要的有二: 1) 正確的設(shè)定數(shù)據(jù)庫(kù)編碼,MySQL4.0以下版本的字符集總是默認(rèn)ISO8859-1,MySQL4.1在安裝的時(shí)候會(huì)讓你選擇。如果你準(zhǔn)備使用UTF- 8,那么在創(chuàng)建數(shù)據(jù)庫(kù)的時(shí)候就要指定好UTF-8(創(chuàng)建好以后也可以改,4.1以上版本還可以單獨(dú)指定表的字符集) 2) 使用3.0.16以上版本的JDBC Driver,那么你就不需要再寫什么characterEncoding=UTF-8 6 開(kāi)發(fā)工具介紹 MyEclipse 3.8 首先添加用戶庫(kù),如下圖將Struts,Spring,Hibernate 的庫(kù)添加到用戶庫(kù)中 pic6.JPG 如果出現(xiàn)環(huán)境問(wèn)題可能你的Struts包有問(wèn)題,請(qǐng)到http://struts.apache.org/download.cgi下載struts-1.2.7-lib.zip 。 具體使用參考http://www.laliluna.de/struts-hibernate-integration-tutorial-en.html 總結(jié) 本文至此已將Struts+Sprng+Hibernate的大致思路以及本人所遇到的難點(diǎn),重點(diǎn)介紹完了。 其中管理員我只完成了對(duì)學(xué)生的部分,其他功能大同小異,有興趣的讀者不妨動(dòng)手試試。最后建議初學(xué)者不要直接使用Spring對(duì)Hibernate的封裝,而是從Hibernate學(xué)起,先要學(xué)會(huì)自己管理Session,Transaction,然后在用Spring,這樣理解會(huì)更深刻。同時(shí)如果你有好的建議,或問(wèn)題請(qǐng)聯(lián)系我 QQ 39315890 Email: mill_lmq@yahoo.com.cn 參考: Spring Framework之最佳實(shí)踐: http://www.gpowersoft.com/tech/Spring/ Hibernate+Spring 對(duì)DAO的處理實(shí)列 http://www.javaeye.com/viewtopic.php?t=7923&start=0&postdays=0&postorder=asc&highlight= Hibernate實(shí)現(xiàn)分頁(yè)查詢的原理分析 http://forum.javaeye.com/viewtopic.php?t=261&start=0&postdays=0&postorder=asc&highlight= Spring的DAO設(shè)計(jì)實(shí)踐 http://www.hibernate.org.cn/viewtopic.php?t=8224&start=0&postdays=0&postorder=asc&highlight= 用 OpenSessionInViewInterceptor 的思路解決Hibernate Lazy問(wèn)題 http://www.javaeye.com/viewtopic.php?t=14631&start=0&postdays=0&postorder=asc&highlight= 上一篇 下一篇 (kukoo的分類目錄[Java]中共10篇) 相關(guān)文章 ? Eclipse快速上手Hibernate--8.關(guān)聯(lián)映射(多對(duì)多) 05.11.11 from joes ? Eclipse快速上手Hibernate--7.關(guān)聯(lián)映射(一對(duì)多)(2) 05.11.11 from joes ? Eclipse快速上手Hibernate--6.關(guān)聯(lián)映射(一對(duì)一)(1) 05.11.11 from joes ? Eclipse快速上手Hibernate--5.組件映射 05.11.11 from joes ? Eclipse快速上手Hibernate--4.繼承映射(3) 05.11.11 from joes

          posted on 2005-11-13 08:14 閱讀(1111) 評(píng)論(1)  編輯  收藏 所屬分類: J2ee

          評(píng)論

          # re: 實(shí)例介紹Struts+Spring+Hibernate開(kāi)發(fā)  回復(fù)  更多評(píng)論   

          頁(yè)面都亂套了,沒(méi)法看啊
          2008-02-13 15:38 | 到此一游
          主站蜘蛛池模板: 国产亚洲精品美女久久久 | 免费国产黄网站在线观看视频| 亚洲高清无码综合性爱视频| 无码一区二区三区亚洲人妻| 黑人粗长大战亚洲女2021国产精品成人免费视频 | 亚洲人成网站影音先锋播放| 中文字幕视频免费| 亚洲精品动漫在线| 女人18毛片特级一级免费视频 | 国产三级免费电影| 青柠影视在线观看免费| 亚洲黑人嫩小videos| 免费观看毛片视频| 免费成人福利视频| 亚洲av无码片vr一区二区三区| 国产成人综合亚洲AV第一页| 99精品视频在线观看免费专区| 中文字幕的电影免费网站| 亚洲人成在线播放网站岛国| 自拍偷自拍亚洲精品被多人伦好爽| 免费在线观看一级毛片| 无码精品一区二区三区免费视频| 美女视频黄免费亚洲| 亚洲AV无码乱码精品国产| 永久中文字幕免费视频网站| 日韩在线视频免费看| 日韩电影免费在线观看中文字幕| 亚洲欧美熟妇综合久久久久| 亚洲精品二区国产综合野狼| 在线观看亚洲天天一三视| 国产专区一va亚洲v天堂| 91精品视频免费| 久久久久免费视频| 亚洲午夜精品一区二区麻豆| 亚洲毛片在线观看| 亚洲精品视频在线观看视频| 亚洲一级毛片在线播放| 曰韩亚洲av人人夜夜澡人人爽 | 两性色午夜视频免费网| 两个人日本WWW免费版| 国内少妇偷人精品视频免费|