一.建立maven項目
二. 完成導包
三.配置resources中的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:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
">
<!-- 對靜态資源進行放行 -->
<mvc:default-servlet-handler />
<!-- 掃描controller部分的包 -->
<!-- @Component元件, @Repository持久層, @Service業務邏輯層, and @Controller控制器 -->
<context:component-scan base-package="com.lirui.web" />
<!-- 添加mvc對@RequestMapping等注解的支援 -->
<mvc:annotation-driven />
<!-- ViewResolver 視圖解析器 (struts2視圖類型類似) -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!-- 設定視圖路徑的前字尾,該配置可以讓我們寫視圖路徑的時候更簡單。 -->
<!-- 希望跳轉jsp是[/WEB-INF/views/字首][xxx變量][.jsp字尾] -->
<!-- * @see #setPrefix -->
<property name="prefix" value="/WEB-INF/views/" />
<!-- * @see #setSuffix -->
<property name="suffix" value=".jsp" />
</bean>
<!-- 錯誤:提示告訴開發者你沒有配置檔案上傳解析器。 -->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!-- 設定上傳檔案的最大尺寸為1MB -->
<property name="maxUploadSize">
<value>1048576</value>
</property>
</bean>
</beans>
三.配置web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
id="WebApp_ID" version="3.1">
<!-- 配置過濾器: 我想讓EntityManager對象在我的jsp頁面顯示完畢後再關閉-->
<filter>
<filter-name>openEM</filter-name>
<filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>openEM</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- 配置解決中文亂碼的問題 -->
<filter>
<filter-name>characterEncodingFilter</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>
<init-param>
<param-name>forceEncoding</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>characterEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!--
配置控制器
Servlet是通路到的時候才建立,我們希望它随着tomcat(伺服器)啟動就建立
啟動時會去找Springmvc的核心配置檔案 -> 預設:/WEB-INF/dispatcherServlet-servlet.xml
-->
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext-mvc.xml</param-value>
</init-param>
<!--load-on-startup:随着伺服器啟動而加載-->
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<!--
符合RESTful風格的軟體 比如../age/18
SpringMVC的RESTful風格,會導緻通路不到靜态資源
-->
<url-pattern>/</url-pattern>
</servlet-mapping>
<!--
啟動Spring
Spring這個監聽器會去找一個預設的配置檔案: [/WEB-INF/applicationContext.xml]
-->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!--配置Spring的核心檔案的位置-->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
</web-app>
四.測試
建立一個domain實體類
@Entity
@Table(name="employee")
public class Employee extends BaseDomain {
private String username;
private String password;
private String email;
private Integer age;
...省略 getset
}
對應資料庫的實體表
建立持久化接口繼承JpaRepository,并且聲明泛型和id類型
隻要繼承了,就可以完成普通crud,jpa自動實作,自動繼承
五.分層
- domain實體類層
- repository持久化層
- service服務層
-
web控制層
層次分明,web調用service,service調用repository底層
六.tomcat的配置
tomcat無法打開首頁
報錯classNotfound。。。。
解決方案:File > Project Structure > Artifacts > 在右側Output Layout右擊項目名,選擇Put into Output Root
如果成功項目就搭建完成
七.懶加載問題
運作時我們會發現傳回了資料資料,但是通過js調試工作可以看到,傳回報錯 no-session
因為查詢了過後entitymanager關閉了。。再懶加載就報錯
在web.xml加上以下代碼即可
<!-- 加上OpenEntityManager -->
<filter>
<filter-name>openEntity</filter-name>
<filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>openEntity</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
解決前台分頁條件和背景字段不一緻
前台請求的分頁條件,我們的分頁對象字段對應不上 page目前頁 rows查詢條數
需要給查詢對象添加set方法 前台資訊封裝對象是使用的set方法
自動封裝
我們傳回給前台的資料,字段比對不上,easyui隻需要total和rows
EasyUi前台需要的資料,total總共的條數 rows代表每頁的條數,我們能查多少條數傳回不管,
反正傳回的是rows每頁的條數
前台的頁數是點選的時候确定的,變化的主要是資料rows 總條數的值用于展示 其他的是前台自動顯示
八.查詢對象
進階查詢需要持久化接口繼承JpaSpecificationExecutor
T findOne(Specification<T> spec);
List<T> findAll(Specification<T> spec);
Page<T> findAll(Specification<T> spec, Pageable pageable);
Specification spec手動實作
//要使用find(Specification)功能需要repository層實作Specification接口
//覆寫方法相當于得到Predicate(控制條件where。。等等)
//然後再使用Specification的查詢功能可以同時傳入分頁對象,排序對象
@Test
public void testJpaSpecificationExecutor() throws Exception {
List<Employee> list = employeeRepository.findAll(new Specification<Employee>() {
@Override
public Predicate toPredicate(Root<Employee> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
//ROOT 1.相當于得到元素 。。要操作的字段。。age..username。。 root.get("元素")
//cb.like(元素,條件)。。
Path path = root.get("username");
Predicate predicate1 = cb.like(path, "%1%");
Path agepath = root.get("age");
Predicate predicate2 = cb.ge(agepath, 30);
Predicate predicate = cb.and(predicate1, predicate2);
return predicate;
}
});
list.forEach(e-> System.out.println(e));
}
使用插件jpa-spec
pom.xml引入:
<!-- jpa的SpecificationSpecification功能封裝 -->
<dependency>
<groupId>com.github.wenhao</groupId>
<artifactId>jpa-spec</artifactId>
<version>3.1.1</version>
<!-- 把所有依賴都過濾 -->
<exclusions>
<exclusion>
<groupId>*</groupId>
<artifactId>*</artifactId>
</exclusion>
</exclusions>
</dependency>
public abstract class BaseQuery {
//所有的持久化接口都需要分頁查詢
//排序。。抽取出來
//分頁需要的字段,每個類的查詢對象各自使用
//分頁查詢需要目前頁,每頁size
private int currentPage=1;
private int pageSize=10;
//相容EasyUI的分頁
//前台傳入的是 目前頁page 目前頁的要請求條數rows 将值設定給我們的
public void setPage(int page) {
this.currentPage = page;
}
public void setRows(int rows) {
this.pageSize = rows;
}
//背景的頁數要減1
private int currentJPAPage;
//第一頁從0開始計算
public int getJpaCurrentPage() {
return currentPage-1;
}
//通過名字來排序
private String orderName;
//前台傳入的是降序還是升序
private String orderType="ASC";
//
public Sort creatSort(){
//如果名字條件不為空才建立sort對象,為空不建立
if (StringUtils.isNotBlank(orderName)){
//建立sort對象 orderType.toUpperCase()取得前台的排序條件
Sort sort = new Sort(Sort.Direction.valueOf(orderType.toUpperCase()),orderName);
}
return null;
}
//定義規範,子類的方法名字都一樣。。
public abstract Specification creatSpec();
public int getCurrentPage() {
return currentPage;
}
public void setCurrentPage(int currentPage) {
this.currentPage = currentPage;
}
public int getPageSize() {
return pageSize;
}
public void setPageSize(int pageSize) {
this.pageSize = pageSize;
}
public String getOrderName() {
return orderName;
}
public void setOrderName(String orderName) {
this.orderName = orderName;
}
public String getOrderType() {
return orderType;
}
public void setOrderType(String orderType) {
this.orderType = orderType;
}
}