天天看點

spring全家桶,spring,springData JPA,springMvc的內建

一.建立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自動實作,自動繼承

五.分層

  1. domain實體類層
  2. repository持久化層
  3. service服務層
  4. web控制層

    層次分明,web調用service,service調用repository底層

六.tomcat的配置

spring全家桶,spring,springData JPA,springMvc的內建
spring全家桶,spring,springData JPA,springMvc的內建

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查詢條數

spring全家桶,spring,springData JPA,springMvc的內建

需要給查詢對象添加set方法 前台資訊封裝對象是使用的set方法

自動封裝

我們傳回給前台的資料,字段比對不上,easyui隻需要total和rows

spring全家桶,spring,springData JPA,springMvc的內建

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;
    }
}