SpringMvc(4.0)+Hibernate(4.0)
1.專案結構截圖:
整個專案的專案結構就如圖上所示,文件後面的程式碼demo裡面的類就可以對應放入路徑即可,專案的成敗還有一個很重要的因素就是包的問題,除錯包是個很痛苦的過程,這裡我也將我這個專案lib裡面的包上傳到百度雲裡面供大家下載,希望大家能成功把自己的Demo專案搭建起來。
百度雲分享下載連結:
2.web.xml配置
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5"> <display-name>SpringMvcDemo</display-name> <welcome-file-list> <welcome-file>index.jsp</welcome-file> </welcome-file-list> <!-- 通過過濾器配置utf-8編碼 --> <filter> <filter-name>Spring character encoding filter</filter-name> <filter-class> org.springframework.web.filter.CharacterEncodingFilter </filter-class> <init-param> <param-name>encoding</param-name> <param-value>utf-8</param-value> </init-param> </filter> <filter-mapping> <filter-name>Spring character encoding filter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <!-- Spring 服務層的配置檔案 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:config/applicationContext.xml</param-value> </context-param> <!-- Spring 容器啟動監聽器 --> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!--Spring MVC 的Servlet它將載入WEB-INF/spring-servlet.xml 的 配置檔案,以啟動Spring MVC模組--> <servlet> <servlet-name>spring</servlet-name> <servlet-class> org.springframework.web.servlet.DispatcherServlet </servlet-class> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>spring</servlet-name> <url-pattern>*.do</url-pattern> </servlet-mapping> </web-app>
3.applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:p="http://www.springframework.org/schema/p" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"> <context:annotation-config /> <context:component-scan base-package="com.ceno" /> <!-- 自動掃描所有註解該路徑 --> <mvc:annotation-driven></mvc:annotation-driven> <aop:aspectj-autoproxy/> <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean"> <property name="dataSource" ref="dataSource" /> <property name="hibernateProperties"> <props> <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop> <prop key="hibernate.hbm2ddl.auto">update</prop> <prop key="show_sql">true</prop> <prop key="format_sql">true</prop> </props> </property> <property name="packagesToScan"> <list> <value>com.ceno.form</value><!-- 掃描實體類--> </list> </property> </bean> <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory" /> </bean> <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver" /> <property name="url" value="jdbc:mysql://localhost:3306/springmvcdemo" /> <property name="username" value="root" /> <property name="password" value="123" /> </bean> <tx:annotation-driven transaction-manager="transactionManager" /> </beans>
4.spring-servlet.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd"> <!-- 掃描控制器包--> <context:component-scan base-package="com.ceno.controller" /> <bean class="org.springframework.web.servlet.mvc.annotation. AnnotationMethodHandlerAdapter"/> <bean id="viewResolver" class="org.springframework.web.servlet.view.UrlBasedViewResolver"> <property name="viewClass" value="org.springframework.web.servlet.view.JstlView" /> <property name="prefix" value="/WEB-INF/jsp/" /> <property name="suffix" value=".jsp" /> </bean> </beans>
- @Component 是一個泛化的概念,僅僅表示一個元件 (Bean) ,可以作用在任何層次。
- @Service 通常作用在業務層,但是目前該功能與 @Component 相同。
- @Constroller 通常作用在控制層,但是目前該功能與 @Component 相同。
- @Repository它用於將資料訪問層 (DAO 層 ) 的類標識為 Spring Bean
預設情況下,對於包含 name 屬性的 @Component、@Repository、 @Service 和 @Controller,會把 name 取值作為 Bean 的名字。如果這個註解不包含 name 值或是其他被自定義過濾器發現的元件,預設 Bean 名稱會是小寫開頭的非限定類名。
@Qualifier
public class Foo { @Autowired @Qualifier(value="bar1") private Bar bar; } |
@Resource和@Autowired
作用相當於@Autowired,只不過@Autowired按byType自動注入,而@Resource預設按 byName自 動注入罷了。@Resource有兩個屬性是比較重要的,分是name和type,Spring將@Resource註解的name屬性解析為bean的 名字,而type屬性則解析為bean的型別。所以如果使用name屬性,則使用byName的自動注入策略,而使用type屬性時則使用byType自 動注入策略。如果既不指定name也不指定type屬性,這時將通過反射機制使用byName自動注入策略。
@Resource裝配順序
1. 如果同時指定了name和type,則從Spring上下文中找到唯一匹配的bean進行裝配,找不到則丟擲異常
2. 如果指定了name,則從上下文中查詢名稱(id)匹配的bean進行裝配,找不到則丟擲異常
3. 如果指定了type,則從上下文中找到型別匹配的唯一bean進行裝配,找不到或者找到多個,都會丟擲異常
4. 如果既沒有指定name,又沒有指定type,則自動按照byName方式進行裝配;如果沒有匹配,則回退為一個原始型別進行匹配,如果匹配則自動裝配;
與通過 XML 配置的 Spring Bean 一樣,通過上述註解標識的 Bean,其預設作用域是"singleton",為了配合這四個註解,在標註 Bean 的同時能夠指定 Bean 的作用域,Spring 2.5 引入了 @Scope 註解。使用該註解時只需提供作用域的名稱就行了,如下所示:
@Scope("prototype") @Repository public class Demo { … } |
6.sessionFactory的CRUD
Session session = sessionFactory.getCurrentSession(); // 獲得Session
Session session = sessionFactory.openSession(); // 開啟Session
對於上述的兩個方法,有以下區別:
1. openSession 從字面上可以看得出來,是開啟一個新的session物件,而且每次使用都是開啟一個新的session,假如連續使用多次,則獲得的session不是同一個物件,並且使用完需要呼叫close方法關閉session。
2. getCurrentSession ,從字面上可以看得出來,是獲取當前上下文一個session物件,當第一次使用此方法時,會自動產生一個session物件,並且連續使用多次時,得到的session都是同一個物件,這就是與openSession的區別之一,簡單而言,getCurrentSession 就是:如果有已經使用的,用舊的,如果沒有,建新的。
在實際開發中,往往使用getCurrentSession多,因為一般是處理同一個事務(即是使用一個數據庫的情況),所以在一般情況下比較少使用openSession或者說openSession是比較老舊的一套介面了;
Create
public void save(Student s){
sessionFactory.getCurrentSession().save(s);
}
Read
讀取列表:
public List<Student> findAll(){
return sessionFactory.getCurrentSession().createQuery("from Student").list();
}
讀取列表(分頁讀取):
public List<Student> listStudentByPage(int page,int size){
Query query=sessionFactory.getCurrentSession().createQuery("from Student");
query.setFirstResult((page-1)*size);
query.setMaxResults(size);
List<Student> list=query.list();
return list;
}
讀取列表(根據Criteria):
public List<Student> queryByCriteria(String name,String passWord){
Criteria criteria=sessionFactory.getCurrentSession().createCriteria(Student.class);
criteria.add(Restrictions.eq("name",name));
//criteria.setFirstResult();
//criteria.setMaxResults();criteria同樣可以實現分頁
criteria.add(Restrictions.eq("passWord",passWord));
List<Student> list=criteria.list();
return list;
}
讀取單個結果(根據條件):
public Student getStudent(String name,String passWord){
Session session=sessionFactory.getCurrentSession();
Query query=session.createQuery("from Student where name=? and passWord=:passWord");
query.setParameter(0,name);
query.setParameter("passWord",passWord);
//query.executeUpdate();用來執行一些更新操作的sql語句
//query.list();返回的就是一個集合
Student student=(Student)query.uniqueResult();
return student;
}
讀取單個結果(根據id):
public Student getStudent(int id){
Student student=(Student)sessionFactory.getCurrentSession().get(Student.class,id);
return student;
}
Update
public void update(Student model) {
sessionFactory.getCurrentSession().update(model);
}
Delete
public void delete(Student model) {
sessionFactory.getCurrentSession().delete(model);
}
NativeSql
也可以使用資料庫的 Native SQL 語言來查詢資料。這對你在要使用資料庫的某些特性的時候(比如說在查詢提示或者 Oracle 中的 CONNECT 關鍵字),這是非常有用的。這就能夠掃清你把原來直接使用 SQL/JDBC 的程式遷移到基於 Hibernate 應用的道路上的障礙。
public List<Student> getByNativeSql(){
Query query=sessionFactory.getCurrentSession().createSQLQuery("select from student");
List<Student> list=query.list();
return list;
}
7.程式碼Demo:
Student實體類:
package com.ceno.form;
import java.io.Serializable;
import javax.persistence.Basic;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
@Entity
@Table(name="student")
public class Student implements Serializable{
private static final long serialVersionUID = 1L;
private Integer id;
private String name;
private String passWord;
@Id
@Basic(optional=false)
@Column(name="id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
@Column(name="name")
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Column(name="password")
public String getPassWord() {
return passWord;
}
public void setPassWord(String psw) {
this.passWord = psw;
}
}
Student的dao操作類:
package com.ceno.dao;
import java.util.List;
import org.hibernate.Criteria;
import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.criterion.Criterion;
import org.hibernate.criterion.Restrictions;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Repository;
import com.ceno.form.Student;
@Repository("entityDao")
public class EntityDao {
@Autowired
private SessionFactory sessionFactory;
public void save(Student s){
sessionFactory.getCurrentSession().save(s);
}
public Student getStudent(String name,String passWord){
Session session=sessionFactory.getCurrentSession();
Query query=session.createQuery("from Student where name=? and passWord=:passWord");
query.setParameter(0,name);
query.setParameter("passWord",passWord);
Student student=(Student)query.uniqueResult();
return student;
}
public Student getStudent(int id){
Student student=(Student)sessionFactory.getCurrentSession().get(Student.class,id);
return student;
}
public List<Student> getByNativeSql(){
Query query=sessionFactory.getCurrentSession().createSQLQuery("select from student");
List<Student> list=query.list();
return list;
}
public List<Student> listStudentByPage(int page,int size){
Query query=sessionFactory.getCurrentSession().createQuery("from Student");
query.setFirstResult((page-1)*size);
query.setMaxResults(size);
List<Student> list=query.list();
return list;
}
public List<Student> queryByCriteria(String name,String passWord){
Criteria criteria=sessionFactory.getCurrentSession().createCriteria(Student.class);
criteria.add(Restrictions.eq("name",name));
//criteria.setFirstResult();
//criteria.setMaxResults();
criteria.add(Restrictions.eq("passWord",passWord));
List<Student> list=criteria.list();
return list;
}
@SuppressWarnings("unchecked")
public List<Student> findAll(){
return sessionFactory.getCurrentSession().createQuery("from Student").list();
}
public void update(Student model) {
sessionFactory.getCurrentSession().update(model);
}
public void delete(Student model) {
sessionFactory.getCurrentSession().delete(model);
}
}
Student的service類:
package com.ceno.service;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import com.ceno.dao.EntityDao;
import com.ceno.form.Student;
@Service("studentService")
@Transactional
public class StudentService {
@Autowired
public EntityDao entityDao;
public void save(Student st){
entityDao.save(st);
}
public void delete(Student obj){
entityDao.delete(obj);
}
public List<Student> findAll() {
return entityDao.findAll();
}
public void update(Student model){
entityDao.update(model);
}
}
Student的controller類:
package com.ceno.controller;
/**
* Created by IntelliJ IDEA.
* User: Administrator
* Date: 14-4-27
* Time: 下午7:09
* To change this template use File | Settings | File Templates.
*/
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Controller;
import org.springframework.ui.ModelMap;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import com.ceno.form.Student;
import com.ceno.service.StudentService;
@Controller
@RequestMapping("/student.do")
public class StudentController{
protected final transient Log log
= LogFactory.getLog(StudentController.class);
@Autowired()
public StudentService studentService;
public StudentController(){}
@RequestMapping
public String load(ModelMap modelMap){
List<Student> list = studentService.findAll();
modelMap.put("list", list);
return "student";
}
@RequestMapping(params = "method=add")
public String add(HttpServletRequest request, ModelMap modelMap) throws Exception{
return "student_add";
}
@RequestMapping(params = "method=save")
public String save(HttpServletRequest request, ModelMap modelMap){
Student st = new Student();
String name = request.getParameter("name");
String passWord = request.getParameter("passWord");
st.setName(name);
st.setPassWord(passWord);
try{
studentService.save(st);
modelMap.put("addstate", "新增成功");
}
catch(Exception e){
e.printStackTrace();
log.error(e.getMessage());
modelMap.put("addstate", "新增失敗b");
}
return load(modelMap);
}
@RequestMapping(params = "method=del")
public void del(@RequestParam("id") String id, HttpServletResponse response){
try{
Student st = new Student();
st.setId(Integer.valueOf(id));
studentService.delete(st);
response.getWriter().print("{\"del\":\"true\"}");
}
catch(Exception e){
log.error(e.getMessage());
e.printStackTrace();
}
}
}
前臺程式碼:
新增Student頁面:
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>新增</title>
<script type="text/javascript"></script>
</head>
<body>
<form method="post" action="student.do?method=save">
<table>
<tr><td>姓名</td><td><input id="name" name="name" type="text" /></td></tr>
<tr><td>密碼</td><td><input id="passWord" name="passWord" type="text" /></td></tr>
<tr><td colSpan="2" align="center"><input type="submit" value="提交"/>
<input type="button" onclick="turnback()" value="返回" /> </td></tr>
</table>
</form>
</body>
</html>
Web.xml的頭配置資訊
<web-app xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.5">
注意裡面的web-app_2_5.xsd,就是這個引起的,在web-app_2_4.xsd中就不會出現這種問題(這個版本的isELIgnored預設設定為false)。
在不改變web.xml2.5版本的情況下解決辦法是:在jsp頁面頭加:<%@page isELIgnored="false"%> 問題得以解決。
還有就是:<%@page isELIgnored="false"%>的優先順序要高於web.xml中的設定,所以在JSP中的設定會蓋掉web.xml中的設定.
攔截器
Xml檔案配置
<!-- 攔截器 -->
<mvc:interceptors>
<!-- 使用bean定義一個Interceptor,直接定義在mvc:interceptors根下面的Interceptor將攔截所有的請求 -->
<bean class="com.ceno.interceptor.AllInterceptor"/>
<mvc:interceptor>
<!--
設定攔截的路徑
<mvc:mapping path="/entryOrJsonController/*" />如果不配置或/*,將攔截所有的Controller
-->
<mvc:mapping path="/student.do?method=save"/>
<bean class="com.ceno.interceptor.TimeInterceptor">
<!-- property為攔截器注入屬性 -->
<!--openingTime 屬性指定上班時間-->
<property name="openingTime">
<value>12</value>
</property>
<!--closingTime屬性指定下班時間-->
<property name="closingTime">
<value>18</value>
</property>
<!--outsideOfficeHoursPage屬性指定提示頁面的URL-->
<property name="outsideOfficeHoursPage">
<value>jsp/return.jsp</value>
</property>
</bean>
</mvc:interceptor>
</mvc:interceptors>