官网:https://mybatis.org/mybatis-3/zh/index.html
环境:
回顾:
框架:配置文件、最好的方式:官网文档
SSM框架:Spring SpringMVC Mybatis
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BMmIzH2v-1690204448185)(MyBatis.assets/image-20210113203554906.png)]
如何获得Mybatis
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.5.3version>
dependency>
数据持久化
为什么需要持久化?
Dao层、Service层、Controller层
什么叫持久层?
最重要的一点:使用的人超级多
思路:搭建环境 - 导入Mybatis - 编写代码 - 测试
全部文件:
mybatis-config.xml(会变)
MybatisUtil.java(不会变)
User.java(操作的表只要是user
,就不会改变)
UserDao.java(接口)
UserMapper.xml(会变)
UserDaoTest.java(小部分改变)
(想项目运行成功,一定保证数据库正常连接!)
CREATE DATABASE `mybatis`;
use `mybatis`;
CREATE TABLE `user` (
`id` INT(20) NOT NULL,
`name` VARCHAR(30) NULL DEFAULT NULL COLLATE 'utf8_general_ci',
`pwd` VARCHAR(30) NULL DEFAULT NULL COLLATE 'utf8_general_ci',
PRIMARY KEY (`id`) USING BTREE
)
COLLATE='utf8_general_ci'
ENGINE=InnoDB
;
INSERT INTO `user` (id,`name`,pwd) VALUES
(1,"陈雨晴",123456),(2,"付姗",123456789),(3,"马雨雨",112233)
第一步:新建一个普通的maven项目
pom.xml
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>5.1.34version>
dependency>
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.5.3version>
dependency>
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>3.8.1version>
<scope>testscope>
dependency>
第一步:创建Maven模块
第二步:
在 src/main/resouce/mybatis-config.xml
DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<environments default="development">
<environment id="development">
<transactionManager type="JDBC" />
<dataSource type="POOLED">
<property name="driver" value="com.mysql.jdbc.Driver" />
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSl=true&useUnicode=true&characterEncoding=utf8" />
<property name="username" value="root" />
<property name="password" value="123456" />
dataSource>
environment>
environments>
configuration>
//sqlSessionFactory - sqlSession
public class MybatisUtil {
private static SqlSessionFactory sqlSessionFactory;// 提升作用域
static {
try {
// 使用mybatis第一步:获取sqlSessionFactory对象
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public static SqlSession getSqlSession() {
return sqlSessionFactory.openSession();
}
}
和mybatis/user
表一一对应的实体类
package com.CYQ.pojo;
public class User {
private int id;
private String name;
private String pwd;
public User() {
super();
}
public User(int id, String name, String pwd) {
super();
this.id = id;
this.name = name;
this.pwd = pwd;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPwd() {
return pwd;
}
public void setPwd(String pwd) {
this.pwd = pwd;
}
@Override
public String toString() {
return "User [id=" + id + ", name=" + name + ", pwd=" + pwd + "]";
}
}
public interface UserDao {
List<User> getUserList();
}
由原来UserDaoImpl转换为一个UserMapper.xml配置文件
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.CYQ.dao.UserDao">
<select id="getUserList" resultType="com.CYQ.pojo.User">
select * from mybatis.user;
select>
mapper>
在mybatis-config.xml中进行配置
<mappers>
<mapper resource="com/CYQ/dao/UserMapper.xml">mapper>
mappers>
public class UserDaoTest {
@Test
public void test() {
//第一步:获取sqlSession对象
SqlSession sqlSession = MybatisUtil.getSqlSession();
//方式一:getMapper
UserDao userDao = sqlSession.getMapper(UserDao.class);//得到接口
List<User> userList = userDao.getUserList();//得到接口内的方法
for (User user : userList) {
System.out.println(user);
}
//关闭sqlSession
sqlSession.close();
}
}
注意点:
出现异常:问题常出现在 -
1.查看UserDao的实现类UserMapper.xml问题
2.是否在mybatis-config.xml中配置
具体:
1.配置文件没有注册(mybatis-config.xml的问题)
2.绑定接口错误(UserMapper.xml的问题)
3.方法名称不对(UserMapper.xml的问题)
4.返回类型不对(UserMapper.xml的问题)
所有的操作,均和接口(UserMapper.xml)和配置文件(Mybatis-config.xml)有关系
mybatis-config.xml
<mapper namespace="com.CYQ.dao.UserMapper">
<select id="getUserList" resultType="com.CYQ.pojo.User">
select * from mybatis.user;
select>
mapper>
namespace的包名要和接口(Dao/Mapper)的名字一致
选择:查询语句
id
:就是对应的namespace
中对应的方法名
resultType
:Sql语句执行的返回值!
parameterType
:参数的类型
步骤:
第一步:编写接口
public interface UserMapper {
//查询全部用户
List<User> getUserList();
//根据id查询用户
User getUserById(int id);
//insert一个用户
int addUser(User user);
//修改用户
int updateUser(User user);
//删除用户
int deleteUser(int id);
}
第二步:编写对应的对应的SQL语句
<mapper namespace="com.CYQ.dao.UserMapper">
<select id="getUserList" resultType="com.CYQ.pojo.User">
select * from mybatis.user;
select>
<select id="getUserById" parameterType="int" resultType="com.CYQ.pojo.User">
select * from mybatis.user where id=#{id}
select>
<insert id="addUser" parameterType="com.CYQ.pojo.User">
insert into mybatis.user(id,name,pwd) values (#{id},#{name},#{pwd});
insert>
<update id="updateUser" parameterType="com.CYQ.pojo.User">
update mybatis.user set name=#{name},pwd=#{pwd} where id=#{id};
update>
<delete id="deleteUser" parameterType="int">
delete from mybatis.user where id=#{id};
delete>
mapper>
第三步:测试
增删改:一定需要提交事务
查:不需要提交事务
public class UserDaoTest {
//查询
@Test
public void test() {
//第一步:获取sqlSession对象
SqlSession sqlSession = MybatisUtil.getSqlSession();
//方式一:getMapper(推荐使用!)
UserMapper userDao = sqlSession.getMapper(UserMapper.class);//得到接口
List<User> userList = userDao.getUserList();
//方式二:(不推荐使用)
//List
for (Object user : userList) {
System.out.println(user);
}
//关闭sqlSession
sqlSession.close();
}
//查询
@Test
public void test01() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
User user = userMapper.getUserById(3);
System.out.println(user.toString());
sqlSession.close();
}
//增删改 - 需要提交事务!!!
@Test
public void test02() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
userMapper.addUser(new User(4, "陈朋", "1995"));
//提交事务:
sqlSession.commit();
//关闭sqlSession
sqlSession.close();
}
//修改
@Test
public void test03() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
userMapper.updateUser(new User(1,"陈雨晴","1997"));
//提交事务:
sqlSession.commit();
//关闭sqlSession
sqlSession.close();
}
//删除
@Test
public void test04() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
userMapper.deleteUser(4);
//提交事务:
sqlSession.commit();
//关闭sqlSession
sqlSession.close();
}
}
假设,我们的实体类(或者数据库中的表),属性(或字段)过多,我们应当考虑使用map!
map传递参数,直接在sql中取出key即可;对象传递参数,需要直接在sql中取对象的属性。
只有一个基本参数的情况下,可以直接在sql中取到
多个参数用Map,或者注解!
对象和map的对比(认真对比)
UserMapper.xml
对象
<insert id="addUser" parameterType="com.CYQ.pojo.User">
insert into mybatis.user(id,name,pwd) values (#{id},#{name},#{pwd});
insert>
map
<insert id="addUser2" parameterType="map">
insert into mybatis.user(id,name,pwd) values (#{userId},#{userName},#{password});
insert>
UserMapperTest.java
对象
@Test
public void test02() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
userMapper.addUser(new User(4, "陈朋", "1995"));
//提交事务:
sqlSession.commit();
//关闭sqlSession
sqlSession.close();
}
map
//增加
@Test
public void test002() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
HashMap<String,Object> map = new HashMap<String, Object>();
map.put("userId", 4);
map.put("userName", "杨帆");
map.put("password", "1996");
userMapper.addUser2(map);
//提交事务:
sqlSession.commit();
//关闭sqlSession
sqlSession.close();
}
模糊查询:一定要防止sql注入的问题
防止SLQ注入的写法,在业务中使用多!!!
第一步:接口
public interface UserMapper {
//查询全部用户
List<User> getUserLike();
}
第二步:SQL
<select id="getUserById" resultType="com.CYQ.pojo.User">
select * from mybatis.user where name like #{value}
select>
<select id="getUserById" resultType="com.CYQ.pojo.User">
select * from mybatis.user where name like "%"#{value}"%"
select>
第三步:测试
@Test
public void test01() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
List<User> userList = userMapper.getUserLike("%陈%");
for (Object user : userList) {
System.out.println(user);
}
sqlSession.close();
}
//防止sql注入
@Test
public void test01() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);//获得接口
List<User> userList = userMapper.getUserLike("陈");
for (Object user : userList) {
System.out.println(user);
}
sqlSession.close();
}
configuration(配置)
<environments default="development">
<environment id="development">
<transactionManager type="JDBC"/>
<dataSource type="POOLED">
<property name="driver" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSl=true&useUnicode=true&characterEncoding=utf8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
dataSource>
environment>
environments>
事务管理器:transactionManager
的type:JDBC和MANAGED两种
数据源:dataSource
的type:UNPOOLED/POOLED/JNDI
Mybatis默认的事务管理器是:JDBC,连接池:POOLED
我们可通过properties属性来实现引用配置文件
这些属性可以在外部配置,并可以进行动态替换。既可以在典型的 Java 属性文件中配置,也可以在 properties 元素的子元素中设置【db.properties】
4.3.1编写配置文件
位置:src/main/resources/db.properties
driver = com.mysql.jdbc.Driver
url = jdbc:mysql://localhost:3306/mybatis?useSSl=true&useUnicode=true&characterEncoding=utf8
username = root
password = 123456
4.3.2在核心配置文件mybatis-config.xml中引入
<properties resource="db.properties" >
<property name="username" value="root"/>
<property name="password" value="1233456"/>
properties>
两种方式:1.直接引入配置文件 2.在中添加标签。
如果两个配置矛盾,优先使用配置文件中的。
环境配置可更改为:
<environments default="development">
<environment id="development">
<transactionManager type="JDBC" />
<dataSource type="POOLED">
<property name="driver" value="${driver}" />
<property name="url" value="${url}" />
<property name="username" value="${username}" />
<property name="password" value="${password}" />
dataSource>
environment>
environments>
类型别名可为 Java 类型设置一个缩写名字
存在的意义:减少类完全限定名的冗余
第一步:修改mybatis-config.xml
<typeAliases>
<typeAlias type="com.CYQ.pojo.User" alias="User"/>
typeAliases>
<typeAliases>
<package name="com.CYQ.pojo"/>
typeAliases>
第二步:修改UserMapper.xml
<select id="getUserList" resultType="User">
select * from mybatis.user;
select>
<select id="getUserList" resultType="user">
select * from mybatis.user;
select>
两种方式:
typeAlias
package
实体类比较少的时候,使用第一种;实体类比较多的时候,使用第二种
第一种可以自定义别名;第二种不可以自定义别名,但是可以通过注解自定义别名。
(推荐)方式一:通过***.xml文件
<mappers>
<mapper resource="com/CYQ/dao/UserMapper.xml">mapper>
mappers>
方式二:通过class文件
<mappers>
<mapper class="com.CYQ.dao.UserMapper"/>
mappers>
注意点:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-F1QZofHV-1690204448188)(MyBatis.assets/image-20210115155310926.png)]
生命周期和作用域,是至关重要的,因为错误的使用会导致非常严重的并发问题
SqlSessionFactoryBuilder
SqlSessionFactory
想象成:数据库连接池
一旦创建就一直存在,没有任何理由丢弃它或重新创建另一个实例
最佳作用域:应用作用域
最简单的就是使用单例模式或者静态单例模式
SqlSession
连接到连接池的一个请求
SqlSession的实例不是线程安全的,因此是不能被共享的,所以最佳作用域:请求/方法作用域
需要关闭,否则资源被占用!
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-45erbNyG-1690204448189)(MyBatis.assets/image-20210115160635011.png)]
这里的每一个Mapper,就代表一个具体的业务
例子见:Mybatis-03 Module
新建一个项目,拷贝之前的,测试实体类字段不一致的情况
数据库结构:
数据库结构:
mybatis(数据库)
user(表)
id(字段)
name(字段)
pwd(字段)
实体类构造:构造类的属性和字段不对应
public class User {
private int id;
private String name;
private String password;
...
}
测试代码:
@Test
public void getUserById() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);// 获得接口
User user = userMapper.getUserById(5);
System.out.println(user.toString());
sqlSession.close();
}
测试出现问题:查不出来pwd
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MUDoEdO4-1690204448189)(MyBatis.assets/image-20210115190327962.png)]
结果集映射:
id name pwd
id name pasword
<mapper namespace="com.CYQ.dao.UserMapper">
<resultMap type="User" id="userMap">
<result column="id" property="id" />
<result column="name" property="name" />
<result column="pwd" property="password" />
resultMap>
<select id="getUserById" resultMap="userMap">
select * from user where id=#{id}
select>
mapper>
resultMap
元素是 MyBatis 中最重要最强大的元素。
ResultMap 的设计思想是:对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系。
1.起别名:
<select id="getUserById" parameterType="int" resultType="User">
select id,name,pwd as password from mybatis.user where id=#{id}
select>
2.结果集映射
<mapper namespace="com.CYQ.dao.UserMapper">
<resultMap type="User" id="userMap">
<result column="id" property="id" />
<result column="name" property="name" />
<result column="pwd" property="password" />
resultMap>
<select id="getUserById" resultMap="userMap">
select * from user where id=#{id}
select>
mapper>
如果一个数据库操作出现异常,我们需要排错。日志就是最好的助手!
曾经:sout,dubug
现在:日志工厂
Mybatis中,包含的种类:
SLF4J
LOG4J 【掌握】
LOG4J2
JDK_LOGGING
COMMONS_LOGGING
STDOUT_LOGGING【掌握】
NO_LOGGING
在Mybatis中,具体使用哪个日志实现,在设置中实现
<settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
settings>
STDOUT_LOGGING - 为标准日志输出
结果显示:
什么是log4J
第一步:导入依赖
<dependency>
<groupId>log4jgroupId>
<artifactId>log4jartifactId>
<version>1.2.17version>
dependency>
第二步:配置properties文件
log4j.properties
#将等级为DEBUG的日志信息输出到console和file这两个目的地,console和file的定义在下面的代码
log4j.rootLogger=DEBUG,console,file
#控制台输出的相关设置
log4j.appender.console = org.apache.log4j.ConsoleAppender
log4j.appender.console.Target = System.out
log4j.appender.console.Threshold=DEBUG
log4j.appender.console.layout = org.apache.log4j.PatternLayout
log4j.appender.console.layout.ConversionPattern=[%c]-%m%n
#文件输出的相关设置
log4j.appender.file = org.apache.log4j.RollingFileAppender
#日志文件生成位置,可以自定义
log4j.appender.file.File=./log/CYQ.log
log4j.appender.file.MaxFileSize=10mb
log4j.appender.file.Threshold=DEBUG
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=[%p][%d{yy-MM-dd HH:mm:ss}][%c]%m%n
#日志输出级别
log4j.logger.org.mybatis=DEBUG
log4j.logger.java.sql=DEBUG
log4j.logger.java.sql.Statement=DEBUG
log4j.logger.java.sql.ResultSet=DEBUG
log4j.logger.java.sql.PreparedStatement=DEBUG
第三步:配置log4J为日志工厂实现
<settings>
<setting name="logImpl" value="LOG4J"/>
settings>
第四步:测试使用
直接junit测试UserMapperTest.java
第一步:导包(一定是log4j的Logger)
import org.apache.log4j.Logger;
第二步:日志对象,参数为当前类的class
static Logger logger = Logger.getLogger(UserDaoTest.class);
第三步:日志级别
@Test
public void testLog4j() {
logger.info("info:进入了testLog4j方法");
logger.debug("debug:进入了testLog4j方法");
logger.error("error:");
}
思考:为什么要分页?
SELECT * FROM user limit startindex,pagesize
SELECT * FROM user limit 3;--[0,3]
使用Mybatis实现分页!(核心:SQL)
1.接口
public interface UserMapper {
//Mybatis实现分页
List<User> getUserByLimit(Map<String, Integer> map);
}
2.UserMapper.xml
<select id="getUserByLimit" parameterType="map"
resultMap="userMap">
SELECT * FROM user limit #{startIndex},#{pageSize}
select>
3.测试
@Test
public void getUserByLimit() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
HashMap<String,Integer> map = new HashMap<String, Integer>();
map.put("startIndex", 0);
map.put("pageSize", 2);
List<User> users = userMapper.getUserByLimit(map);
for (User user : users) {
System.out.println(user);
}
sqlSession.close();
}
不再使用SQL实现分页
1.接口:
public interface UserMapper {
//rouBounds实现分页
List<User> getUserByRowBounds();
}
2.UserMapper.xml
<select id="getUserByRowBounds" resultMap="userMap">
SELECT * FROM user
select>
3.测试
@Test
public void getUserByRowBounds() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
//RowBounds实现
//每页显示两个,从第二个对象开始
RowBounds rowBounds = new RowBounds(1,2);//底标从0开始
//通过java代码实现分页
List<Object> userList = sqlSession.selectList("com.CYQ.dao.UserMapper.getUserByRowBounds",null,rowBounds);
for (Object object : userList) {
System.out.println(object);
}
sqlSession.close();
}
Mybatis-04
关于接口的理解
三个面向区别
1.注解直接在接口上实现:
public interface UserMapper {
@Select("select * from user")
List<User> getUsers();
}
2.在核心配置文件(mybatis-config.xml)中绑定接口
<mappers>
<mapper class="com.CYQ.dao.UserMapper"/>
mappers>
3.测试
@Test
public void test() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
List<User> users = mapper.getUsers();
for (User user : users) {
System.out.println(user);
}
sqlSession.close();
}
}
本质:反射机制实现
底层:动态代理
代理模式示意图:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DoWoD8yc-1690204448192)(MyBatis.assets/image-20210116182759016.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-96BVGasS-1690204448192)(MyBatis.assets/image-20210116185705344.png)]
我们可以在工具类创建的时候,实现自动提交事务!
第一步:设置自动提交事务为true
在MybatisUtil.java中设置
public static SqlSession getSqlSession() {
//默认为false,我们设置为true,true - 代表自动提交
SqlSession sqlSession = sqlSessionFactory.openSession(true);
return sqlSession;
}
第二步:接口
public interface UserMapper {
//方法存在多个参数,所有的参数前面必须加上:@Param注解
@Select("select * from user where id=#{id}")
User getUserById(@Param("id")int id);
@Insert("insert into user (id,name,pwd) values (#{id},#{name},#{password})")
int addUser(User user);
@Update("update user set name=#{name},pwd=#{password} where id=#{id}")
int updateUser(User user);
@Delete("delete from user where id=#{id}")
int deleteUser(@Param("id")int id);
}
第三步:测试
public class UserDaoTest {
@Test
public void test() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
// List users = mapper.getUsers();
// User user = mapper.getUserById(4);
// mapper.addUser(new User(6,"汪苏泷","1989"));
mapper.updateUser(new User(5, "许嵩", "1987"));
sqlSession.close();
}
}
关于@Param()注解
1.基本类型的参数,无论有多少个,每个都需要加
2.引用类型不用加
3.如果只有一个基本类型,可以忽略,但是建议大家写上
4.我们在SQL中引用的,就是我们在@Param()中设定的属性名!!!
#{},${}区别
第一步:安装LomBok插件
具体步骤参考:安装LomBok
第二步:在项目中导入lombok的jar包(Maven仓库形式)
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.12version>
<scope>providedscope>
dependency>
第三步:写注解
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private int id;
private String name;
private String password;
}
@Getter/@Setter
@FiledNameConstants
@ToString
@EqualsAndHashCode
@AllArgsConstructor/@RequiredArgsConstructor/@NoArgsConstructor
@Data
@Builder
@Singular
@Delegate
@value
@Accessors
@wither
@NonNull
@Cleanup
@Synchronized
@Log
@SneakyThrows
说明部分:
@Data:无参构造/Getter&Setter/toString/hashCode/equals
@AllArgsConstructor:全参构造
@NoArgsConstructor:无参构造
优点:
缺点:
Mybatis-05
多对一:
创建相关表
CREATE TABLE `teacher` (
`id` INT(10) NOT NULL,
`name` VARCHAR(30) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=INNODB DEFAULT CHARSET=UTF8;
INSERT INTO teacher(`id`, `name`) VALUES (1, '秦老师');
CREATE TABLE `student` (
`id` INT(10) NOT NULL,
`name` VARCHAR(30) DEFAULT NULL,
`tid` INT(10) DEFAULT NULL,
PRIMARY KEY (`id`),
KEY `fktid` (`tid`),
CONSTRAINT `fktid` FOREIGN KEY (`tid`) REFERENCES `teacher` (`id`)
) ENGINE=INNODB DEFAULT CHARSET=UTF8;
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('1', '小明', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('2', '小红', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('3', '小张', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('4', '小李', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('5', '小王', '1');
第一步:导入lombok
第二步:新建实体类:Teacher/Student
//实体类:学生
@Data
public class Student {
private int id;
private String name;
//学生需要关联一个老师
private Teacher teacher;
}
//实体类:老师
@Data
public class Teacher {
private int id;
private String name;
}
第三步:编写接口文件:TeacherMapper.java StudentMapper.java
编写对应xml文件:TeacherMapper.xml StudentMapper.xml
第四步:核心文件(mybatis-config.xml)中配置
第五步:测试
StudentMapper.xml
<mapper namespace="com.CYQ.dao.StudentMapper">
<select id="getStudent" resultMap="StudentTeacher">
select * from student
select>
<resultMap type="Student" id="StudentTeacher">
<result property="id" column="id" />
<result property="name" column="name" />
<association property="teacher" column="tid" javaType="Teacher" select="getTeacher" />
resultMap>
<select id="getTeacher" resultType="Teacher">
select * from teacher where id=#{tid}
select>
mapper>
<mapper namespace="com.CYQ.dao.StudentMapper">
<select id="getStudent2" resultMap="StudentTeacher2">
select s.id sid,s.name sname,t.name tname
from student s ,teacher t
where s.tid=t.id
select>
<resultMap type="Student" id="StudentTeacher2">
<result property="id" column="sid"/>
<result property="name" column="sname"/>
<association property="teacher" javaType="Teacher">
<result property="name" column="tname"/>
association>
resultMap>
mapper>
回顾MySQL多对一查询方式
Mybatis-06
比如:一个老师拥有多个学生!
对于老师而言,就是一对多的关系!
第一步:导入lombok
第二步:新建实体类:Teacher/Student
//实体类:学生
@Data
public class Student {
private int id;
private String name;
private int tid;
}
//实体类:老师
@Data
public class Teacher {
private int id;
private String name;
// 一个老师有多个学生
private List<Student> students;
}
第三步:编写接口文件:TeacherMapper.java StudentMapper.java
编写对应xml文件:TeacherMapper.xml StudentMapper.xml
第四步:核心文件(mybatis-config.xml)中配置
第五步:测试
TeacherMapper.xml
<mappers>
<select id="getTeacher2" resultMap="TeacherStudent2">
select * from teacher where id=#{tid}
select>
<resultMap type="teacher" id="TeacherStudent2">
<collection property="students" javaType="ArrayList" ofType="Student" select="getStduentByTeacherId" column="id">
collection>
resultMap>
<select id="getStduentByTeacherId" resultType="Student">
select * from student where tid=#{id}
select>
mappers>
TeacherMapper.xml
<mapper namespace="com.CYQ.dao.TeacherMapper">
<select id="getTeacher" resultMap="TeacherStudent">
select s.id sid,s.name sname,t.name tname,t.id tid
from student s,teacher t
where s.tid=t.id and t.id=#{tid}
select>
<resultMap type="Teacher" id="TeacherStudent">
<result property="id" column="tid"/>
<result property="name" column="tname"/>
<collection property="students" ofType="Student">
<result property="id" column="sid"/>
<result property="tid" column="tid"/>
collection>
resultMap>
mapper>
1.关联 - association 【多对一】
2.集合:collection 【一对多
3.javaType & ofType
注意点:
面试高频:
Mybatis-07
什么是动态SQL:
动态SQL就是指根据不同的条件,生成不同的SQL语句
if
choose(when,otherwise)
trim(where,set)
foreach
创建相关表
CREATE TABLE `blog`(
`id` VARCHAR(50) NOT NULL COMMENT '博客id',
`title` VARCHAR(100) NOT NULL COMMENT '博客标题',
`author` VARCHAR(30) NOT NULL COMMENT '博客作者',
`create_time` DATETIME NOT NULL COMMENT '创建时间',
`views` INT(30) NOT NULL COMMENT '浏览量'
)ENGINE=INNODB DEFAULT CHARSET=utf8
第一步:导包
第二步:编写实体类
@Data
public class Blog {
private String id;
private String title;
private String author;
private Date createTime;
private int views;
}
第三步:
编写接口BlogMapper.java
ublic interface BlogMapper {
//插入数据:
int addBlog(Blog blog);
}
编写*Mapper.xml
<mapper namespace="com.CYQ.dao.BlogMapper">
mapper>
第四步:核心配置文件(mybatis-config.xml)中配置
<mappers>
<mapper class="com.CYQ.dao.BlogMapper"/>
mappers>
第五步:测试
BlogMapper.xml
<mapper>
<select id="queryBlogIf" parameterType="map" resultType="Blog">
select * from blog where 1=1
<if test="title!=null">
and title = #{title}
if>
<if test="author!=null">
and author=#{author}
if>
select>
mapper>
测试:
@Test
public void queryBlogIf() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
HashMap<Object, Object> map = new HashMap<Object, Object>();
map.put("title", "大娱乐家");
map.put("author", "汪苏泷");
List<Blog> blogs = mapper.queryBlogIf(map);
for (Blog blog : blogs) {
System.out.println(blog);
}
sqlSession.close();
}
相当于case … switch …
<mapper>
<select id="queryBlogChoose" parameterType="map"
resultType="Blog">
select * from blog
<where>
<choose>
<when test="title!=null">
title = #{title}
when>
<when test="author!=null">
and author=#{author}
when>
<otherwise>
and views=#{views}
otherwise>
choose>
where>
select>
mapper>
where
<mapper>
<select id="queryBlogIf" parameterType="map" resultType="Blog">
select * from blog
<where>
<if test="title!=null">
and title = #{title}
if>
<if test="author!=null">
and author=#{author}
if>
where>
select>
mapper>
set
<mapper>
<update id="updateBlog" parameterType="map">
update blog
<set>
<if test="title!=null">
title = #{title},
if>
<if test="author!=null">
author=#{author}
if>
set>
where id=#{id}
update>
mapper>
所谓的动态SQL,本质还是SQL语句,只是我们可以在sql层面,去执行一个逻辑代码
有时候,我们会将一些功能的 部分,提取出来,方便复用!
第一步:使用标签,抽取出公共的部分
第二步:在需要使用sql的部分,使用标签引用即可
<mapper namespace="com.CYQ.dao.BlogMapper">
<!-- 使用sql抽出公共的部分 -->
<sql id="if-title-author">
<if test="title!=null">
and title = #{title}
</if>
<if test="author!=null">
and author=#{author}
</if>
</sql>
<select id="queryBlogIf" parameterType="map" resultType="Blog">
select * from blog
<where>
<!-- 在需要使用sql的部分,使用include -->
<include refid="if-title-author"/>
</where>
</select>
</mapper>
注意:
select * from user where 1=1 and (id=1 or id=2 or id=3)
<select id="queryBlogForeach" parameterType="map" resultType="Blog">
select * from blog
<where>
<foreach collection="ids" item="id" open="and (" close=")" separator="or">
id=#{id}
foreach>
where>
select>
测试:
@Test
public void queryBlogForeach() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
HashMap<Object, Object> map = new HashMap<Object, Object>();
int[] ids = {1,2,3};
map.put("ids", ids);
mapper.queryBlogForeach(map);
sqlSession.close();
}
动态SQL就是在拼接SQL语句,我们只要保证SQL的正确性,按照SQL的格式,去排列组合就可以了
建议:现在MySQL中写出完整的SQL,再对应的去修改成我们的动态SQL
查询 : 连接数据库,耗资源
一次查询的结果,暂存在可以直接取到的地方!!! - 内存 - 放在内存中的这些数据就叫“缓存”
我们再次查询相同的数据的时候,直接走“缓存”就可以,不用走数据库
1、什么是缓存 [ Cache ]?
2、为什么使用缓存?
3、什么样的数据能使用缓存?
一级缓存也叫本地缓存:
测试步骤:
第一步:开启日志 mybatis-config.xml
<settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
settings>
第二步:测试,在同一个Session中查询两次相同记录
@Test
public void getUserById() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.getUserById(3);
System.out.println(user);
User user2 = mapper.getUserById(3);
System.out.println(user2);
System.out.println(user==user2);
sqlSession.close();
}
第三步:查看日志输出
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LQNmyCsa-1690204448193)(MyBatis.assets/image-20210119123231303.png)]
缓存失效的情况
1.增删改操作,可能会改变原来的数据,所以必定会刷新缓存!
2.查询不同的东西
3.查询不同的***Mapper.xml
4.手动清理缓存
@Test
public void getUserById() {
SqlSession sqlSession = MybatisUtil.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.getUserById(3);
System.out.println(user);
sqlSession.clearCache();//手动请理缓存
sqlSession.close();
}
小结:一级缓存是默认开启的,只在一次Sqlsession中有效,也就是拿到 连接到关闭 这个时间段内的缓存!!!
一级缓存相当于map(或者说:一级缓存就是一个map)
第一步:开启全局缓存 - mybatis-config.xml
<settings>
<setting name="cacheEnabled" value="true"/>
settings>
第二步:在要使用二级缓存的***Mapper.xml中配置
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
<cache />
第三步:测试
实体类要记得序列化
小结:
1.只要开启了二级缓存,在同一个Mapper下就有效
2.所有的数据都会先放在一级缓存中
3.只有当会话提交或关闭的时候,才会提交到二级缓存
先访问二级缓存,再访问一级缓存
使用步骤:
第一步:导包
<dependency>
<groupId>org.mybatis.cachesgroupId>
<artifactId>mybatis-ehcacheartifactId>
<version>1.1.0version>
dependency>
第二步:配置ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
updateCheck="false">
<diskStore path="./tmpdir/Tmp_EhCache"/>
<defaultCache
eternal="false"
maxElementsInMemory="10000"
overflowToDisk="false"
diskPersistent="false"
timeToIdleSeconds="1800"
timeToLiveSeconds="259200"
memoryStoreEvictionPolicy="LRU"/>
<cache
name="cloud_user"
eternal="false"
maxElementsInMemory="5000"
overflowToDisk="false"
diskPersistent="false"
timeToIdleSeconds="1800"
timeToLiveSeconds="1800"
memoryStoreEvictionPolicy="LRU"/>
ehcache>