MyBatis【MyBatis的增删改查操作与单元测试】
单元测试(unit testing),是指对软件中的最⼩可测试单元进⾏检查和验证的过程就叫单元测试
单元测试是开发者编写的⼀⼩段代码,⽤于检验被测代码的⼀个很⼩的、很明确的(代码)功能是否正确。执⾏单元测试就是为了证明某段代码的执⾏结果是否符合我们的预期。如果测试结果符合我们的预期,称之为测试通过,否则就是测试未通过(或者叫测试失败)
(可以搭配日志打印)(不用手动测试)(有事务回滚机制)Spring Boot 项⽬创建时会默认单元测试框架 spring-boot-test,⽽这个单元测试框架主要是依靠另⼀个著名的测试框架 JUnit 实现的,打开 pom.xml 就可以看到,以下信息是 Spring Boot 项⽬创建是⾃动添加的:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
如果pom.xml文件内没有我们可以手动添加
在我们创建的XXXMapper接口中



@Slf4j注解 打印日志
@Resource注解进行UserMapper属性注入


import com.example.demo.model.User;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
import javax.annotation.Resource;
class UserControllerTest {
private UserController userController;
void save() {
User user = new User();
user.setId(1);
user.setName("Java Test");
user.setPassword("123");
boolean result = userController.save(user);
// 使⽤断⾔判断最终的结果是否符合预期
Assertions.assertEquals(true, result);
// Assertions.assertTrue(result);
}
}

UserMapper接口修改用户信息方法:

UserMapper.xml 进行SQL修改用户信息语句传输:

单元测试:

单元测试结果:

UserMapper接口删除用户信息方法:
// 删除方法{根据id删除这一条数据}
public int del(("id") Integer id);
UserMapper.xml 进行SQL删除用户信息语句传输:
<!-- 根据用户id删除用户 -->
<delete id="del">
delete from userinfo where id=#{id}
</delete>
单元测试:
void del() {
int result = userMapper.del(2);
System.out.println(result);
}
单元测试解结果:

UserMapper接口增加用户信息方法:
// 添加方法
public int add(UserInfo userInfo);
UserMapper.xml 进行SQL增加用户信息语句传输:
<!-- 添加用户,返回受影响的行数 -->
<insert id="add">
insert into userinfo(username,password,photo)
values(#{username},#{password},#{photo})
</insert>
单元测试:
void add() {
UserInfo userInfo = new UserInfo();
userInfo.setUsername("王五");
userInfo.setPassword("123");
userInfo.setPhoto("default.png");
int result = userMapper.add(userInfo);
System.out.println(result);
}
单元测试解结果:

默认情况下返回的是受影响的⾏号,如果想要返回⾃增 id,具体实现如下
UserMapper接口增加用户信息方法:
// 添加方法
// 添加用户返回受影响的行数
public int addGetId(UserInfo userInfo);
UserMapper.xml 进行SQL增加用户信息语句传输:
●useGeneratedKeys:这会令 MyBatis 使⽤ JDBC 的 getGeneratedKeys ⽅法来取出由数据库内部⽣成的主键
(⽐如:像 MySQL 和 SQL Server 这样的关系型数据库管理系统的⾃动递增字段),默认值:false
●keyProperty:指定能够唯⼀识别对象的属性,MyBatis 会使⽤ getGeneratedKeys 的返回值或 insert 语句的 selectKey ⼦元素设置它的值,
默认值:未设置(unset)。如果⽣成列不⽌⼀个,可以⽤逗号分隔多个属性名称
<!-- 添加用户,返回受影响的行数和自增id -->
<insert id="addGetId" useGeneratedKeys="true" keyProperty="id">
insert into userinfo(username,password,photo)
values(#{username},#{password},#{photo})
</insert>
单元测试:
void addGetId() {
UserInfo userInfo = new UserInfo();
userInfo.setUsername("王五");
userInfo.setPassword("123");
userInfo.setPhoto("default.png");
System.out.println("添加之前 user id:" + userInfo.getId());
int result = userMapper.addGetId(userInfo);
System.out.println("受影响的行数:" + result);
System.out.println("添加之后 user id:" + userInfo.getId());
}
单元测试解结果:

UserMapper接口查询用户信息方法:
// 查询方法{根据用户id查询用户}
// @Param("username")是mybatis中设置xml文件的SQL中查询username
public UserInfo getUserByUsername(("username") String username);
UserMapper.xml 进行SQL查询用户信息语句传输:
这里我们要添加resultType="com.example.demo.model.UserInfo来选择要查询对象目录地址
<!-- 根据用户username 查询用户信息 -->
<select id="getUserByUsername" resultType="com.example.demo.model.UserInfo">
select * from userinfo where username=#{username}
</select>
单元测试:
void getUserByUsername() {
UserInfo userInfo = userMapper.getUserByUsername("admin");
//日志打印
log.info("用户信息:" + userInfo);
}
单元测试解结果:

● #{}:预编译处理
● ${}:字符直接替换
预编译处理是指:
MyBatis 在处理#{}时,会将 SQL 中的 #{} 替换为?号,使⽤ PreparedStatement的 set ⽅法来赋值
直接替换:
是MyBatis 在处理 ${} 时,就是把 ${} 替换成变量的值
#{}查询代码:


${}查询代码:
在这里插入图片描述


定义不同: #{}是预处理; ${}是直接处理
使用数据类型不同:#{}适用于所有数据类型; ${}只适用于数值类型
安全性问题:#{}性能高,不会引起数据库安全问题; ${}会引起数据库SQL注入问题
UserMapper接口查询用户信息方法:
// 登陆功能
public UserInfo login(("username") String usernaem,("password")String password);
UserMapper.xml 进行SQL查询用户信息语句传输:
<!-- 根据用户输入username 和 password 信息实现登陆用户 -->
<select id="login" resultType="com.example.demo.model.UserInfo">
select * from userinfo where username='${username}' and password='${password}'
</select>
单元测试:
当我我们输入正常的用户名和密码时
void login() {
String username = "XXX";
String password = "XXX";
// "' or 1 =' 1" SQL注入问题(使用$ 严重危险问题)
UserInfo userInfo = userMapper.login(username,password);
log.info("用户信息"+userInfo);
}
单元测试结果:

单元测试:
当我我们输入带有SQL语句注入的密码时
void login() {
String username = "admin";
String password = "' or 1 =' 1";
// "' or 1 =' 1" SQL注入问题(使用$ 严重危险问题)
UserInfo userInfo = userMapper.login(username,password);
log.info("用户信息"+userInfo);
}
单元测试结果:
这个时候就会查询到这个表里所有的用户信息了,这就是${}的SQL注入安全问题

我们发现
这个占位符缺陷这么大
,
拿我们为什么还有设置这个占位符呢
,
因为存在并合理
,
{}这个占位符缺陷这么大,拿我们为什么还有设置这个占位符呢,因为存在并合理,
这个占位符缺陷这么大,拿我们为什么还有设置这个占位符呢,因为存在并合理,{}是可以使用SQL语句中的关键字这一功能的
UserMapper接口查询用户信息方法:
// 获取列表,根据创建时间进行正序或倒序
public List<UserInfo> getOrderList(("order") String order);
UserMapper.xml 进行SQL查询用户信息语句传输:
当我们使用#{}占位符进行查询时
<!-- 根据用户createtime信息进行排序 查询用户信息 -->
<select id="getOrderList" resultType="com.example.demo.model.UserInfo">
select * from userinfo order by createtime #{order}
</select>
单元测试:
void getOrderList() {
List<UserInfo> list = userMapper.getOrderList("desc");
log.info("列表"+list);
}
单元测试结果:

这时候我们就想到被我们抛弃的${}


UserMapper接口查询用户信息方法:
// 模糊查询(like)用户名信息
public List<UserInfo> getListName(("username") String username);
UserMapper.xml 进行SQL查询用户信息语句传输:
<select id="findUserByName2" resultType="com.example.demo.model.UserInFo">
select * from userinfo where username like '%#{username}%'
</select>
单元测试:
void login() {
String username = "admin";
String password = "admin";
// "' or 1 =' 1" SQL注入问题(使用$ 严重危险问题)
UserInfo userInfo = userMapper.login(username,password);
log.info("用户信息"+userInfo);
}
单元测试结果:
我们会因为不能直接用’%#{username}%'进行模糊查询

select * from userinfo where username like '%#{username}%'相当于:
SQL查询select * from userinfo where username like '%'username'%';
这个是不能直接使⽤ ${},可以考虑使⽤ mysql 的内
<!-- 根据用户输入username 模糊信息实现查询用户们信息 -->
<select id="getListName" resultType="com.example.demo.model.UserInfo">
select * from userinfo where username like concat ('%',#{username},'%')
</select>

resultMap 使⽤场景:
● 字段名称和程序中的属性名不同的情况,可使⽤ resultMap 配置映射
● ⼀对⼀和⼀对多关系可以使⽤ resultMap 映射并查询数据。
字段名和属性名不同的情况
例如 MySQL表中的 用户名属性叫 username ,而我我们在对象使用的属性名是user
而这时我们就需要在UserMapper.xml中使用resultMap进行修改了

resultType VS resultMap 具体使用:

很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的: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?并散列所有无济于事。
我有一些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
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/
我遵循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
我已经构建了一些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
我在app/helpers/sessions_helper.rb中有一个帮助程序文件,其中包含一个方法my_preference,它返回当前登录用户的首选项。我想在集成测试中访问该方法。例如,这样我就可以在测试中使用getuser_path(my_preference)。在其他帖子中,我读到这可以通过在测试文件中包含requiresessions_helper来实现,但我仍然收到错误NameError:undefinedlocalvariableormethod'my_preference'.我做错了什么?require'test_helper'require'sessions_hel
只是想确保我理解了事情。据我目前收集到的信息,Cucumber只是一个“包装器”,或者是一种通过将事物分类为功能和步骤来组织测试的好方法,其中实际的单元测试处于步骤阶段。它允许您根据事物的工作方式组织您的测试。对吗? 最佳答案 有点。它是一种组织测试的方式,但不仅如此。它的行为就像最初的Rails集成测试一样,但更易于使用。这里最大的好处是您的session在整个Scenario中保持透明。关于Cucumber的另一件事是您(应该)从使用您的代码的浏览器或客户端的角度进行测试。如果您愿意,您可以使用步骤来构建对象和设置状态,但通常您
我有:When/^(?:|I)follow"([^"]*)"(?:within"([^"]*)")?$/do|link,selector|with_scope(selector)doclick_link(link)endend我打电话的地方:Background:GivenIamanexistingadminuserWhenIfollow"CLIENTS"我的HTML是这样的:CLIENTS我一直收到这个错误:.F-.F--U-----U(::)failedsteps(::)nolinkwithtitle,idortext'CLIENTS'found(Capybara::Element