学习前需要掌握:
持久化是将程序数据在持久状态和瞬时状态间转换的机制。
为什么需要持久化服务呢?那是由于内存本身的缺陷引起的
什么是持久层?
Mybatis就是帮助程序猿将数据存入数据库中 , 和从数据库中取数据 .
传统的jdbc操作 , 有很多重复代码块 .比如 : 数据取出时的封装 , 数据库的建立连接等等… , 通过框架可以减少重复代码,提高开发效率 .
MyBatis 是一个半自动化的ORM框架 (Object Relationship Mapping) -->对象关系映射
所有的事情,不用Mybatis依旧可以做到,只是用了它,所有实现会更加简单!技术没有高低之分,只有使用这个技术的人有高低之别
MyBatis的优点
先将父工程的src目录删除
导入我们需要的资源
<dependencies>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>5.1.47version>
dependency>
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.5.2version>
dependency>
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.12version>
dependency>
dependencies>
说明:导入依赖在父工程中导入,这样的话我们每创建子工程就不用在导入依赖
可能出现问题说明:Maven静态资源过滤问题
解决方法如下,在pom.xml文件中添加
<build>
<resources>
<resource>
<directory>src/main/javadirectory>
<includes>
<include>**/*.propertiesinclude>
<include>**/*.xmlinclude>
includes>
<filtering>falsefiltering>
resource>
<resource>
<directory>src/main/resourcesdirectory>
<includes>
<include>**/*.propertiesinclude>
<include>**/*.xmlinclude>
includes>
<filtering>falsefiltering>
resource>
resources>
build>
在模块中的资源文件夹中创建mybatis.config.xml文件
<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/db1?mybatis?UseSSL=true&userUnicode=true&charcterEncoding=UTF-8"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
dataSource>
environment>
environments>
<mappers>
<mapper resource="org/mybatis/example/BlogMapper.xml"/>
mappers>
configuration>
说明:可以同时存在多套环境,上面的是默认的环境
修改mapper地址
<mappers>
<mapper resource="com/xiaozhi/mapper/userMapper.xml"/>
mappers>
说明:这个一定要是你自己的mapper配置文件所在的路径,不然会报以下错误
/**
* @author xiaozhi
* @description 获得SqlSession的工具类
* @create 2020-12-2020/12/30 11:07
*/
public class MybatisUtils {
private static SqlSessionFactory sqlSessionFactory;
static {
try {
String resource = "mybatis-config.xml";
// 创建一个流读取文件
InputStream inputStream = Resources.getResourceAsStream(resource);
// 获取到sqlSessionFactory工厂类
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
}
// 获取sqlSession连接
public static SqlSession getSession() {
return sqlSessionFactory.openSession();
}
}
public class User {
private Integer id;
private String name;
private String password;
private String email;
说明:这里我之前就在数据库中创建好的一个表
public interface UserMapper {
// 获取所有的用户信息
public List<User> selectUser();
}
要创建一个Mapper.xml文件
<mapper namespace="com.xiaozhi.mapper.UserMapper">
<select id="selectUser" resultType="com.xiaozhi.pojo.User">
/*标签内写sql语句*/
select * from user
select>
mapper>
补充:这里我们是查询操作,所以是select标签
public class UserMapperTest {
//第一种方式
@Test
public void Test1(){
// 1获取到sqlSession对象
SqlSession session = MybatisUtils.getSession();
// 2得到mapper接口
UserMapper mapper = session.getMapper(UserMapper.class);
// 3调用方法
List<User> users = mapper.selectUser();
for (User user:users) {
System.out.println(user);
}
// 关闭资源
session.close();
}
// 第二种方式
@Test
public void Test2(){
// 1获取到sqlSession对象
SqlSession session = MybatisUtils.getSession();
List<User> users = session.selectList("com.xiaozhi.mapper.UserMapper.selectUser");
for (User user:users) {
System.out.println(user);
}
}
}
推荐使用第一种方式,结构清晰
IDEA连接数据库出现的问题
解决
通过show variables like’%time_zone’; 查看当前的时区设置
在mysq命令行模式下输入 set global time_zone=’+8:00’; 修改时区,然后重启就可以了
基本步骤
// 根据id查询用户
User selectUserById(Integer id);
<select id="selectUserById" resultType="com.xiaozhi.pojo.User">
select * from user where id=#{id}
select>
@Test
public void Test3(){
// c测试使用id来查询用户
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class); // 得到接口
User user = mapper.selectUserById(1);
System.out.println(user);
session.close();
}
// 插入用户信息
int insertUser(User user);
<insert id="insertUser" parameterType="com.xiaozhi.pojo.User">
insert into user values (#{id},#{username},#{password},#{email});
insert>
// 修改用户信息
int updateUser(User user);
<update id="updateUser" parameterType="com.xiaozhi.pojo.User">
update user set username=#{username},password=#{password},email=#{email} where id = #{id}
update>
// 删除用户信息
int deleteUser(int id);
<delete id="deleteUser" parameterType="com.xiaozhi.pojo.User">
delete from user where id = #{id}
delete>
注意:增删改操作一定要提交事物
情况:我们需要修改密码,这个时候只需要传id和password参数就可以了,我们以往的做法是将整个对象放入,然后调用它的属性来进行修改,在传少个参数的情况下是可以通过map来进行赋值的
注意
示例
写好方法,参数是map
// 修改用户密码
int setPwd(Map<String,Object> map);
参数类型是map的,然后通过key来取值
<update id="setPwd" parameterType="map">
update user set password = #{pwd} where id = #{id};
update>
第一种方式
通过注释的方式
// 根据用户名和密码来查询用户 - 第一种方式
User selectUserByNamePaw(@Param("username") String username,@Param("password") String password);
<select id="selectUserByNamePaw" resultType="com.xiaozhi.pojo.User">
select * from user where username=#{username} and password=#{password}
select>
@Test
public void Test4(){
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
User xiaozhi = mapper.selectUserByNamePaw("xiaozhi", "123456");
System.out.println(xiaozhi);
session.close();
}
第二种方式
万能的map:传一个map集合的参数,里面放的是我们的用户名和密码,在xml文件中我们在通过map的key值来输出到sql语句中
// 第二种方式
User selectUserByNamePaw2(Map<String,Object> map);
<select id="selectUserByNamePaw2" parameterType="map" resultType="com.xiaozhi.pojo.User">
select * from user where username=#{username} and password=#{password}
select>
// 第二种方式
@Test
public void Test5(){
Map<String, Object> map = new HashMap<String, Object>();
map.put("username","xiaozhi");
map.put("password","123456");
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
User user = mapper.selectUserByNamePaw2(map);
System.out.println(user);
session.close();
}
在jdbc中我们用?代替我们的参数值,假设我们要做一个根据id来查询用户信息的操作,这个?也就是id值,假设我们客户传的是这样的值"3 and id = 1",那么我们的sql语句就会变成是这样的:
select * from user where id = 3 and id = 1;很明显进行了一个字符串拼接,这样就造成了我们的sql语句出现问题,为了避免这种情况,我们要将这个值给写死,用类型或者是其他的方式
示例 - 查询有a字母的用户
<!--模糊查询-->
List<User> selectUserByA(String value);
<select id="selectUserByA" resultType="com.xiaozhi.pojo.User">
select * from user where username like #{value}
select>
说明:上面的方式会导致SQL注入的问题
<select id="selectUserByA" resultType="com.xiaozhi.pojo.User">
select * from user where username like "%"#{value}"%"
select>
所以,我们可以将sql语句写死
mybatis-config.xml 系统核心配置文件
MyBatis 的配置文件包含了会深深影响 MyBatis 行为的设置和属性信息。
下面是可以进行配置的:
configuration(配置)
- properties(属性
- settings(设置)
- typeAliases(类型别名)
- typeHandlers(类型处理器)
- objectFactory(对象工厂)
- plugins(插件
- environments(环境配置)
- environment(环境变量)
- transactionManager(事务管理器)
- dataSource(数据源)
- databaseIdProvider(数据库厂商标识)
- mappers(映射器)
说明:注意标签的位置,顺序不对是会报错的,一定要按人家的规范来
我们可以阅读 mybatis-config.xml 上面的dtd的头文件!
这些属性可以在外部进行配置,并可以进行动态替换。你既可以在典型的 Java 属性文件中配置这些属性,也可以在 properties 元素的子元素中设置。
第一步 :在资源目录下新建一个db.properties
driver=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/db1?useSSL=true&useUnicode=true&characterEncoding=utf8
username=root
password=root
第二步 : 将文件导入properties 配置文件
<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="${username}"/>
<property name="password" value="${password}"/>
dataSource>
environment>
environments>
<mappers>
<mapper resource="com/xiaozhi/dao/UserMapper.xml"/>
mappers>
configuration>
properties元素中的resource是数据源,导入的方式有两种:
①直接引入
②在properties元素的子元素property中设置值,有name和value属性,value属性指向的是文件中的名字,name属性是用来取值的
<properties resource="db.properties">
<property name="driver" value="driver"/>
<property name="url" value="url"/>
<property name="username" value="root"/>
<property name="password" value="password"/>
properties>
可以配置多套运行环境
<environments default="development">
<environment id="development">
<transactionManager type="JDBC">
<property name="..." value="..."/>
transactionManager>
<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>
<transactionManager type="[ JDBC | MANAGED ]"/>
这两种事务管理器类型都不需要设置任何属性。
具体的一套环境,通过设置id进行区别,id保证唯一!
file:///
的 URL),或类名和包名等。映射器是MyBatis中最核心的组件之一,在MyBatis 3之前,只支持xml映射器,即:所有的SQL语句都必须在xml文件中配置。而从MyBatis 3开始,还支持接口映射器,这种映射器方式允许以Java代码的方式注解定义SQL语句,非常简洁。引入资源方式
<mappers>
<mapper resource="org/mybatis/builder/PostMapper.xml"/>
mappers>
<mappers>
<mapper url="file:///var/mappers/AuthorMapper.xml"/>
mappers>
<mappers>
<mapper class="org.mybatis.builder.AuthorMapper"/>
mappers>
<mappers>
<package name="org.mybatis.builder"/>
mappers>
Mapper文件
<mapper namespace="com.kuang.mapper.UserMapper">
mapper>
namespace中文意思:命名空间,作用如下:
说明:有类别名之后我们就不用在写类名了,写我们设定好的名字就可以找到指定的类
注意:注册的地方不能使用这种方式
类型别名可为 Java 类型设置一个缩写名字。 它仅用于 XML 配置,意在降低冗余的全限定类名书写。例如:
<typeAliases>
<typeAlias alias="Author" type="domain.blog.Author"/>
<typeAlias alias="Blog" type="domain.blog.Blog"/>
<typeAlias alias="Comment" type="domain.blog.Comment"/>
<typeAlias alias="Post" type="domain.blog.Post"/>
<typeAlias alias="Section" type="domain.blog.Section"/>
<typeAlias alias="Tag" type="domain.blog.Tag"/>
typeAliases>
当这样配置时,Blog
可以用在任何使用 domain.blog.Blog
的地方。
代码实现
mybatis-config.xml中添加
<typeAliases>
<typeAlias alias="User" type="com.xiaozhi.pojo.User"/>
typeAliases>
修改UserMapper.xml
<select id="selectUser" resultType="User">
select * from user
select>
也可以指定一个包名,MyBatis 会在包名下面搜索需要的 Java Bean,比如:
<typeAliases>
<package name="domain.blog"/>
typeAliases>
没有注解的情况:会使用 Bean 的首字母小写的非限定类名来作为它的别名。 比如 domain.blog.Author
的别名为 author
;
<typeAliases>
<package name="com.xiaozhi"/>
typeAliases>
<select id="selectUser" resultType="user">
select * from user
select>
有注解的情况:则别名为其注解值。见下面的例子:
@Alias("author")
public class Author {
...
}
代码实现
@Alias("xiaozhi")
public class User {
<select id="selectUser" resultType="xiaozhi">
select * from user
select>
下面是一些为常见的 Java 类型内建的类型别名。它们都是不区分大小写的,注意,为了应对原始类型的命名重复,采取了特殊的命名风格。
别名 | 映射的类型 |
---|---|
_byte | byte |
_long | long |
_short | short |
_int | int |
_integer | int |
_double | double |
_float | float |
_boolean | boolean |
string | String |
byte | Byte |
long | Long |
short | Short |
int | Integer |
integer | Integer |
double | Double |
float | Float |
boolean | Boolean |
date | Date |
decimal | BigDecimal |
bigdecimal | BigDecimal |
object | Object |
map | Map |
hashmap | HashMap |
list | List |
arraylist | ArrayList |
collection | Collection |
iterator | Iterator |
这是 MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为。
目前需要知道的三个:
一个配置完整的设置如下
<settings>
<setting name="cacheEnabled" value="true"/>
<setting name="lazyLoadingEnabled" value="true"/>
<setting name="multipleResultSetsEnabled" value="true"/>
<setting name="useColumnLabel" value="true"/>
<setting name="useGeneratedKeys" value="false"/>
<setting name="autoMappingBehavior" value="PARTIAL"/>
<setting name="autoMappingUnknownColumnBehavior" value="WARNING"/>
<setting name="defaultExecutorType" value="SIMPLE"/>
<setting name="defaultStatementTimeout" value="25"/>
<setting name="defaultFetchSize" value="100"/>
<setting name="safeRowBoundsEnabled" value="false"/>
<setting name="mapUnderscoreToCamelCase" value="false"/>
<setting name="localCacheScope" value="SESSION"/>
<setting name="jdbcTypeForNull" value="OTHER"/>
<setting name="lazyLoadTriggerMethods" value="equals,clone,hashCode,toString"/>
settings>
分析一下Mybatis的执行过程!
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RIsoJ162-1610026690310)(F:\Java学习\截图\image-20210102214812239.png)]
作用域理解:
解决:字段名和属性名不一致
说明:在数据库中,两个英文的时候通常是用_隔开,而在我们的java程序中是第二个字母大写这样来区分的,那么这个就会导致两者的名字不一致,所以我们可以使用ResultMap来解决
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-U4eNORce-1610026690311)(F:\Java学习\截图\image-20210102230906347.png)]
方式一:在SQL语句中添加别名就可以了
<select id="selectUser" resultType="user">
select id,username name,password from user
select>
说明:这种方式比较笨重,不建议使用,一般使用第二种方式
方式二:通过ResultMap元素来进行映射
在ResultMap元素中有两个属性,id表示名字,type表示要修改的类
子元素result中的两个属性
通过这两个属性来进行映射,property相当于是map中的key,column相当于是value,通过key来取值
具体实现
实体类
public class User {
private Integer id;
private String name;
private String password;
代码
<resultMap id="map" type="User">
<result column="username" property="name"/>
<result column="id" property="id"/>
resultMap>
<select id="selectUser" resultMap="map">
select id,username,password from user
select>
说明:result可以有多个
通过百度去学习
作用:进行排错,很清晰的展现给我们看
在我们之前排错都时用的sout和debug,今后就使用日志来进行排错
日志具体的实现是在设置中设定的,默认是没有开启的,要去手动打开
mybatis-config.xml配置文件中添加
<settings>
<setting name="logImpl" value=""/>
settings>
有这几个value
注意事项:空格一定不能要,有空格会报错的,名字也一定要正确
标准的(STDOUT_LOGGING)不用导包,其他的是要导包才能使用的
<settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
settings>
说明:因为是自带的,所以直接设置就好了
通过maven仓库引入log4j
<dependency>
<groupId>log4jgroupId>
<artifactId>log4jartifactId>
<version>1.2.17version>
dependency>
#将等级为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/mybatis.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.sql.PreparedStatement=DEBUG
<settings>
<setting name="logImpl" value="LOG4J"/>
settings>
常用的三个方法
public class UserMapperTest {
private static Logger logger = Logger.getLogger(UserMapperTest.class);
@Test
public void Test1() {
logger.info("info:进入到UserMapper方法");
logger.debug("debug:进入到UserMapper方法");
logger.error("error:进入到UserMapper方法");
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
List<User> users = mapper.selectUser();
for (User user : users) {
System.out.println(user);
}
session.close();
}
}
做法:将分页的两个参数放到map中或者给直接设两个参数,然后传给sql语句执行
修改mapper文件
<select id="selectUserLimit" resultMap="User" parameterType="map">
select * from user limit #{startIndex} , #{pageSize}
select>
Mapper接口,参数为map
// 获取所有的用户信息
List<User> selectUserLimit(Map<String,Integer> map);
测试
我们除了使用Limit在SQL层面实现分页,也可以使用RowBounds在Java代码层面实现分页,当然此种方式作为了解即可。我们来看下如何实现的!
RowBounds实现分页的方式依赖于用全类名创建的方式,在创建RowBounds对象的时候传入两个参数,这两个参数分别是起始位置和一页的数量
代码实现
// 用RowBounds实现分页
List<User> getUserByRowBounds();
<select id="getUserByRowBounds" resultType="User">
select * from user
select>
@Test
public void Test2(){
SqlSession session = MybatisUtils.getSession();
RowBounds rowBounds = new RowBounds(0,4);
List<User> users = session.selectList("com.xiaozhi.dao.UserMapper.getUserByRowBounds", null, rowBounds);
for (User user : users) {
System.out.println(user);
}
session.close();
}
需要的时候看官方文档使用就可以了
官方文档:https://pagehelper.github.io/
关于接口的理解
接口从更深层次的理解,应是定义(规范,约束)与实现(名实分离的原则)的分离。
接口的本身反映了系统设计人员对系统的抽象理解。
接口应有两类:
一个体有可能有多个抽象面。抽象体与抽象面是有区别的。
三个面向区别
在openSession中传一个true值即可!
// 获取sqlSession连接
public static SqlSession getSession() {
return sqlSessionFactory.openSession(true);
}
mybatis最初配置信息是基于 XML ,映射语句(SQL)也是定义在 XML 中的。而到MyBatis 3提供了新的基于注解的配置。不幸的是,Java 注解的的表达力和灵活性十分有限。最强大的 MyBatis 映射并不能用注解来构建
sql 类型主要分成 :
底层实现:利用了反射得到注解里面的sql语句,然后进行解析,本质上利用了jvm的动态代理机制
**注意:**利用注解开发就不需要mapper.xml映射文件了 .
在接口的方法上添加注解
// 使用注解来写sql语句
@Select("select * from user")
List<User> getAllUser();
在核心文件中注册绑定接口
<mappers>
<mapper class="com.xiaozhi.dao.UserMapper"/>
mappers>
测试
@Test
public void Test3(){
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
List<User> allUser = mapper.getAllUser();
for (User user : allUser) {
System.out.println(user);
}
}
// 插入语句
@Insert("insert into user values(#{id},#{name},#{password})")
int addtUser(User user);
// 更新
@Update("update user set username = #{name} ,password = #{password} where id = #{id}")
int updateUser(User user);
// 删除
@Delete("delete from user where id = #{id}")
int deleteUser(@Param("id") int id);
#{} 的作用主要是替换预编译语句(PrepareStatement)中的占位符? 【推荐使用】
INSERT INTO user (name) VALUES (#{name});
INSERT INTO user (name) VALUES (?);
${} 的作用是直接进行字符串替换
INSERT INTO user (name) VALUES ('${name}');
INSERT INTO user (name) VALUES ('kuangshen');
使用注解和配置文件协同开发,才是MyBatis的最佳实践!
作用:偷懒,通过注释可以不用去写get、set方法、toString和equals、有参和无参构造器…等等。
缺点:不能对构造器进行重载,可以手动编写弥补缺点
去IDEA插件商店中下载
在项目中导入lombox的jar包,引入maven依赖
<dependencies>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.10version>
<scope>providedscope>
dependency>
dependencies>
使用注释
Features
@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
@var
experimental @var
@UtilityClass
Lombok config system
Code inspections
Refactoring actions (lombok and delombok)
测试
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private Integer id;
private String name;
private String passord;
}
说明:result只能处理简单额属性,不能处理复杂属性,比如对象、集合。
association元素用于处理实体类对象,对应多对一
collection元素用于处理集合,对应一对多
1、导入lombok,引入Maven依赖
<dependencies>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.10version>
<scope>providedscope>
dependency>
dependencies>
2、创建两个表并插入数据
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');
3、创建对应的两实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Student {
private Integer id;
private String name;
private Teacher teacher;
}
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Teacher {
private String name;
private Integer id;
}
4、编写对应的xml文件
TeacherMapper.xml
<mapper namespace="com.xiaozhi.dao.TeacherMapper">
mapper>
StudentMapper.xml
<mapper namespace="com.xiaozhi.dao.StudentMapper">
mapper>
情况:查询学生,学生和老师的关系时多对一
association元素:用于处理实体类对象,对应多对一
代码实现
1、给StudentMapper接口增加方法
// 查询所有学生
// 子查询
public List<Student> getStudentTeacher();
// 连表查询
public List<Student> getStudentTeacher2();
2、编写对应的Mapper文件
<select id="getStudentTeacher2" resultMap="StudentTeacher2">
select t1.id sid,t1.name sname,t2.name tname
from student t1,teacher t2
where t1.tid = t2.id
select>
<resultMap id="StudentTeacher2" type="Student">
<id property="id" column="sid"/>
<result property="name" column="sname"/>
<association property="teacher" javaType="Teacher">
<result property="name" column="tname"/>
association>
resultMap>
<select id="getStudentTeacher" resultMap="StudentTeacher">
select * from student
select>
<resultMap id="StudentTeacher" type="Student">
<association property="teacher" column="tid"
javaType="Teacher" select="getTeacher"/>
resultMap>
<select id="getTeacher" resultType="teacher">
select * from teacher where id = #{id}
select>
3、测试
情况:查询老师,老师和学生的关系是一对多
collection元素:用于处理集合,对应一对多
ofType属性是指集合中的泛型
代码实现
1、给TeacherMapper接口增加方法
public interface TeacherMapper {
// 实现一对多
// 连表查询的方式
public Teacher getTeacher(@Param("id") int id);
// 子查询的方式
public Teacher getTeacher2(@Param("id") int id);
}
2、编写对应的Mapper文件
<select id="getTeacher" resultMap="TeacherStudent">
SELECT t2.id sid,t2.name sname,t1.id tid,t1.name tname
FROM teacher t1,student t2
WHERE t1.`id` = t2.`tid` and t1.id = #{id}
select>
<resultMap id="TeacherStudent" type="Teacher" >
<id property="id" column="tid"/>
<result property="name" column="tname"/>
<collection property="students" ofType="Student">
<id property="id" column="sid"/>
<result property="name" column="sname"/>
<result property="tid" column="tid"/>
collection>
resultMap>
<select id="getTeacher2" resultMap="TeacherStudent2">
select * from teacher where id = #{id}
select>
<resultMap id="TeacherStudent2" type="Teacher">
<collection property="students" column="id" ofType="Student" select="getStudentByTeacherId"/>
resultMap>
<select id="getStudentByTeacherId" resultType="Student">
select * from student where tid = #{id}
select>
3、测试
1、关联-association
2、集合-collection
3、所以association是用于一对一和多对一,而collection是用于一对多的关系
4、JavaType和ofType都是用来指定对象类型的
5、映射无非就是将属性和序列对应起来,比如学生表中的tid(外键)可以映射成是老师的id,然后再进行查询,这也就是我们所说的结果嵌套查询(子查询)
什么是动态SQL:动态SQL指的是根据不同的查询条件 , 生成不同的Sql语句,就是sql语句拼接
作用:如果你使用过 JDBC 或其它类似的框架,你应该能理解根据不同条件拼接 SQL 语句有多痛苦,例如拼接时要确保不能忘记添加必要的空格,还要注意去掉列表最后一个列名的逗号。利用动态 SQL,可以彻底摆脱这种痛苦。
如果你之前用过 JSTL 或任何基于类 XML 语言的文本处理器,你对动态 SQL 元素可能会感觉似曾相识。在 MyBatis 之前的版本中,需要花时间了解大量的元素。借助功能强大的基于 OGNL 的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类,现在要学习的元素种类比原来的一半还要少。
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
@AllArgsConstructor
@NoArgsConstructor
public class Blog {
private Integer id;
private String title;
private String author;
private Date createTime;
private Integer views;
}
接口
public interface BlogMapper {
}
mybatis-config.xml配置文件
<mappers>
<mapper class="com.xiaozhi.dao.BlogMapper"/>
mappers>
说明:类似java中的if语句
1、BlogMapper接口添加
// 查询数据
List<Blog> selectBlog(Map map);
2、编写BlogMapper.xml文件
<select id="selectBlog" 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>
3、测试
说明:在if语句的列子中我们可以看到where后面跟着的是 1 = 1,这种写法是错误的,我们可以使用where标签来代替我们的where关键字
我们可以直接写我们的SQL语句,不用在后面加where关键字,where它会帮我们判断,如果标签中有条件匹配的它会自动给我们加上where关键字,第一个判断的不用加AND或OR关键字,只需要写上接在where关键字后面的内容就可以了,如果第一个没有被匹配到,那么往后的匹配到了它就会自动帮我们删除AND或OR关键字,然后加上where关键字
注意:除了第一个之外不用加关键字,往后都要加
代码实现
<!--查询数据-->
<select id="selectBlog" parameterType="map" resultType="blog">
select * from blog
<where>
<if test="title != null">
title = #{title}
</if>
<if test="author != null">
AND author = #{author}
</if>
</where>
</select>
在进行修改语句的时候我们都会在字段后面加逗号将每个字段隔开,如果我们按平常的语句来进行拼接,那么最后一个字段后面就会有逗号,这个时候就会导致SQL语句出问题,因此我们可以用set标签来解决这个问题,它可以进行加set关键字和删除逗号的操作
代码实现
<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>
是where标签和set标签的老大,它负责来定制他们两个
这是官网中的两个例子,我们可以看到它可以定制where的前缀和定义set标签的后缀,根据需求来定制
<trim prefix="WHERE" prefixOverrides="AND |OR ">
...
trim>
<trim prefix="SET" suffixOverrides=",">
...
trim>
作用:将公共部分独立出来,需要的时候就引用,重复的代码不用多次写,便利
代码实现
<sql id="public">
<if test="title != null">
title = #{title}
if>
<if test="author != null">
AND author = #{author}
if>
sql>
<select id="selectBlog" parameterType="map" resultType="blog">
select * from blog
<where>
<include refid="public">include>
where>
select>
<update id="updateBlog" parameterType="map">
update blog
<set>
<include refid="public">include>
set>
where id = #{id}
update>
说明:类似java中的switch语句,它只能选择其中一个进行sql语句拼接,同时还要注意它的结果和传参数的顺序是没有关系的。
代码实现
1、添加方法
// 根据不同值查询结果
List<Blog> selectBlogChoose (Map map);
2、编写xml文件
<select id="selectBlogChoose" parameterType="map" resultType="blog">
select * from blog
<where>
<choose>
<when test="author != null">
author = #{author}
when>
<when test="title != null">
and title = #{title}
when>
choose>
where>
select>
3、测试
@Test
public void Test3(){
SqlSession session = MybatisUtils.getSession();
BlogMapper mapper = session.getMapper(BlogMapper.class);
Map map = new HashMap();
map.put("title","我的世界");
map.put("author","小智");
List<Blog> blogs = mapper.selectBlogChoose(map);
for (Blog blog : blogs) {
System.out.println(blog);
}
session.close();
}
说明:类似java中的for循环
说明:集合中有多少个值它就会取多少个值进行拼接
代码实现
1、编写接口
List<Blog> selectBlogForEach(Map map);
2、编写xml文件
<select id="selectBlogForEach" parameterType="map" resultType="blog">
SELECT * FROM blog
<where>
<foreach collection="list" item="item" open="and (" close=")" separator="or">
author = #{item}
foreach>
where>
select>
3、测试
@Test
public void Test5(){
SqlSession session = MybatisUtils.getSession();
BlogMapper mapper = session.getMapper(BlogMapper.class);
HashMap map = new HashMap();
ArrayList<String> list = new ArrayList<>();
list.add("小智");
list.add("花花公子");
list.add("霍金");
map.put("list",list);
mapper.selectBlogForEach(map);
session.close();
}
最终的结果
2、为什么使用缓存?
3、什么样的数据能使用缓存?
Mybatis缓存
MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率。
MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存
主从复制:复制出多个数据库,每个数据库的数据库的数据一致,当一个数据库中有增删改的操作,那么其他数据库会同步的更新,这个就是主从复制
说明:默认就是开启一级缓存(本地缓存),SqlSession级别,在获取连接和关闭连接之间,连接关闭了它就没了,它其实就是一个map集合,将第一次查询的数据放在集合中,当我们再次查询同样的语句时就会取出map集合中的值,以达到快速查询的效果
代码示例
1、创建一个实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class User implements Serializable {
private Integer id;
private String username;
private String password;
}
2、创建一个Mapper接口
// 根据id查询用户
User selectUserById(@Param("id") int id);
3、编写对应的xml文件
<select id="selectUserById" resultType="user">
select * from user where id = #{id}
select>
4、测试
@Test
public void Test1(){
SqlSession session = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
User user1 = mapper.selectUserById(1);
System.out.println(user1);
User user2 = mapper.selectUserById(1);
System.out.println(user2);
session.close();
}
结果显示
可以这样写
<cache/>
也可以给它增加参数值
<cache
eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
这个更高级的配置创建了一个 FIFO 缓存,每隔 60 秒刷新,最多可以存储结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此对它们进行修改可能会在不同线程中的调用者产生冲突。
注意:在没有策略的时候,需要对实体类进行序列化
可用的清除策略有:
LRU
– 最近最少使用:移除最长时间不被使用的对象。FIFO
– 先进先出:按对象进入缓存的顺序来移除它们。SOFT
– 软引用:基于垃圾回收器状态和软引用规则移除对象。WEAK
– 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。默认的清除策略是 LRU。
flushInterval(刷新间隔)属性可以被设置为任意的正整数,设置的值应该是一个以毫秒为单位的合理时间量。 默认情况是不设置,也就是没有刷新间隔,缓存仅仅会在调用语句时刷新。
size(引用数目)属性可以被设置为任意正整数,要注意欲缓存对象的大小和运行环境中可用的内存资源。默认值是 1024。
readOnly(只读)属性可以被设置为 true 或 false。只读的缓存会给所有调用者返回缓存对象的相同实例。 因此这些对象不能被修改。这就提供了可观的性能提升。而可读写的缓存会(通过序列化)返回缓存对象的拷贝。 速度上会慢一些,但是更安全,因此默认值是 false。
提示 二级缓存是事务性的。这意味着,当 SqlSession 完成并提交时,或是完成并回滚,但没有执行 flushCache=true 的 insert/delete/update 语句时,缓存会获得更新。
代码实现
1、在mybatis-config.xml文件中显示出来开启了,二级缓存的默认值是true,显示出来增加可读性
<settings>
<setting name="cacheEnabled" value="true"/>
settings>
2、在mapper对应的xml文件中添加cache标签
<mapper namespace="com.xiaozhi.dao.UserMapper">
<cache/>
<select id="selectUserById" resultType="user">
select * from user where id = #{id}
select>
mapper>
3、测试
// 二级缓存
@Test
public void Test2(){
SqlSession session = MybatisUtils.getSession();
SqlSession session2 = MybatisUtils.getSession();
UserMapper mapper1 = session.getMapper(UserMapper.class);
UserMapper mapper2 = session2.getMapper(UserMapper.class);
User user = mapper1.selectUserById(1);
System.out.println(user);
session.close();
User user2 = mapper2.selectUserById(1);
System.out.println(user2);
session2.close();
}
结果显示
说明:我们可以看到第二次查询的时候它是和缓存交互的,并没有到数据库中查询
说明:用户请求过来先去查找二级缓存有没有数据,有的话就返回,没有的话就往一级缓存找,有就返回,没有就到数据库中查询
EhCache 是一个纯Java的进程内缓存框架,具有快速、精干等特点,是Hibernate中默认的CacheProvider。
Ehcache是一种广泛使用的开源Java分布式缓存。主要面向通用缓存,Java EE和轻量级容器。它具有内存和磁盘存储,缓存加载器,缓存扩展,缓存异常处理程序,一个gzip缓存servlet过滤器,支持REST和SOAP api等特点。
实现:cache标签中的type属性中指定自定义的缓存实现类
1、导入依赖
<dependency>
<groupId>org.mybatis.cachesgroupId>
<artifactId>mybatis-ehcacheartifactId>
<version>1.1.0version>
dependency>
2、在mapper.xml中使用对应的缓存即可
<mapper namespace="com.xiaozhi.dao.UserMapper">
<select id="selectUserById" resultType="user">
select * from user where id = #{id}
select>
mapper>
3、编写ehcache.xml文件,如果在加载时未找到/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>
说明:一般都是使用redis缓存数据库来进行缓存的,所以自定义缓存了解一下即可
①接口名和它对应的xml文件不对应的话就会报 org.apache.ibatis.binding.BindingException 异常