草庐IT

项目一:基于Spring Boot的增删改查(包括测试流程)

コイの夢 2023-04-10 原文

使用Springboot实现简单的增删改查(仅记录,不喜勿喷)

首先介绍一下SpringBoot基本框架:

Spring Boot框架层次从上至下可分为5层:

分别View层,Controller层,Service层,Mapper层,Model层

一:View层:视图根据接受到的数据最终展示页面给用户浏览,需要与Controller层结合起来使用

二:Controller层:负责具体的业务模块流程的控制,响应用户的请求,调用Service层的接口来控制业务流程,决定使用何种视图并准备响应数据,并把接收到的参数传给Mapper,调用Mapper的方法接口

三:Service层:主要负责业务模块的逻辑应用设计,同时有一些是关于数据库处理的操作,但是不是直接和底层数据库相关联,而是首先设计接口,再设计实现其中的类,在接口实现方法中需要导入Mapper层,接着在Spring的配置文件中配置其实现的关联,从而在应用中调用Service接口来进行业务处理

四:Mapper层:主要是做数据持久层的工作,同时提供增删改查工作,Mapper层也是首先设计接口,再设计实现其中的类,具体实现在mapper.xml文件中,然后就可以在模块中调用此接口来进行数据业务的处理

五:Model层:创建对象,包括构造器和get、set方法和toString方法

Controller层:沟通前后端,注解为@RestController

Service层:沟通Mapper层和Controller层,注解为@Service

Mapper层:沟通数据库和Service层,注解为@Repository

对应每个实体,比如说user,需要分别的Controller层,Service层,Mapper层,

Mapper层里面包含一个接口和一个实体类impl,接口负责确定数据连接层的方法,

而impl文件就是直接用来连接数据库的一个实体类

具体的web项目:

controller层->service层(serviceimpl实现service接口)->mapper层->mapper.xml文件

解释:

controller层调用了service的具体功能和方法,service层由service对应的接口和实现类组成,serviceimpl实现service的相关接口同时完成相关的业务逻辑处理

service层(serviceimpl)再调用mapper层的接口进行业务逻辑应用的处理。mapper层的接口在对应的xml配置文件中进行配置、实现和关联,故mapper层的任务就是想数据库发送sql语句,完成数据的处理任务

下面开始进行准备工作:

1.新建项目,名称前缀为ProBuildLog

2.新建数据库wemovetest,新建数据表user,并填写部分数据

3.新建application.yml文件,文件代码如下:

server:
  port: 8080
spring:
  datasource:
    driverClassName: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/wemovetest?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8
    username: root
    password: 123456


mybatis-plus:
  #mapper映射文件路径
  mapper-locations: classpath*:mapper/*Mapper.xml
  #配置扫描包别名
  type-aliases-package: com.example.wemove.domain

4.新建以下几个包和包中的类,如图:

5编写相关类和接口的代码:

一:controller层

import com.example.wemove.service.ProBuildLogService;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*;

import javax.annotation.Resource;
@RestController
@RequestMapping("/user")
public class ProBuildLogController{
@Resource
private ProBuildLogService userService;
@RequestMapping("/show")                     //该注解直接返回字符串
    public String show(){
        return "user";
    }
    /**
     * 查找角色
     * @param id
     * @return
     */
    @GetMapping( "/getUser/{id}")
    public ProBuildLog getUser(@PathVariable Long id) {
        return userService.selectUserById(id);
    }
  	/**
     * 新增角色
     */
    @PostMapping
    public boolean insertUser(@Validated @RequestBody ProBuildLog role) {
        return userService.insertUser(role);
    }
    /**
     * 删除角色
     */
    @DeleteMapping("/{roleIds}")
    public boolean deleteUser(@PathVariable Long roleIds) {
        return userService.deleteUserById(roleIds);
    }
  	/**
     * 状态修改
     */
    @PutMapping("/changeStatus")
    public boolean changeStatus(@RequestBody ProBuildLog role) {
        return userService.updateUser(role);
    }

}

二:service层接口

package com.example.wemove.service;
import com.example.wemove.domain.ProBuildLog;
public interface ProBuildLogService{
  	/**
     * 根据id查询用户
     * @param id
     * @return
     */
    ProBuildLog selectUserById(Long id);
  	/**
     * 新增保存角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    boolean insertUser(ProBuildLog role);
    /**
     * 通过角色ID删除角色
     *
     * @param roleId 角色ID
     * @return 结果
     */
    boolean deleteUserById(Long roleId);
    /**
     * 修改保存角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    boolean updateUser(ProBuildLog role);
}

三:service层接口的imp

package com.example.wemove.service.impl;

import com.example.wemove.domain.ProBuildLog;
import com.example.wemove.mapper.ProBuildLogMapper;
import com.example.wemove.service.ProBuildLogService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
public class ProBuildLogServiceImpl implements ProBuildLogService  {
    /**
     * 注入mapper到service层
     */
    @Autowired
    private ProBuildLogMapper userMapper;          //创建了一个mapper类的变量
  	/**
     * 通过角色ID查询角色
     *
     * @param id 角色ID
     * @return 角色对象信息
     */
    @Override
    public ProBuildLog selectUserById(Long id) {
        return userMapper.selectUserById(id);
    }
  	/**
     * 新增保存角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean insertUser(ProBuildLog role) {
        // 新增角色信息
        return userMapper.insertUser(role)>0;
    }
  	/**
     * 修改保存角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean updateUser(ProBuildLog role) {
        // 修改角色信息
       return  userMapper.updateUser(role)>0;
    }
	    /**
     * 通过角色ID删除角色
     *
     * @param roleId 角色ID
     * @return 结果
     */
    @Override
    @Transactional
    public boolean deleteUserById(Long roleId) {
        return userMapper.deleteUserById(roleId)>0;
    }
}
}

l四:Mapper层接口

package com.example.wemove.mapper;

import com.example.wemove.domain.ProBuildLog;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface ProBuildMapper{
    /**
     * 根据id查询用户
     * @param id
     * @return 角色列表
     */
    ProBuildLog selectUserById(Long id);

    /**
     * 修改角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    int updateUser(ProBuildLog role);

    /**
     * 新增角色信息
     *
     * @param role 角色信息
     * @return 结果
     */
    int insertUser(ProBuildLog role);

    /**
     * 通过角色ID删除角色
     *
     * @param roleId 角色ID
     * @return 结果
     */
    int deleteUserById(Long roleId);
}

五:Mapper层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.wemove.mapper.ProBuildLogMapper">
<resultMap id="ProBuildLogResult" type="ProBuildLog">                                       //id是命名,type是类型
<id property="id" column="id"/>                                                             //property是domian里面变量的名称,column是数据库里面的名称
        <result property="name" column="name"/>
        <result property="age" column="age"/>
    </resultMap>
    <select id="selectUserById" parameterType="long" resultMap="ProBuildLogResult">         //id是命名,parameterType是接口类型
        select * from user where id=#{id}
    </select>
    <insert id="insertUser" parameterType="ProBuildLog">
        insert into user() value(#{id},#{name},#{age})
    </insert>

    <update id="updateUser" parameterType="ProBuildLog">
        update user set name=#{name},age=#{age} where id=#{id}
    </update>

    <delete id="deleteUserById" parameterType="Long">
        delete from user where id=#{id}
    </delete>

</mapper>

6.启动项目即可:

流程解析:
首先是Controller层,该层自动注入了Service层的接口,并且通过该接口的调用了Service层impl的方法,而impl层自动注入了Mapper层的接口,并且通过该接口调用了Mapper层xml的方法,xml文件中的方法通过对应的MySQL语句进行对数据库的操作,从而完成数据处理任务

如图:(图片转载)

 

7.项目测试:

项目成功启动后,打开ApiFox或PostMan开始进行数据的测试。

一:查询角色:

创建查询角色接口,接口类型为GET,接口路径为http://localhost:8080+/数据表名称+/注解

eg:http://localhost:8080/user/getUser/1

如图:

 

二:修改角色:

创建修改角色接口,接口类型为PUT,接口路径为http://localhost:8080+/数据表名称+/注解

eg:http://localhost:8080/user/changeStatus

修改前:

 _____________________________________________________________________________

 

_____________________________________________________________________________

修改后:

 

三:删除角色:

创建删除角色接口,接口类型为DELETE,接口路径为http://localhost:8080+/数据表名称+/注解

eg:http://localhost:8080/use/1

删除后如图:

 

四:新增角色:

创建新增角色接口,接口类型为POST,接口路径为http://localhost:8080+/数据表名称

eg:http://localhost:8080/user

_____________________________________________________________________________

新增后:

 

至此,我们已经完成了增删改查的相关工作

感谢浏览,欢迎指教和交流 

有关项目一:基于Spring Boot的增删改查(包括测试流程)的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  4. ruby - Ruby 的 Hash 在比较键时使用哪种相等性测试? - 2

    我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的:classAattr_reader:xdefinitialize(inner)@inner=innerenddefx;@inner.x;enddef==(other)@inner.x==other.xendenda=A.new(o)#oisjustanyobjectthatallowso.xb=A.new(o)h={a=>5}ph[a]#5ph[b]#nil,shouldbe5ph[o]#nil,shouldbe5我试过==、===、eq?并散列所有无济于事。

  5. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  6. ruby-on-rails - 项目升级后 Pow 不会更改 ruby​​ 版本 - 2

    我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby​​版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby​​版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘

  7. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  8. ruby-on-rails - 新 Rails 项目 : 'bundle install' can't install rails in gemfile - 2

    我已经像这样安装了一个新的Rails项目:$railsnewsite它执行并到达:bundleinstall但是当它似乎尝试安装依赖项时我得到了这个错误Gem::Ext::BuildError:ERROR:Failedtobuildgemnativeextension./System/Library/Frameworks/Ruby.framework/Versions/2.0/usr/bin/rubyextconf.rbcheckingforlibkern/OSAtomic.h...yescreatingMakefilemake"DESTDIR="cleanmake"DESTDIR="

  9. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

  10. ruby - 即使失败也继续进行多主机测试 - 2

    我已经构建了一些serverspec代码来在多个主机上运行一组测试。问题是当任何测试失败时,测试会在当前主机停止。即使测试失败,我也希望它继续在所有主机上运行。Rakefile:namespace:specdotask:all=>hosts.map{|h|'spec:'+h.split('.')[0]}hosts.eachdo|host|begindesc"Runserverspecto#{host}"RSpec::Core::RakeTask.new(host)do|t|ENV['TARGET_HOST']=hostt.pattern="spec/cfengine3/*_spec.r

随机推荐