哔哩哔哩狂神说Java-MyBatis学习网址
文章只为分享学习经验和自己复习用,学习还是该去查看正规视频网站和官方文档才更有效
https://mybatis.org/mybatis-3/zh/index.html
环境:
回顾:
SSM框架:配置文件的。最好的方法:看官网文档
如何获得MyBatis:
Maven仓库 https://mvnrepository.com/search?q=Mybatis
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.5.7version>
dependency>
Github https://github.com/
中文文档 https://mybatis.org/mybatis-3/zh/getting-started.html
主要是指:数据持久化
持久化就是将程序的数据在持久状态和瞬时状态转化的过程
瞬时状态:内存(断电即失)
持久状态:数据库(Jdbc),io文件持久化
为什么需要持久化?
有一些对象,不能让它丢掉
内存太贵,所以需要硬盘进行持久化
Dao层、Service层、Controller层
什么叫层?
主要工作:帮助程序员将数据存入到数据库中
优点:
最重要一点:使用的人多
思路:搭建环境 --> 导入MyBatis --> 编写代码 --> 测试
查看中文文档 https://mybatis.org/mybatis-3/zh/index.html
CREATE DATABASE `mybatis`;
USE `mybatis`;
CREATE TABLE `user`(
`id` INT(20) NOT NULL PRIMARY KEY,
`name` VARCHAR(30) DEFAULT NULL,
`pwd` VARCHAR(30) DEFAULT NULL
)ENGINE=INNODB DEFAULT CHARSET=utf8;
INSERT INTO `user`(`id`,`name`,`pwd`) VALUES
(1,'张三','123456'),
(2,'李四','234567'),
(3,'王五','345678')
1.创建一个普通的maven项目
2.删除src目录 (就可以把此工程当做父工程了,然后创建子工程)
3.pom.xml导入maven依赖,输入父类打包类
<project xmlns="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.0modelVersion>
<groupId>com.ssxxzgroupId>
<artifactId>Myabtis-StudyartifactId>
<packaging>pompackaging>
<version>0.0.1-SNAPSHOTversion>
<dependencies>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>8.0.12version>
dependency>
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.5.4version>
dependency>
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.12version>
<scope>testscope>
dependency>
dependencies>
project>
4.创建一个Module
父类pom.xml会自动添加子类关联
<modules>
<module>mybatis-01module>
modules>
子类会自动继承jar包
<project xmlns="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.0modelVersion>
<parent>
<groupId>com.ssxxzgroupId>
<artifactId>Myabtis-StudyartifactId>
<version>0.0.1-SNAPSHOTversion>
parent>
<artifactId>mybatis-01artifactId>
<build>
<resources>
<resource>
<directory>src/mian/resourcesdirectory>
<includes>
<include>**/*.propertiesinclude>
<include>**/*.xmlinclude>
includes>
<filtering>truefiltering>
resource>
<resource>
<directory>src/mian/javadirectory>
<includes>
<include>**/*.propertiesinclude>
<include>**/*.xmlinclude>
includes>
<filtering>truefiltering>
resource>
resources>
build>
project>
编写mybatis的核心配置文件,连接数据库
在resources中创建核心配置文件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&characterEncoding=UTF-8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
dataSource>
environment>
environments>
configuration>
编写mybatis工具类
创建SqlSession工厂模式,获得SqlSession实例
package com.ssxxz.utils;
import java.io.IOException;
import java.io.InputStream;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
//SqlSessionFactory -->SqlSession
public class MybatisUtils {
private static SqlSessionFactory sqlSessionFactory;
static {
try {
//使用Mybatis第一步 :获取sqlSessionFactory对象
String resource = "org/mybatis/example/mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
//既然有了 SqlSessionFactory,顾名思义,我们可以从中获得 SqlSession 的实例.
public static SqlSession getSqlSession() {
return sqlSessionFactory.openSession();
}
}
SqlSession 提供了在数据库执行 SQL 命令所需的所有方法。
实体类
package com.ssxxz.pojo;
public class User {
private int id;
private String name;
private String pwd;
public User() {
}
public User(int id, String name, String pwd) {
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 + "]";
}
}
Dao接口(UserDao也就是mapper)
package com.ssxxz.dao;
//Dao等价于mapper,以后都会写mapper
import java.util.List;
import com.ssxxz.pojo.User;
public interface UserDao {
//获取所有的用户
List<User> getUserList();
}
接口实现类 (由原来的UserDaoImpl转变为一个Mapper配置文件)
select返回类com.ssxxz.dao→UserMapper.xml
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.UserDao">
<select id="getUserList" resultType="com.ssxxz.pojo.User">
select * from mybatis.user;
select>
mapper>
package com.ssxxz.dao;
import java.util.Iterator;
import java.util.List;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
import com.ssxxz.pojo.User;
import com.ssxxz.utils.MybatisUtils;
public class UserDaoTest {
@Test
public void test() {
//第一步:获得SqlSession对象
SqlSession sqlSession = MybatisUtils.getSqlSession();
//执行
//方式一:getMapper
//利用SqlSession对象内的getMapper方法,获得UserDao.class这个类
UserDao userDao = sqlSession.getMapper(UserDao.class);
//利用UserDao获得List方法
List<User> userList = userDao.getUserList();
//方法二,旧版不建议使用
//.selectList根据方法的返回值去选择,selectMap...
List<User> userList = sqlSession.selectList("com.ssxxz.dao.UserDao.getUserList");
for (User user : userList) {
System.out.println(user);
}
//关闭
sqlSession.close();
}
}
每次收到HTTP请求,就可以打开一个SqlSession,返回一个响应,就关闭它
(1)MapperRegistry
org.apache.ibatis.binding.BindingException: Type interface com.kuang.dao.UserDao is not known to the MapperRegistry.
MapperRegistry是什么?
核心配置文件中注册mappers
<mappers>
<mapper resource="com/ssxxz/dao/UserMapper.xml"/>
mappers>
(2)找不到.xml文件
导入依赖
或者手动移到target文件中
因为我使用的eclipse依赖没有效果,所以只能手动,这个问题以后再解决。
搭建好框架后,增删改查只需要修改 接口类UserMapper.java
,SQL语句UserMapper.xml
和 测试类UserDaoTest.java
namespace中的包名要和 Dao/maper 接口的包名一致
选择,查询语句
编写接口
//查询全部的用户 select
List<User> getUserList();
//根据ID查询用户 select
User getUserById(int id);
编写对应的Mapper中的sql语句
<select id="getUserList" resultType="com.ssxxz.pojo.User">
select * from mybatis.user;
select>
<select id="getUserById" parameterType="int" resultType="com.ssxxz.pojo.User">
select * from mybatis.user where id = #{id};
select>
测试
//查看全部内容
@Test
public void test() {
//第一步:获得SqlSession对象
SqlSession sqlSession = MybatisUtils.getSqlSession();
//执行方式一:getMapper
//利用SqlSession对象内的getMapper方法,获得UserDao.class这个类
UserDao userDao = sqlSession.getMapper(UserDao.class);
//利用UserDao获得List方法
List<User> userList = userDao.getUserList();
for (User user : userList) {
System.out.println(user);
}
//关闭
sqlSession.close();
}
//查看指定内容
@Test
public void getUserById() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
User user = userDao.getUserById(1);
System.out.println(user);
sqlSession.close();
}
接口
//添加一个用户 insert
int addUser(User user);
SQL
<insert id="addUser" parameterType="com.ssxxz.pojo.User" >
insert into mybatis.user(id,name,pwd) values(#{id},#{name},#{pwd});
insert>
测试
//添加指定内容
@Test
public void addUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
userDao.addUser(new User(4,"嘻嘻","987654"));
sqlSession.commit();
sqlSession.close();
}
增删改,需要使用 .commit 提交事务,不然数据库会不更新内容
接口
//修改用户 update
int updateUser(User user);
SQL
<update id="updateUser" parameterType="com.ssxxz.pojo.User">
update mybatis.user set name=#{name},pwd=#{pwd} where id =#{id};
update>
测试
//修改指定内容
@Test
public void updateUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
userDao.updateUser(new User(4,"哈哈","654321"));
sqlSession.commit();
sqlSession.close();
}
接口
//删除一个用户
int deleteUser(int id);
SQL
<delete id="updateUser" parameterType="int">
delete from mybatis.user where id = #{id};
delete>
测试
//删除指定内容
@Test
public void deleteUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao mapper = sqlSession.getMapper(UserDao.class);
mapper.deleteUser(4);
sqlSession.commit();
sqlSession.close();
}
全部代码
接口
package com.ssxxz.dao;
//Dao等价于mapper,以后都会写mapper
import java.util.List;
import com.ssxxz.pojo.User;
public interface UserDao {
//查询全部的用户 select
List<User> getUserList();
//根据ID查询用户 select
User getUserById(int id);
//添加一个用户 insert
int addUser(User user);
//修改用户 update
int updateUser(User user);
//删除一个用户
int deleteUser(int id);
}
SQL
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.UserDao">
<select id="getUserList" resultType="com.ssxxz.pojo.User">
select * from mybatis.user;
select>
<select id="getUserById" parameterType="int" resultType="com.ssxxz.pojo.User">
select * from mybatis.user where id = #{id};
select>
<insert id="addUser" parameterType="com.ssxxz.pojo.User" >
insert into mybatis.user(id,name,pwd) values(#{id},#{name},#{pwd});
insert>
<update id="updateUser" parameterType="com.ssxxz.pojo.User">
update mybatis.user set name=#{name},pwd=#{pwd} where id =#{id};
update>
<delete id="updateUser" parameterType="int">
delete from mybatis.user where id = #{id};
delete>
mapper>
测试类
package com.ssxxz.dao;
import java.util.Iterator;
import java.util.List;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
import com.ssxxz.pojo.User;
import com.ssxxz.utils.MybatisUtils;
public class UserDaoTest {
//查看全部内容
@Test
public void test() {
//第一步:获得SqlSession对象
SqlSession sqlSession = MybatisUtils.getSqlSession();
//执行方式一:getMapper
//利用SqlSession对象内的getMapper方法,获得UserDao.class这个类
UserDao userDao = sqlSession.getMapper(UserDao.class);
//利用UserDao获得List方法
List<User> userList = userDao.getUserList();
for (User user : userList) {
System.out.println(user);
}
//关闭
sqlSession.close();
}
//查看指定内容
@Test
public void getUserById() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
User user = userDao.getUserById(1);
System.out.println(user);
sqlSession.close();
}
//添加指定内容
@Test
public void addUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
userDao.addUser(new User(4,"嘻嘻","987654"));
sqlSession.commit();
sqlSession.close();
}
//修改指定内容
@Test
public void updateUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
userDao.updateUser(new User(4,"哈哈","654321"));
sqlSession.commit();
sqlSession.close();
}
//删除指定内容
@Test
public void deleteUser() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserDao mapper = sqlSession.getMapper(UserDao.class);
mapper.deleteUser(4);
sqlSession.commit();
sqlSession.close();
}
}
假设,我们的实体类,或者数据库中的表,字段或者参数过多,我们应该考虑使用Map!
用万能Map插入用户
//用万能Map插入用户
int addUser2(Map<String,Object> map);
<insert id="addUser2" parameterType="com.ssxxz.pojo.User">
insert into user (id,name,pwd) values (#{userid},#{username},#{password})
insert>
@Test
public void addUser2(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
//获取map
HashMap<String, Object> map = new HashMap<String, Object>();
//赋值
map.put("userid",4);
map.put("username","呵呵");
map.put("password",147258);
mapper.addUser2(map);
//提交事务
sqlSession.commit();
//关闭资源
sqlSession.close();
}
同时只要是非空,还可以知插入id或者name,不需要填写全部内容
使用Map获得User查询
//用万能Map插入用户
User getUserById2(Map<String,Object> map);
<select id="getUserById2" parameterType="map" resultType="com.ssxxz.pojo.User">
select * from mybatis.user where id = #{id} adn name=#{name};
select>
@Test
public void getUserById2(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
//获取map
HashMap<String, Object> map = new HashMap<String, Object>();
//赋值
map.put("id",2);
// map.put("name","呵呵");
mapper.getUserById2(map);
//提交事务
sqlSession.commit();
//关闭资源
sqlSession.close();
}
Java代码执行的时候,传递通配符% %
第一种,在测试类,在输入的值中加入% %,写死
List<User> getUserLike(String value);
<select id="getUserLike" resultType="com.ssxxz.pojo.User">
select * from mybatis.user where name like #{value}
select>
@Test
public void getUserLike(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
List<User> userList = mapper.getUserLike("%李%");
for(Use user : userList){
System.out.println(user)
}
sqlSession.close();
}
第二种,在sql拼接中使用通配符,但是这个情况下会存在sql注入,应该防止这个情况
<select id="getUserLike" resultType="com.ssxxz.pojo.User">
select * from mybatis.user where name like "%"#{value}"%"
</select>
mybatis-config.xml
Mybatis的配置文件包含了会深深影响MyBatis行为的设置和属性信息。
MyBatis 可以配置成适应多种环境
不过要记住:尽管可以配置多个环境,但每个 SqlSessionFactory 实例只能选择一种环境
学会使用配置多套运行环境!
事物管理器
数据源
池:用完可以回收
<transactionManager type="JDBC"/>
<dataSource type="POOLED">
MyBatis默认的事务管理器就是JDBC ,连接池:POOLED(事务管理器还有一种MANAGED)
我们可以通过properties属性来实现引用配置文件
这些属性可以在外部进行配置,且可动态替换。你既可以在典型的 Java 属性文件中配置这些属性,也可以在 properties 元素的子元素中设置。【db.poperties】
properties必须放在第一位,xml中有固定的顺序
The content of element type "configuration" must match "(properties?,settings?,typeAliases?,typeHandlers?,objectFactory?,objectWrapperFactory?,reflectorFactory?,plugins?,environments?,databaseIdProvider?,mappers?)".
第一种,代码配置文件编写所有内容
编写一个外部配置文件db.properties,所有内容全部写在里面
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/mybatis?useSSL=true&;useUnicode&;characterEncoding=UTF-8
name=root
pwd=123456
核心文件,导入外部配置文件db.properties
DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<properties resource="db.properties"/>
<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="${name}"/>
<property name="password" value="${pwd}"/>
dataSource>
environment>
environments>
<mappers>
<mapper resource="com/ssxxz/dao/UserMapper.xml"/>
mappers>
configuration>
第二种,外部配置文件只写一部分,核心文件内写一部分
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/mybatis?useSSL=true&;useUnicode&;characterEncoding=UTF-8
DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<properties resource="db.properties">
<property name="username" value="root"/>
<property name="password" value="123456"/>
properties>
<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>
<mappers>
<mapper resource="com/ssxxz/dao/UserMapper.xml"/>
mappers>
configuration>
类型别名可为 Java 类型设置一个短名字,它只和 XML 配置有关。
存在的意义仅在于用来减少类完全限定名的冗余
typeAliases应该存于与第三个位置,有两种类型typeAlias
,package
typeAlias
<typeAliases>
<typeAlias type="com.ssxxz.pojo.User" alias="User"/>
typeAliases>
package
也可以指定一个包,MyBatist会在包名下面搜索需要的 Java Bean,在没有注解的情况下,会使用该包首字母小写的包名,比如扫描实体类的包,它的默认别名就为这个类的类名User,使用的就是首字母小写的包名user
<typeAliases>
<package name="com.ssxxz.pojo"/>
typeAliases>
若有注解@Alias,则别名为其注解值。见下面的例子:
@Alias("haha")
public class User {
...
}
这是 MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为。
MapperRegistry:注册绑定我们的Mapper文件
方式一:【推荐使用】
<mappers>
<mapper resource="com/ssxxz/dao/UserMapper.xml"/>
mappers>
方式二:使用class文件绑定注册
<mappers>
<mapper class="com.ssxxz.dao.UserMapper"/>
mappers>
class注意点:
方式三:使用包扫描进行注入
<mappers>
<package name="com.ssxxz.dao"/>
mappers>
注意点:
生命周期和作用域是至关重要的,因为错误的使用会导致非常严重的并发问题。
整个生命周期
SqlSessionFactoryBuilder
SqlSessionFactory
说白了就是可以想象为:数据库连接池
SqlSessionFactory一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建一个实例。
因此SqlSessionFactory的最佳作用域是应用作用域(ApplocationContext)。
最简单的就是使用单例模式或静态单例模式,保证全局只有一份变量。
SqlSession
作用域之间的关系
这里面的每一个Mapper,就代表一个具体的业务
新建一个项目,拷贝之前的,测试实体类字段不一致的情况,pwd/password
public class User {
private int id;
private String name;
private String password;
...
}
测试出现问题,password并没有输出内容
原因是类型处理器,自动填写数据库数据pwd,但是我们实际输入的是password,所以没有匹配上
// select * from user where id = #{id}
// 类型处理器,上面中的 * 其实是 id,name,pwd
// select id,name,pwd from user where id = #{id}
解决方法:
起别名pwd as password
<select id="getUserById" resultType="User">
select id,name,pwd as password from USER where id = #{id}
select>
resultMap
数据库 id name pwd / 实体类 id name password 想办法将pwd变成password
<resultMap id="UserMap" type="User">
<result column="id" property="id">result>
<result column="name" property="name">result>
<result column="pwd" property="password">result>
resultMap>
<select id="getUserById" resultMap="UserMap">
select* from USER where id = #{id}
select>
resultMap 元素是 MyBatis 中最重要最强大的元素。
ResultMap 的设计思想是,对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。
ResultMap 的优秀之处——你完全可以不用显式地配置它们。(不一样的转,一样的不用转,id、name可以不转)
如果这个世界总是这么简单就好了。
如果一个数据库操作,出现了异常,我们需要排错。但是在没有更好的工具情况下,日志就是最好的助手!
曾经:sout、debug
现在:日志工厂
存在于
中
在MyBatis中具体使用哪一个日志实现,在设置中设定
STDOUT_LOGGING 标准日志输出
在mybatis核心配置文件mybatis-config.xml中,配置我们的日志!默认第二位
<settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
settings>
注意细节:大小写,空格标点符号,任何错误都有可能报错
工作中经常会用到Log4j
Log4j是Apache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台、文件、GUI组件,甚至是套接口服务器等;
我们也可以控制每一条日志的输出格式;
通过定义每一条日志信息的级别,我们能够更加细致地控制日志的生成过程;
可以通过一个配置文件来灵活地进行配置,而不需要修改应用的代码。
1、先导入log4j的依赖包
<dependency>
<groupId>log4jgroupId>
<artifactId>log4jartifactId>
<version>1.2.17version>
dependency>
2、创建log4j.properties
#将等级为DEBUG的日志信息输出到console和file这两个目的地,console和file的定义在下面的代码
log4j.rootLogger=DEBUG,console,file
#控制台输出的相关设置
#控制台输出由ConsoleAppender
#日志实现结果System.out
#日志实现的级别DEBUG
#日志格式[%c]-%m%n
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
#文件输出的相关设置
#File输出保存日志的地址
log4j.appender.file = org.apache.log4j.RollingFileAppender
log4j.appender.file.File=./log/rzp.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}][%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.sq1.PreparedStatement=DEBUG
学会分析其中内容,具体设置内容从网上学习
3、配置settings为log4j实现
<settings>
<setting name="logImpl" value="log4j"/>
settings>
4、Log4j的使用,直接测试运行刚才的查询
配置后内容大致相同,就是多了些前缀
1、在要使用Log4j的类中,导入包 import org.apache.log4j.Logger; (注意导入apache的包)
2、日志对象,参数为当前类的class对象
Logger logger = Logger.getLogger(UserDaoTest.class);
3、日志级别
@Tset
public void testLog4j{
logger.info("info: 进入了log4j"); //信息,普通提示信息
logger.debug("debug: 进入了log4j"); //调试语句
logger.error("error:进入了log4j"); //错误,trycatch语句
}
1.info
2.debug
3.erro
思考:为什么分页?
减少当前数据的处理量
语法:SELECT * from user limit startIndex,pageSize;
SELECT * from user limit 3; #[0,n]
使用MyBatis实现分页,核心SQL
1.接口
//分页
List<User> getUserByLimit(Map<String,Integer> map);
2.Mapper.xml
<select id="getUserByLimit" parameterType="map" resultMap="UserMap">
select * from user limit #{startIndex},#{pageSize}
select>
3.测试
@Test
public void getUserByLimit(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
HashMap<String, Integer> map = new HashMap<String, Integer>();
map.put("startIndex",1);
map.put("pageSize",2);
List<User> list = mapper.getUserByLimit(map);
for (User user : list) {
System.out.println(user);
}
}
不再使用SQL实现分页
1.接口
//分页2
List<User> getUserByRowBounds();
2.mapper.xml
<!--分页查询2-->
<select id="getUserByRowBounds">
select * from user limit #{startIndex},#{pageSize}
</select>
3.测试
public void getUserByRowBounds(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
//RowBounds实现分页
RowBounds rowBounds = new RowBounds(1, 2);
//通过Java代码层面实现分页
List<User> userList = sqlSession.selectList("com.kaung.dao.UserMapper.getUserByRowBounds", null, rowBounds);
for (User user : userList) {
System.out.println(user);
}
sqlSession.close();
}
mybatis分页插件
注:不管使用哪种分页它的底层都是limi
了解即可,万一以后公司的架构师,说要使用,你要指定它是什么东西
解耦,可拓展,提高复用,分层开发中,上层不用管具体的实现,大家都遵守共同的标准,使得开发变得容易,规范性更好
1、注解在接口上实现,org.apache.ibatis.annotations包
@Select("select * from user")
List<User> getUsers();
2、需要在核心配置文件中绑定接口
<mappers>
<mapper class="com.ssxxz.dao.UserMapper"/>
mappers>
3、测试结果,id和name可以显示,但是pwd无法显示。
使用注解来映射简单语句会使代码显得更加简洁,然而对于稍微复杂一点的语句,Java 注解就力不从心了,并且会显得更加混乱。因此,如果你需要完成很复杂的事情,那么最好使用 XML来映射语句。
本质:反射机制实现
底层:动态代理
结婚(目标对象),婚庆公司(代理对象),你(真实对象)
MyBatis详细执行流程
我们可以在工具类创建的时候不需要 .commit,可以实现自动提交事务!
MybatisUtils.java,openSession(true);为true自动提交事务
public static SqlSession getSqlSession(){
return sqlSessionFactory.openSession(true);
}
条件查询
//方法存在多个参数,所有的参数前面必须加上@Param("id")注解
@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 = #{uid}")
int deleteUser(@Param("uid") int id);
注意:
【我们必须要将接口注册绑定到我们的核心配置文件中!】
Lombok项目是一个Java库,它会自动插入编辑器和构建工具中,Lombok提供了一组有用的注释,用来消除Java类中的大量样板代码。仅五个字符(@Data)就可以替换数百行代码从而产生干净,简洁且易于维护的Java类。
使用步骤:
1、在IDEA中安装Lombok插件
2、在项目中导入lombok的jar包
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.10version>
dependency>
3、在程序上加注解
@Getter and @Setter
@FieldNameConstants
@ToString
@EqualsAndHashCode
@AllArgsConstructor, @RequiredArgsConstructor and @NoArgsConstructor
@Log, @Log4j, @Log4j2, @Slf4j, @XSlf4j, @CommonsLog, @JBossLog, @Flogger, @CustomLog
@Data
@Builder
@SuperBuilder
@Singular
@Delegate
@Value
@Accessors
@Wither
@With
@SneakyThrows
@val
使用说明:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private int id;
private String name;
private String password;
}
多对一
alter table student ADD CONSTRAINT fk_tid foreign key (tid) references teacher(id)
导入lombo和Utils
pom.xml
<project xmlns="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.0modelVersion>
<parent>
<groupId>com.ssxxzgroupId>
<artifactId>Myabtis-StudyartifactId>
<version>0.0.1-SNAPSHOTversion>
parent>
<groupId>com.ssxxz.mavengroupId>
<artifactId>mybatis-06artifactId>
<dependencies>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.10version>
dependency>
dependencies>
project>
MybatisUtils.java
package com.ssxxz.utils;
import java.io.IOException;
import java.io.InputStream;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
public class MybatisUtils {
private static SqlSessionFactory sqlSessionFactory;
static {
try {
//使用Mybatis第一步 :获取sqlSessionFactory对象
String resource = "org/mybatis/example/mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
//既然有了 SqlSessionFactory,顾名思义,我们可以从中获得 SqlSession 的实例.
public static SqlSession getSqlSession() {
return sqlSessionFactory.openSession();
}
}
新建实体类Teacher,Student
Teacher.java
package com.ssxxz.pojo;
public class Teacher {
private int id;
private String name;
}
Student.java
package com.ssxxz.pojo;
import lombok.Data;
@Data
public class Student {
private int id;
private String name;
//学生需要关联一个老师,但是老师不确定,所以不能写死,引入一个老师对象
private Teacher teacher ;
}
建立Teacher,Student的Mapper接口
TeacherMapper.java
package com.ssxxz.dao;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import com.ssxxz.pojo.Teacher;
public interface TeacherMapper {
@Select("select * from teacher where id = #{tid}")
Teacher getTeacher(@Param("tid") int id);
}
StudentMapper.java
package com.ssxxz.dao;
public interface StudentMapper {
}
建立Mapper.xml文件
TeacherMapper.xml
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.TeacherMapper">
mapper>
StudentMapper.xml
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.StudentMapper">
mapper>
在核心配置文件中绑定注册我们的Mapper接口或者文件 【方式很多,随心选】
mybatis-config.xml
DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<properties resource="db.propenrties"/>
<settings>
<setting name="iogImpi" value="STDOUT_LOGGING"/>
settings>
<typeHandlers>
<package name="com.ssxxz.pojo"/>
typeHandlers>
<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="${pwd}"/>
dataSource>
environment>
environments>
<mappers>
<mapper class="com.ssxxz.dao.TeacherMapper"/>
<mapper class="com.ssxxz.dao.StudentMapper"/>
mappers>
configuration>
db.propenrties
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/mybatis?useSSL=true&;useUnicode&;characterEncoding=UTF-8
username=root
pwd=123456
测试查询是否能够成功
MyTest.java
package com.ssxxz;
import org.apache.ibatis.session.SqlSession;
import com.ssxxz.dao.TeacherMapper;
import com.ssxxz.pojo.Teacher;
import com.ssxxz.utils.MybatisUtils;
public class MyTest {
public static void main(String[] args) {
SqlSession sqlSession = MybatisUtils.getSqlSession();
TeacherMapper mapper = sqlSession.getMapper(TeacherMapper.class);
Teacher teacher = mapper.getTeacher(1);
System.out.println(teacher);
sqlSession.close();
}
}
如果想获得学生同时获得老师
sql语句:select s.id,s.name,t.name from student s,teacher t where s.tid = t.id
select 查找,显示:s.id,s.name,t.name,由表格:student s表,teacher t表 where且 s.tid = t.id
但是方法怎么写?
先写StudentMapper.java接口
package com.ssxxz.dao;
public interface StudentMapper {
public List<Student> getStudent();
}
StudentMapper.xml 配置文件
<select id="getStudent" resultMap="StudentTeacher">
select * from student
select>
<resultMap id="StudentTeacher" type="student">
<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 = #{id}
select>
MyTest.java
package com.ssxxz;
import org.apache.ibatis.session.SqlSession;
import org.util.List;
import com.ssxxz.dao.TeacherMapper;
import com.ssxxz.pojo.Teacher;
import com.ssxxz.utils.MybatisUtils;
import com.ssxxz.dao.StudentMapper;
import com.ssxxz.pojo.Student;
public class MyTest {
public static void main(String[] args) {
SqlSession sqlSession = MybatisUtils.getSqlSession();
TeacherMapper mapper = sqlSession.getMapper(StudentMapper.class);
List<Student> studentList = mapper.getStudent();
for(Student student : studentList){
System.out.println(student);
}
sqlSession.close();
}
}
<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 id="StudentTeacher2" type="student">
<result property="id" column="sid"/>
<result property="name" column="sname"/>
<association property="teacher" javaType="Teacher">
<result property="name" column="tname">result>
association>
resultMap>
回顾Mysql多对一查询方式:
复制上面的代码
Studenmapper.java/.xml
MybatisUtils
db.propenrties
mybatis-config.xml
修改的实体类
Student.java
@Data
public class Student {
private int id;
private String name;
private int tid;
}
Teacher.java
@Data
public class Teacher {
private int id;
private String name;
//一个老师拥有多个学生
private List<Student> students;
}
修改的Mapper
TeacherMapper.java
package com.ssxxz.dao;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import com.ssxxz.pojo.Teacher;
public interface TeacherMapper {
List<Teacher> getTeacher();
}
Teachermapper.xml
DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.TeacherMapper">
<select id="getTeacher" resultYype="Teacher">
select * from mybatis.teacher;
select>
mapper>
测试
public class MyTest {
@Test
public void test(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
for(Student teacher : sqlSession.getMapper(TeacherMapper.class).getTeacher()){
System.out.println(teacher);
}
sqlSession.close();
}
}
测试成功查询出老师,但是没有学生内容
TeacherMapper.java
public interface TeacherMapper {
//获取指定老师下的所有学生及老师的信息
Teacher getTeacher(@Param("tid" int id));
}
sql语句
select s.id sid, s.name sname, t.name tname, t.id tid from student s,teacher t where s.tid = t.id
<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 tid = #{tid}
select>
<resultMap id="TeacherStudent" type="Teacher">
<result property="id" column="tid"/>
<result property="name" column="tname"/>
<collection property="students" ofType="Student">
<result property="id" column="sid"/>
<result property="name" column="sname"/>
<result property="tid" column="tid"/>
collection>
resultMap>
<select id="getTeacher2" resultMap="TeacherStudent2">
select * from mybatis.teacher where id = #{tid}
select>
<resultMap id="TeacherStudent2" type="Teacher">
<collection property="students" javaType="ArrayList" ofType="Student" select="getStudentByTeacherId" column="id"/>
resultMap>
<select id="getStudentByTeacherId" resultType="Student">
select * from mybatis.student where tid = #{id}
select>
理解:
注意点:
保证SQL的可读性,尽量保证通俗易懂
注意一对多和多对一,属性名和字段的问题
如果问题不好排查错误,可以使用日志,建议使用Log4j
避免慢SQL,当数据量多的时候,需要使用索引,不然你的代码会比别人慢很多
面试高频
什么是动态SQL:动态SQL就是根据不同的条件生成不同的SQL语句
所谓的动态SQL,本质上还是SQL语句,只是我们可以在SQL层面,去执行一个逻辑代码
如果你之前用过 JSTL 或任何基于类 XML 语言的文本处理器,你对动态 SQL 元素可能会感觉似曾相识。在 MyBatis 之前的版本中,需要花时间了解大量的元素。借助功能强大的基于 OGNL 的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类,现在要学习的元素种类比原来的一半还要少。
vCREATE TABLE `mybatis`.`blog` (
`id` int(10) NOT NULL AUTO_INCREMENT COMMENT '博客id',
`title` varchar(30) NOT NULL COMMENT '博客标题',
`author` varchar(30) NOT NULL COMMENT '博客作者',
`create_time` datetime(0) NOT NULL COMMENT '创建时间',
`views` int(30) NOT NULL COMMENT '浏览量',
PRIMARY KEY (`id`)
)
创建一个基础工程
1、导包
pom.xml
<project xmlns="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.0modelVersion>
<parent>
<groupId>com.ssxxzgroupId>
<artifactId>Myabtis-StudyartifactId>
<version>0.0.1-SNAPSHOTversion>
parent>
<artifactId>mybatis-08artifactId>
<dependencies>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.10version>
dependency>
dependencies>
project>
2、编写配置文件resources
mybatis-config.xml
DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<properties resource="db.propenrties"/>
<settings>
<setting name="iogImpi" value="STDOUT_LOGGING"/>
<setting name="mapUnderscoreToCamelCase" value="true"/>
settings>
<typeHandlers>
<package name="com.ssxxz.pojo"/>
typeHandlers>
<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="${pwd}"/>
dataSource>
environment>
environments>
<mappers>
<mapper class="com.ssxxz.dao.BlogMapper"/>
mappers>
configuration>
db.propenrties
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/mybatis?useSSL=true&;useUnicode&;characterEncoding=UTF-8
username=root
pwd=123456
3、编写实体类
pojo
@Data
public class Blog {
private String id;
private String title;
private String author;
private Date createTime;// 属性名和数据空字段名create_time不一致
private int views;
}
核心文件配置驼峰命名转换
<setting name="mapUnderscoreToCamelCase" value="true"/>
4、编写实体类对应 Mapper接口 和 Mapper.xml 文件
dao
BlogManpper.java
public interface BlogManpper{
//插入数据
int addBook(Blog blog);
}
BlogManpper.xml
xxxxxxxxxx
DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ssxxz.dao.BlogManpper">
<insert id="addBlog" parameterType="blog">
insert into mybatis.blog(id,title,author,create_time,views)
values (#{id},#{title},#{author},#{createTime},#{views})
insert>
mapper>
utils
IDutils.java 获得随机id
public void IDutils(){
public static String getId(){
//将随机生成的id中的-符号,换成“空”
return UUID.randomUUID().toString().replaceAll("-","")
}
}
(在企业中,id一般都是随机数保证唯一性)
MybatisUtils.java
package com.ssxxz.utils;
import java.io.IOException;
import java.io.InputStream;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
public class MybatisUtils {
private static SqlSessionFactory sqlSessionFactory;
static {
try {
//使用Mybatis第一步 :获取sqlSessionFactory对象
String resource = "org/mybatis/example/mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
//既然有了 SqlSessionFactory,顾名思义,我们可以从中获得 SqlSession 的实例.
public static SqlSession getSqlSession() {
return sqlSessionFactory.openSession();
}
}
4、测试
MyTest
publice class MyTest(){
@Test
public void addInitBlog(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
Blog blog = new Blog();
blog.setId(IDutils.getId());
blog.setTitle("Mybatis如此简单");
blog.setAuthor("狂神说");
blog.setCreateTime(new Date());
blog.setViews(9999);
mapper.addBlog(blog);
blog.setID(IDutils.getId());
blog.setTitle("Spring如此简单");
mapper.addBlog(blog);
blog.setID(IDutils.getId());
blog.setTitle("微服务如此简单");
mapper.addBlog(blog);
sqlSession.close();
}
}
BlogManpper.java 增加一个查询的接口
public interface BlogManpper{
//插入数据
int addBook(Blog blog);
//查询博客
List<Blog> queryBlogIF(Map map);
}
BlogManpper.xml if判断
如果map 内"title!=null"执行下面,"author!=null"执行下面
<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>
测试
publice class MyTest(){
@Test
public void addInitBlog(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
HashMap map = new HashMap();
map.put("title","Java如此简单");
map.put("author","狂神说");
List<Blog> blogs = mapper.queryBlogIf(map);
for(Blog blog : blogs){
System.out.println(blog);
}
sqlSession.close();
}
}
when,只要满足一个判断就结束,如果都不满足执行otherwise
<select id="queryBlogChoose" parameterType="map" resultType="blog">
select * from mybatis.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>
都不成立,最少也要输入views
where
where 元素只会在子元素返回任何内容的情况下才插入 “WHERE” 子句。而且,若子句的开头为 “AND” 或 “OR”,where 元素也会将它们去除。
比如代码为:
<select id="queryBlogChoose" parameterType="map" resultType="blog">
select * from mybatis.blog wher
<where>
<if test="">
id = #{id}
if>
<if test="">
and name = #{name}
if>
where>
select>
但这个语法不正确,如果当第一条不成立,就会出现下面情况
所以需要配合where标签使用
<select id="queryBlogChoose" parameterType="map" resultType="blog">
select * from mybatis.blog
<where>
<if test="title!=null">
and title = #{title}
if>
<if test="author!=null">
and author = #{author}
if>
where>
select>
set
set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号(这些逗号是在使用条件语句给列赋值时引入的)
<update id="updateBlog" parameterType="map">
update mybatis.blog
<set>
<if test="title!=null">
title = #{title},
if>
<if test="author!=null">
and author = #{author}
if>
set>
where id =#{id}
update>
trim
<trom prefix="" prefix="" prefixOverrides="" suffix="" suffixOverrides="">trom>
所谓的动态SQL,本质还是SQL语句,只是我们可以在SQL层面,去执行一个逻辑代码
有的时候,我们可能会将一些功能的部分抽取出来,方便服用!
1、使用SQL标签抽取公共部分可
<sql id="if-title-author">
<if test="title!=null">
title = #{title}
if>
<if test="author!=null">
and author = #{author}
if>
sql>
2、在需要使用的地方使用Include标签引用即可
<select id="queryBlogIF" parameterType="map" resultType="blog">
select * from blog
<where>
<include refid="if-title-author">include>
where>
select>
注意事项:
最好基于单标来定义SQL片段,简单些不要有太多的链表查询
不要存在where标签,只需要简单if判断
比如拼接该代码( id=1 or id=2 or id=3)
select * from user where 1=1 and
<foreach item="id" collection="ids"
open="(" separtor="or" close=")">
#{id}
foreach>
collection集合,item遍历的名字,open开头,separtor分隔符,close结尾
例题:查询第1-2-3号记录的博客
接口类
List<Blog> queryBlogForeach(Map map);
sql
<select id="queryBlogForeach" parameterType="map" resultType="blog">
select * form mybatis.blog
<where>
<foreach collection="ids" item="id" open="and(" close=")" separator="or"> id = #{id}
foreach>
where>
select>
Test
@Test
public void queryBlogForEach(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
//创建一map往方法里传
HashMap map = new HashMap();
//创建一个集合往sql里传
ArrayList<Integer> ids = new ArrayList<Integer>();
ids.add(1);
ids.add(2);
ids.add(3);
map.put("ids",ids);
List<Blog> blogs = mapper.queryBlogForeach(map);
for(Blog blog : blogs){
System.out.println(blog);
}
sqlSession.close();
}
结果
动态SQL就是在拼接SQL语句,我们只要保证SQL的正确性,按照SQL的格式,去排列组合就可以了
建议:
先在Mysql中写出完整的SQL,再对应的去修改成我们的动态SQL实现通用即可
查询 →需要连接数据库→就很耗资源!
一次查询的结果,给他暂存一个可以直接取到的地方 --> 内存:缓存
我们再次查询的相同数据的时候,直接走缓存,不走数据库了
中间的memcached就是缓存
1、什么是缓存[Cache]?
2、为什么使用缓存?
3、什么样的数据可以使用缓存?
可用的清除策略有:
LRU
– 最近最少使用:移除最长时间不被使用的对象。(默认LRU)FIFO
– 先进先出:按对象进入缓存的顺序来移除它们。SOFT
– 软引用:基于垃圾回收器状态和软引用规则移除对象。WEAK
– 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。测试步骤:
1、开启日志,查看运行情况
2、测试在一个Session中查询两次记录
@Test
public void test1() {
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.getUserById(1);
System.out.println(user);
System.out.println("=====================================");
User user2 = mapper.getUserById(1);
System.out.println(user2 == user);
}
3、查看日志输出
查询两个,但是sql只运行了一次,说明第二次查询是从缓存里查询的
缓存失效的情况:
查询不同的东西
增删改操作,可能会改变原来的数据,所以必定会刷新缓存
两个id1查询中间,添加了修改id2的程序,缓存就发生了变化,id1==id1(false)
查询不同的Mapper.xml
手动清理缓存
sqlSession.clearCache();
小结:一级缓存默认是开启的,只在一次SqlSession中有效,也就是拿到连接到关闭连接这个区间段!
高级方法
flushInterval(刷新间隔)属性可以被设置为任意的正整数,设置的值应该是一个以毫秒为单位的合理时间量。 默认情况是不设置,也就是没有刷新间隔,缓存仅仅会在调用语句时刷新。
size(引用数目)属性可以被设置为任意正整数,要注意欲缓存对象的大小和运行环境中可用的内存资源。默认值是 1024。
readOnly(只读)属性可以被设置为 true 或 false。只读的缓存会给所有调用者返回缓存对象的相同实例。 因此这些对象不能被修改。这就提供了可观的性能提升。而可读写的缓存会(通过序列化)返回缓存对象的拷贝。 速度上会慢一些,但是更安全,因此默认值是 false。
步骤:
1、开启全局缓存
<setting name="cacheEnabled" value="true"/>
2、在Mapper.xml中自定义使用缓存
<cache
eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
同时SQL可以单独手动控制全局缓存的开关
<select id="queryUserById" resultType="user" useCache="false">
select * form user where id =#{id}
select>
3、测试
问题:我们需要将实体类序列化,否则就会报错
User.java
public class User implements Serializable{
private int id;
private String name;
private String pwd;
}
小结:
序列化:
缓存顺序:
注意:
只有查询才有缓存,根据数据是否需要缓存(修改是否频繁选择是否开启)useCache=“true”,调优的时候可以用到
<select id="getUserById" resultType="user" useCache="true">
select * from user where id = #{id}
select>
缓存的意义还是提高效率
Ehcache是一种广泛使用的开源Java分布式缓存,主要面向通用缓存
1、导包
<dependency>
<groupId>org.mybatis.cachesgroupId>
<artifactId>mybatis-ehcacheartifactId>
<version>1.2.1version>
dependency>
2、在mapper中指定使用我们的ehcache缓存实现
<cache type="org.mybatis.caches.ehcache.EhcacheCache"/>
3、它可以自定义配置,定义缓存策越
ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
updateCheck="false">
<diskStore path="java.io.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>
实现Cache接口,可以自定义策略但需要很多数据库底层相关知识
现阶段都在使用Redis数据库来做缓存,使用K-V键值对来实现。