zoukankan      html  css  js  c++  java
  • SpringBoot+Thymeleaf+MyBatis 实现RESTful API

    1.项目结构

    2.创建数据库表

    这里使用MySQL5.7。

    CREATE TABLE `user` (
      `id` int(13) NOT NULL AUTO_INCREMENT COMMENT '主键',
      `name` varchar(33) DEFAULT NULL COMMENT '姓名',
      `age` int(3) DEFAULT NULL COMMENT '年龄',
      `createTime` datetime DEFAULT NULL,
      PRIMARY KEY (`id`)
    ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    

    3.代码详解

    引入Thymeleaf、SpringBoot整合mybatis的jar包、mysql的驱动包。

    <!-- pom.xml -->
    <!-- ... -->
    <dependencies>
        <!-- ... -->
    	
        <dependency>
              <groupId>org.springframework.boot</groupId>
              <artifactId>spring-boot-starter-thymeleaf</artifactId>
        </dependency>
        <dependency>
              <groupId>org.mybatis.spring.boot</groupId>
              <artifactId>mybatis-spring-boot-starter</artifactId>
              <version>1.3.2</version>
        </dependency>
        <dependency>
              <groupId>mysql</groupId>
              <artifactId>mysql-connector-java</artifactId>
              <version>5.1.38</version><!--$NO-MVN-MAN-VER$-->
        </dependency>
    </dependencies>
    <!-- ... -->
    
    # application.properties
    server.port=8081
    server.servlet.context-path=/restful-demo
    spring.thymeleaf.cache=false
    spring.thymeleaf.prefix=classpath:/templates/
    spring.thymeleaf.check-template-location=true
    spring.thymeleaf.suffix=.html
    spring.thymeleaf.encoding=UTF-8
    spring.thymeleaf.mode=HTML
    spring.datasource.url=jdbc:mysql://localhost:3306/restful_sql?useSSL=false
    spring.datasource.username=root
    spring.datasource.password=root
    spring.datasource.driver-class-name=com.mysql.jdbc.Driver
    spring.mvc.hiddenmethod.filter.enabled=true
    mybatis.config-location=classpath:mybatis-config.xml
    mybatis.mapper-locations=classpath:mapper-mapping/*.xml
    logging.level.com.example.demo.mapper=debug
    

    spring.thymeleaf.*是Thymeleaf相关的配置,spring.datasource.*是连接数据库相关的配置。
    spring.mvc.hiddenmethod.filter.enabled=true启用HiddenHttpMethodFilter,以允许PUT请求或者DELETE请求。
    mybatis.config-location=classpath:mybatis-config.xml指定mybatis的配置文件。
    mybatis.mapper-locations=classpath:mapper-mapping/*.xml指定mybatis的映射文件。
    logging.level.com.example.demo.mapper=debug配置打印包com.example.demo.mapper的日志。

    实体类:

    package com.example.demo.model;
    
    import java.time.LocalDateTime;
    
    import org.springframework.format.annotation.DateTimeFormat;
    
    import com.fasterxml.jackson.annotation.JsonFormat;
    
    public class User {
        private Integer id;
        private String name;
        private int age;
        @DateTimeFormat(pattern="yyyy-MM-dd HH:mm:ss")
        @JsonFormat(pattern="yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
        private LocalDateTime createTime;
        
        // setter and getter
        
    }
    

    Controller:

    package com.example.demo.controller;
    
    import java.time.LocalDateTime;
    import java.util.ArrayList;
    import java.util.HashMap;
    import java.util.List;
    import java.util.Map;
    
    import javax.servlet.http.HttpServletRequest;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Controller;
    import org.springframework.ui.Model;
    import org.springframework.web.bind.annotation.CrossOrigin;
    import org.springframework.web.bind.annotation.ModelAttribute;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestMethod;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;
    import org.springframework.web.servlet.ModelAndView;
    
    import com.example.demo.model.User;
    import com.example.demo.service.UserService;
    
    @Controller
    public class UserController {
        @Autowired
        private UserService userService;
    	
        @ModelAttribute
        public void getUserById(@RequestParam(value="id",required=false) Integer id, Model model){
              if (id != null) {
                    User user = userService.getUserById(id);
                    if(user != null) {
                          model.addAttribute("user", user);
                    }
              }
        }
    
        // ...
    	
        @RequestMapping("/index2")
        public ModelAndView index2(){
              ModelAndView modelAndView = new ModelAndView("user/index2");
              // modelAndView.addObject("users", userService.findAll());
              return modelAndView;
        }
    	
        @ResponseBody
        @RequestMapping("/getUsers")
        public Map<String, Object> getUsers(HttpServletRequest request){
              int pageNO = Integer.parseInt(request.getParameter("page"));// 当前页
              int pageSize = Integer.parseInt(request.getParameter("rows"));// 每页行数
              Map<String, Object> params = new HashMap<>();
              params.put("name", request.getParameter("name"));
              return userService.getUsers(pageNO, pageSize, params);
        }
    	
        /*
         ** Restful API设计
            /user          POST	保存
            /user/{id}     PUT      更新
            /user/{id}     GET      查询
            /user/{id}     DELETE	删除
        */
        @ResponseBody
        @RequestMapping(value="/user", method=RequestMethod.POST)
        public String saveUser(User user){
              user.setCreateTime(LocalDateTime.now());
              userService.saveUser(user);
              return "success";
        }
    	
        @ResponseBody
        @RequestMapping(value="/user/{id}", method=RequestMethod.PUT)
        public String updateByPrimaryKey(@PathVariable("id") Integer id, @ModelAttribute("user") User user){
              // 参数中的id值已经封装到User对象里面,而且这里借助@ModelAttribute使得修改不会影响User对象的其他字段值,而只修改表单中提交的字段值。
              userService.updateByPrimaryKey(user);
              return "success";
        }
    	
        @ResponseBody
        @RequestMapping(value="/user/{id}", method=RequestMethod.GET)
        public User getUserById(@PathVariable("id") Integer id){
              return userService.getUserById(id);
        }
    	
        @ResponseBody
        @RequestMapping(value="/user/{id}", method=RequestMethod.DELETE)
        public String deleteUserById(@PathVariable("id") Integer id){
              userService.deleteByPrimaryKey(id);
              return "success";
        }
    }
    

    Service:

    package com.example.demo.service;
    
    import java.util.HashMap;
    import java.util.List;
    import java.util.Map;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;
    import org.springframework.transaction.annotation.Transactional;
    
    import com.example.demo.mapper.UserMapper;
    import com.example.demo.mapper.UserMapper2;
    import com.example.demo.model.User;
    
    @Transactional
    @Service
    public class UserService {
        @Autowired
        private UserMapper userMapper;
        @Autowired
        private UserMapper2 userMapper2;
    	
        // ...
    
        public Map<String, Object> getUsers(int pageNO, int pageSize, Map<String, Object> params) {
              List<User> users = userMapper2.getUsers(params);
              int count = 0;
              if(users.size()>=1){
                    count = users.size();
                    int fromIndex = (pageNO-1)*pageSize;
                    int toIndex = pageNO*pageSize;
                    users = users.subList(fromIndex, count>toIndex?toIndex:count); // 分页
              }
              Map<String, Object> map = new HashMap<>();
              map.put("rows", users);
              map.put("total", count);
              return map;
        }
    
        public User getUserById(Integer id) {
              return userMapper.getUserById(id);
        }
    
        public void saveUser(User user) {
              userMapper2.insert(user);
        }
    
        public void updateByPrimaryKey(User user) {
              userMapper2.updateByPrimaryKey(user);
        }
    
        public void deleteByPrimaryKey(Integer id) {
              userMapper2.deleteByPrimaryKey(id);
        }
    	
    }
    

    Mapper:

    @Mapper
    public interface UserMapper2 {
        List<User> getUsers(Map<String, Object> params);
    
        void insert(User user);
    
        void updateByPrimaryKey(User user);
    
        void deleteByPrimaryKey(Integer id);
    
        @Select("SELECT * FROM User u WHERE u.id = #{id}")
        User getUserById(@Param("id") Integer id);
    
    }
    

    注解@Mapper标注的类可以被其他类通过@Autowired注入。
    可以直接在接口上面通过注解@Select来写SQL,也可以通过映射文件来写SQL。
    mybatis映射文件:

    <!-- UserMapper.xml -->
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
    <mapper namespace="com.example.demo.mapper.UserMapper2">
        <select id="getUsers" parameterType="Map" resultType="com.example.demo.model.User">
              select * from user u where 1=1
              <if test="name!=null">
                    and u.name like concat('%',#{name},'%')
              </if>
        </select>
    	
        <insert id="insert" parameterType="com.example.demo.model.User" >
              insert into user (name, age, createTime) values (#{name}, #{age}, #{createTime})
        </insert>
        <update id="updateByPrimaryKey" parameterType="com.example.demo.model.User" >
              update user set name = #{name}, age = #{age} where id = #{id}
        </update>
        <delete id="deleteByPrimaryKey" parameterType="java.lang.Integer" >
              delete from user where id = #{id}
        </delete>
    </mapper>
    

    HTML页面:

    <!-- index2.html -->
    <!DOCTYPE html>
    <html xmlns:th="http://www.thymeleaf.org">
    <head>
    <meta charset="UTF-8">
    <script th:src="@{/jquery-easyui/jquery.min.js}"></script>
    <script th:src="@{/jquery-easyui/jquery.easyui.min.js}"></script>
    <script th:src="@{/jquery-easyui/locale/easyui-lang-zh_CN.js}"></script>
    <link rel="stylesheet" type="text/css" th:href="@{/jquery-easyui/themes/default/easyui.css}" />
    <link rel="stylesheet" type="text/css" th:href="@{/jquery-easyui/themes/icon.css}" />
    <style>
        input[class^="easyui-"] {
               160px;
        }
        #toolbar,
        .userForm {
              font-size:1rem;
        }
        #toolbar>p {
              margin:0;
              margin-top:1em;
              margin-left:1em;
              margin-bottom:.5em;
        }
        #toolbar>p label,
        .userForm label {
              margin-right:1em;
        }
        .userForm p {
              50%;
              margin-left:auto;
              margin-right:auto;
        }
        #toolbar .button {
              margin-left:1em;
              margin-bottom:.5em;
        }
        .saveButton {
              text-align:center;
              margin-top:1em;
              margin-bottom:1em;
        }
    </style>
    <title>User Index</title>
    </head>
    <body>
    <table id="users"></table>
    <div id="toolbar">
        <p><label for="name_s">名称</label><input class="easyui-searchbox" id="name_s" /></p>
        <div class="button">
              <a class="easyui-linkbutton" data-options="iconCls:'icon-add',plain:true" onclick="addUser()">添加</a>
              <a class="easyui-linkbutton" data-options="iconCls:'icon-edit',plain:true" onclick="editUser()">修改</a>
              <a class="easyui-linkbutton" data-options="iconCls:'icon-remove',plain:true" onclick="removeUser()">删除</a>
        </div>
    </div>
    <div class="easyui-dialog" data-options="closed:true,modal:true" id="userDialog" style="450px;">
    <div class="userForm">
        <form id="userForm" method="post">
              <input id="id" name="id" type="hidden" />
              <p><label for="name">名称</label><input class="easyui-textbox" id="name" name="name" data-options="required:true" /></p>
              <p><label for="age">年龄</label><input class="easyui-numberbox" id="age" name="age" data-options="required:true,min:0" /></p>
        </form>
        <div class="saveButton">
              <a class="easyui-linkbutton" data-options="iconCls:'icon-save'" onclick="saveUser()" id="save">保存</a>
        </div>
    </div>
    </div>
    <script th:inline="javascript">
        var basePath = /*[[${#request.getContextPath()}]]*/ "http://localhost:8081/restful-demo";
    </script>
    <!-- 下面使用"th:inline='none'",避免报错! -->
    <script th:inline="none">
    $(function(){
        $('#users').datagrid({
              url:basePath+'/getUsers',
              rownumbers:true,
              pagination:true,
              singleSelect:true,
              pageNumber:1,
              pageSize:5,
              pageList:[5,10,15],
              toolbar : '#toolbar',
              columns:[[
                    {
                          field:'id',
                          title:'ID',
                          100
                    },
                    {
                          field:'name',
                          title:'名称',
                          100
                    },
                    {
                          field:'age',
                          title:'年龄',
                          100
                    },
                    {
                          field:'createTime',
                          title:'创建时间',
                          150,
                          align:'center',
                          formatter:function (value) {
                                return value?`<span title="${value}">${value}</span>`:'';
                          }
                    },
                    ]]
              });
    	
              $('#name_s').searchbox({
                    searcher:function(value){
                          $('#users').datagrid('load',{
                                name: value,
                          });
                    }
              });
    	
              $('#userForm').form({
                    onLoadSuccess: function(data){
                          if(data){
                                // 当加载完表单数据后,才启用保存按钮。
                                $('#save').linkbutton('enable');
                    }
              }
        });
    });
    
    function addUser(){
        $('#userDialog').panel('setTitle', '添加用户').dialog('center').dialog('open');
    }
    
    async function editUser(){
        var row = $("#users").datagrid('getSelected');
        if(row){
              $('#save').linkbutton('disable');
              var user = await $.get(basePath+'/user/'+row.id);
              $('#userForm').form('load', user);
              $('#userDialog').panel('setTitle', '修改用户').dialog('center').dialog('open');
        } else {
              $.messager.alert('警告操作!', '请选择一条记录!', 'info');
        }
    }
    
    function removeUser(){
        var row = $("#users").datagrid('getSelected');
        if(row){
              $.messager.confirm('确认操作', "确定要删除选中的记录?", function (flag) {
                    if(flag){
                          $.post(basePath+'/user/'+row.id,{_method:'DELETE'},function (data) {
                                if(data==="success"){
                                      $.messager.alert('提示信息', '删除成功!','info');
                                      $('#users').datagrid('reload');
                                } else {
                                      $.messager.alert('提示信息','删除失败!','info');
                                }
                          });
                    }
              });
        } else {
            $.messager.alert('警告操作!', '请选择一条记录!', 'info');
        }
    }
    
    function saveUser(){
        var url = basePath+'/user';
        var id = $('#id').val();
        if(id){
              url += "/"+id;
        }
        $('#userForm').form('submit',{
              url: url,
              onSubmit: function(param){
                    if(id){
                          param._method='PUT';
                    }
                    return $(this).form('enableValidation').form('validate');
              },
              success: function(data){
                    if(data==="success"){
                          clearForm();
                          $("#users").datagrid('reload');
                          $.messager.alert('提示信息','保存成功!','info');
                    }
              }
        });
    }
    
    function clearForm(){
        $('#userForm').form('clear');
        $('#userDialog').dialog('close');
    }
    </script>
    </body>
    </html>
    

    HTML页面使用了Thymeleaf的语法和EasyUI框架。
    展示页面:



    示例代码下载地址:https://pan.baidu.com/s/17Kq4gUrUh160jr3fLTSjKg(提取码:r14t)

  • 相关阅读:
    NHibernate 使用点滴
    看了二十四画生的文章才发现ASP.NET Portal Starter Kit中调整顺序的一个Bug
    闲话静态构造函数
    ASP.NET Portal starter Kit 之页面配置文件
    VB智能中文提示的一个小工具 VBCommenter 1.2.5
    61条面向对象设计的经验原则
    asp.net Portal Starter kit改造Portal的Html文本编辑器
    Wrox出版社的 Professional DotNetNuke Asp.NET Portals [E文版] 电子书的下载地址
    asp.net Datagrid 资源
    C#与vb.net的区别
  • 原文地址:https://www.cnblogs.com/gzhjj/p/13600911.html
Copyright © 2011-2022 走看看