thymeleaf springboot CRUD (新手向)

thymeleaf springboot CRUD (新手向)

前言

我是以我原来mybatis+druid+mysql做CRUD为基础https://www.jianshu.com/p/4b285b5b34f8,主要借鉴了https://blog.csdn.net/zhuyu19911016520/article/details/81537154。这个网址的没做分页,我的做了能在我的程序上运行的分页处理以及一些优化。其中,包括显示上一页与下一页(如果有的话),当前是第几页以及总共有多少页。在删除时不会删了最后一页的最后一个,直接跳到首页(这是不知为何的机制所致),而是对使用者友好的删除后还在当前页面(如果当前页面仍然有数据)。由于刚接触,所以在对于工具所提供的接口只会使用,背后的机制是不了解的。总之,网上找的代码大概意思能看懂,具体代码层面的工作流程需要通过修复一个一个bug,以及加些新功能或者优化一起学习。

放一下thymeleaf官网 https://www.thymeleaf.org/。如果英语差,可以看菜鸟教程。

项目目的

前后端共同实现mysql分页增删改查

文件结构


上代码

pom.xml

<?xmlversion="1.0"encoding="UTF-8"?>

<projectxmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">

<modelVersion>4.0.0</modelVersion>

<parent>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-parent</artifactId>

<version>1.5.1.RELEASE</version>

<relativePath/><!--lookupparentfromrepository-->

</parent>

<groupId>com.vue</groupId>

<artifactId>mybatis_druid_mysql</artifactId>

<version>0.0.1-SNAPSHOT</version>

<name>mybatis_druid_mysql</name>

<description>mybatis_druid_mysql</description>

<properties>

<java.version>1.8</java.version>

</properties>

<dependencies>

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-data-jpa</artifactId>

</dependency>

<dependency>

<groupId>org.mybatis.spring.boot</groupId>

<artifactId>mybatis-spring-boot-starter</artifactId>

</dependency>

<!--web依赖,包含servlet,内置tomcat等-->

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-web</artifactId>

</dependency>

<!--mysql依赖-->

<dependency>

<groupId>mysql</groupId>

<artifactId>mysql-connector-java</artifactId>

<scope>runtime</scope>

</dependency>

<!--mybatis依赖-->

<dependency>

<groupId>org.mybatis.spring.boot</groupId>

<artifactId>mybatis-spring-boot-starter</artifactId>

<version>1.3.2</version>

</dependency>

<!--druid依赖包,配合springBoot项目使用-->

<dependency>

<groupId>com.alibaba</groupId>

<artifactId>druid-spring-boot-starter</artifactId>

<version>1.1.14</version>

</dependency>

<dependency>

<groupId>org.projectlombok</groupId>

<artifactId>lombok</artifactId>

</dependency>

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-thymeleaf</artifactId>

<!--<version>2.3.4.RELEASE</version>-->

</dependency>

</dependencies>

<build>

<plugins>

<plugin>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-maven-plugin</artifactId>

</plugin>

</plugins>

</build>

</project>

Controller

importcom.mybatis_druid_mysql.demo.Entity.Customer;

importcom.mybatis_druid_mysql.demo.Service.Service;

importorg.springframework.beans.factory.annotation.Autowired;

importorg.springframework.data.domain.Page;

importorg.springframework.ui.ModelMap;

importorg.springframework.web.bind.annotation.ModelAttribute;

importorg.springframework.web.bind.annotation.RequestMapping;

importorg.springframework.web.bind.annotation.RequestParam;

importjava.util.Iterator;

@org.springframework.stereotype.Controller

@RequestMapping("customers")

publicclassController{

@Autowired

privateServiceservice;

@RequestMapping("edit")

publicStringedit(ModelMapmap,@RequestParam(value="pageNum",defaultValue="0")intpageNum,@RequestParam(value="pageSize",defaultValue="2")intpageSize,@RequestParam(defaultValue="0")intid){

//isAdd : 向前端页面返回一个是新增与编辑的标识

if(id>0){

map.addAttribute("isAdd",0);

map.addAttribute("customer",service.getById(id));

map.addAttribute("pageNum",pageNum);

}else{

map.addAttribute("isAdd",1);

map.addAttribute("customer",newCustomer());

map.addAttribute("pageNum",pageNum);

       }

return"customer/edit";

   }

@RequestMapping("goingToDel")

publicStringgoingToDel(ModelMapmap,@RequestParam(value="pageNum",defaultValue="0")intpageNum,@RequestParam(value="pageSize",defaultValue="2")intpageSize,@RequestParam(value="id")intid){

map.addAttribute("customer",service.getById(id));

map.addAttribute("pageNum",pageNum);

return"customer/goingToDel";

   }

@RequestMapping("list")

publicStringlist(ModelMapmap,@RequestParam(value="pageNum",defaultValue="0")intpageNum,@RequestParam(value="pageSize",defaultValue="2")intpageSize) {

System.out.println("============================");

Page<Customer>customers=service.getUserList(pageNum,pageSize);

System.out.println("总页数"+customers.getTotalPages());

System.out.println("当前页是:"+pageNum);

System.out.println("分页数据:");

Iterator<Customer>u=customers.iterator();

while(u.hasNext()){

System.out.println(u.next().toString());

       }

map.addAttribute("customers",customers);

map.addAttribute("pageNum",pageNum);

return"customer/list";

   }

//新增和编辑

@RequestMapping("save")

publicStringsave(ModelMapmap,@RequestParam(value="pageNum",defaultValue="0")intpageNum,@RequestParam(value="pageSize",defaultValue="2")intpageSize,@ModelAttributeCustomercustomer){

if(customer==null){

return"customer/list";

       }

if(customer.getId()!=null&&customer.getId()>0){

service.edit(customer);

}else{

service.add(customer);

       }

Page<Customer>customers=service.getUserList(pageNum,pageSize);

map.addAttribute("customers",customers);

map.addAttribute("pageNum",pageNum);

return"customer/list";

   }

@RequestMapping("del")

publicStringdel(ModelMapmap,@RequestParam(value="pageNum",defaultValue="0")intpageNum,@RequestParam(value="pageSize",defaultValue="2")intpageSize,@ModelAttributeCustomercustomer){

if(customer==null){

return"customer/list";

       }

service.delete(customer.getId());

Page<Customer>customers=service.getUserList(pageNum,pageSize);

Iterator<Customer>u=customers.iterator();

map.addAttribute("customers",customers);

map.addAttribute("pageNum",pageNum);

System.out.println(pageNum);

if(u.hasNext()){

System.out.println("1");

return"redirect:list?pageNum="+pageNum;

       }

elseif(customers.getTotalPages()!=0) {

pageNum--;

System.out.println("2");

return"redirect:list?pageNum="+pageNum;

       }

else{

System.out.println("3");

return"redirect:list?pageNum=0";

       }

   }

}

ModelMap对象主要用于传递控制方法处理数据到结果页面,也就是说我们把结果页面上需要的数据放到ModelMap对象中即可。通过以下方法向页面传递参数:

publicModelMapaddAttribute(StringattributeName,ObjectattributeValue){...}

publicModelMapaddAttribute(ObjectattributeValue){...}

publicModelMapaddAllAttributes(Collection<?>attributeValues) {...}

publicModelMapaddAllAttributes(Map<String,?>attributes){...}

thymeleaf配置

###ThymeLeaf??

spring.thymeleaf.mode=HTML5

#????????HTML,XMLTEXTJAVASCRIPT

spring.thymeleaf.encoding=UTF-8

#???????

spring.thymeleaf.content-type=text/html

#????,?????

spring.thymeleaf.cache=false

#?????false,?????????????

spring.thymeleaf.prefix=classpath:/templates/

spring.thymeleaf.suffix=.html

spring.thymeleaf.prefix= classpath:/templates/spring.thymeleaf.suffix= .html

设置了thymeleaf的寻址地址以及文件格式。

Controller中return "customer/list";就是在thymeleaf的寻址地址基础上再寻址找到相应的页面。

至于return "redirect:list?pageNum=0"; 这是个重定位。相当于调用Controller方法 http://127.0.0.1:8080/customers/list?pageNum=0

edit.html

<!DOCTYPEhtml>

<htmllang="en">

<head>

<metacharset="UTF-8"/>

<title>新增、编辑客户</title>

</head>

<body>

<formth:action="@{/customers/save}"method="post">

<!--//新增时不存在-->

<divth:if="${isAdd} == 0">

<labelstyle="color: red">id</label>

<inputtype="text"name="id"readonly="readonly"th:field="${customer.id}"style="background-color: red"/>

</div>

<div>

<label>name</label>

<inputtype="text"name="name"th:field="${customer.name}"/>

</div>

<div>

<label>age</label>

<inputtype="text"name="age"th:field="${customer.age}"/>

</div>

<div>

<label>used</label>

<inputtype="text"name="used"th:field="${customer.used}"/>

</div>

<div>

<inputtype="submit"value="提交"/>

</div>

<div>

<inputtype="button"value="返回"onclick="JavaScript:history.go(-1)"/>

</div>

<!--<div>-->

<!--<inputtype="button"value="删除"onclick='remove(this)'/>-->

<!--</div>-->

</form>

</body>

</html>

goingToDel.html

<!DOCTYPEhtml>

<htmllang="en">

<head>

<metacharset="UTF-8"/>

<title>删除客户</title>

</head>

<body>

<formth:action="@{/customers/del(pageNum=${pageNum},id=${customer.id})}"method="post">

<div>

<label>id</label>

<inputtype="text"name="id"readonly="readonly"th:field="${customer.id}"/>

</div>

<div>

<label>name</label>

<inputtype="text"name="name"readonly="readonly"th:field="${customer.name}"/>

</div>

<div>

<label>age</label>

<inputtype="text"name="age"readonly="readonly"th:field="${customer.age}"/>

</div>

<div>

<label>used</label>

<inputtype="text"name="used"readonly="readonly"th:field="${customer.used}"/>

</div>

<div>

<inputtype="submit"value="确定删除"/>

</div>

<div>

<inputtype="button"value="返回"onclick="JavaScript:history.go(-1)"/>

</div>

</form>

</body>

</html>

list.html

<!DOCTYPEhtml>

<htmllang="en"xmlns:th="http://www.thymeleaf.org">

<head>

<metacharset="UTF-8"/>

<title>客户列表</title>

<style>

table{border-collapse:collapse;}

table,th,td{border:1pxsolidblack;padding:5px;text-align:center;}

</style>

<linkrel="stylesheet"th:href="@{/css/bootstrap.css}"/>

</head>

<bodyclass="container">

<br/>

<br/>

<ath:href="edit">新增用户</a>

<br/>

<br/>

<table>

<thead>

<thwidth="20%">id</th>

<thwidth="20%">name</th>

<thwidth="20%">age</th>

<thwidth="20%">used</th>

<thwidth="20%">操作</th>

</thead>

<tbody>

<trth:each="customer : ${customers}">

<tdth:text="${customer.id}"></td>

<tdth:text="${customer.name}"></td>

<tdth:text="${customer.age}"></td>

<tdth:text="${customer.used}"></td>

<td>

<ath:href="@{/customers/edit(id=${customer.id})}">编辑</a>&nbsp;&nbsp;

<ath:href="@{/customers/goingToDel(pageNum=${pageNum},id=${customer.id})}">删除</a>

</td>

</tr>

</tbody>

</table>

<divth:text="当前页数"></div>

<divth:text="${pageNum}"></div>

<divclass="modal-footer no-margin-top">

<ulclass="pagination pull-right no-margin">

<!--首页-->

<li>

<ath:href="'/customers/list?pageNum=0'">首页</a>

</li>

<!--上一页-->

<lith:if="${customers.hasPrevious()}">

<ath:href="'/customers/list?pageNum=' + ${customers.previousPageable().getPageNumber()}"th:text="上一页"></a>

</li>

<!--中间页-->

<lith:each="pageNum:${#numbers.sequence(0, customers.getTotalPages()-1)}">

<ath:href="'/customers/list?pageNum=' + ${pageNum}"th:text="${pageNum + 1}"th:style="'font-weight:bold;background: #6faed9;'"></a>

</li>

<!--下一页-->

<lith:if="${customers.hasNext()}">

<ath:href="'/customers/list?pageNum=' + ${customers.nextPageable().getPageNumber()}"th:text="下一页"></a>

</li>

<!--尾页-->

<li>

<ath:href="'/customers/list?pageNum=' + ${customers.getTotalPages() - 1}">尾页</a>

</li>

</ul>

</div>

</body>

</html>

UserRepositrory

publicinterfaceUserRepositoryextendsJpaRepository<Customer,Long>{

}

Service

@org.springframework.stereotype.Service

publicinterfaceService{

//新增

voidadd(Customercustomer);

//编辑

voidedit(Customercustomer);

//删除

voiddelete(intid);

//获取单个

CustomergetById(intid);

//获取所有

List<Customer>listByAll();

Page<Customer>getUserList(intpageNum,intpageSize);

}

ServiceImpl

import com.mybatis_druid_mysql.demo.Entity.Customer;

import com.mybatis_druid_mysql.demo.Mapper.Mapper;

import com.mybatis_druid_mysql.demo.repository.UserRepository;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.data.domain.Page;

import org.springframework.data.domain.PageRequest;

import org.springframework.data.domain.Pageable;

import org.springframework.data.domain.Sort;

import java.util.List;

@org.springframework.stereotype.Service

public class ServiceImpl implements Service {

@Autowired

private Mapper mapper;

@Autowired

private UserRepository userRepository;

@Override

public void add(Customer customer) {

mapper.insert(customer);

}

@Override

public void edit(Customer customer) {

mapper.updateByPrimaryKey(customer);

}

@Override

public void delete(int id) {

mapper.deleteByPrimaryKey(id);

}

@Override

public Customer getById(int id) {

return mapper.selectByPrimaryKey(id);

}

@Override

public List listByAll() {

return mapper.listByAll();

}

@Override

public Page getUserList(int pageNum, int pageSize) {

Pageable pageable = new PageRequest(pageNum, pageSize);

Page customers = userRepository.findAll(pageable);

return customers;

}

}

其余的都在https://www.jianshu.com/p/4b285b5b34f8

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容