Mybatis基础-复习-面试

MyBatis 是一款优秀的持久层框架,它支持自定义 SQL、存储过程以及高级映射。
MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。
MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java POJO(Plain Old Java Objects,普通老式 Java 对象)为数据库中的记录。

持久化
持久化就是将程序的数据在持久状态和瞬时状态转化的过程

内存--- 断电即失

数据库(jdbc) io文件持久化

为什么要持久化?
持久化技术封装了数据访问细节,为大部分业务逻辑提供面向对象的API。

  1. 通过持久化技术可以减少访问数据库数据次数,增加应用程序执行速度;
  2. 代码重用性高,能够完成大部分数据库操作;
  3. 松散耦合,使持久化不依赖于底层数据库和上层业务逻辑实现,更换数据库时只需修改配置文件而不用修改代码。
    持久层
    完成持久化工作的代码块
    层界限十分明显

什么是MyBatis

MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了google code,并且改名为MyBatis 。2013年11月迁移到Github。
MyBatis 是一款优秀的持久层框架,它支持自定义 SQL、存储过程以及高级映射。
MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。

MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java POJO
搭建环境
CREATE DATABASE mybatis;
USE mybatis;

CREATE TABLE user (

id int(20) NOt NULL,
name VARCHAR(30)  DEFAULT NULL,
pwd  VARCHAR(30)  DEFAULT NULL,
PRIMARY KEY (id)

)
ENGINE=INNODB
CHARSET=utf8
;
INSERT INTO USER (name,pwd)
VALUES
('lisi','123'),
('wangwu','123'),
('test','123');
导入依赖




mysql
mysql-connector-java
5.1.32




org.mybatis
mybatis
3.5.6




junit
junit
4.13.2






src/main/resources

/.properties
/.xml

true


src/main/java

/.properties
/.xml

true





PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">


















从 XML 中构建 SqlSessionFactory
每个基于 MyBatis 的应用都是以一个 SqlSessionFactory 的实例为核心的。SqlSessionFactory 的实例可以通过 SqlSessionFactoryBuilder 获得。而 SqlSessionFactoryBuilder 则可以从 XML 配置文件或一个预先配置的 Configuration 实例来构建出 SqlSessionFactory 实例。
从 XML 文件中构建 SqlSessionFactory 的实例非常简单,建议使用类路径下的资源文件进行配置。 但也可以使用任意的输入流(InputStream)实例,比如用文件路径字符串或 file:// URL 构造的输入流。MyBatis 包含一个名叫 Resources 的工具类,它包含一些实用方法,使得从类路径或其它位置加载资源文件更加容易。

public class MybatisUtils {

InputStream inputStream;
SqlSessionFactory sqlSessionFactory;
{
try {
String resource = "/mybatis-config.xml";
inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory= new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
}
//既然有了 SqlSessionFactory,顾名思义,我们可以从中获得 SqlSession 的实例。
// SqlSession 提供了在数据库执行 SQL 命令所需的所有方法。
// 你可以通过 SqlSession 实例来直接执行已映射的 SQL 语句s
public SqlSessionFactory getSqlSessionFactory() {
return sqlSessionFactory;
}
从 SqlSessionFactory 中获取 SqlSession
既然有了 SqlSessionFactory,顾名思义,我们可以从中获得 SqlSession 的实例。SqlSession 提供了在数据库执行 SQL 命令所需的所有方法。你可以通过 SqlSession 实例来直接执行已映射的 SQL 语句
public interface UserMapper {

public List  getUserList();

}


PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">





public class UserTest {

@Test
public void test(){

    //获取sqlsession对象
    SqlSession sqlSession= MybatisUtils.getSqlSession();

    //方式一 getmapper

    UserMapper user = sqlSession.getMapper(UserMapper.class);
    List userList = user.getUserList();
    for (User user1:userList){
        System.out.println(user1);
    }
sqlSession.close();
}

}

作用域(Scope)和生命周期

理解我们之前讨论过的不同作用域和生命周期类别是至关重要的,因为错误的使用会导致非常严重的并发问题。


提示 对象生命周期和依赖注入框架

依赖注入框架可以创建线程安全的、基于事务的 SqlSession 和映射器,并将它们直接注入到你的 bean 中,因此可以直接忽略它们的生命周期。 如果对如何通过依赖注入框架使用 MyBatis 感兴趣,可以研究一下 MyBatis-Spring 或 MyBatis-Guice 两个子项目。


SqlSessionFactoryBuilder

这个类可以被实例化、使用和丢弃,一旦创建了 SqlSessionFactory,就不再需要它了。 因此 SqlSessionFactoryBuilder 实例的最佳作用域是方法作用域(也就是局部方法变量)。 你可以重用 SqlSessionFactoryBuilder 来创建多个 SqlSessionFactory 实例,但最好还是不要一直保留着它,以保证所有的 XML 解析资源可以被释放给更重要的事情。

SqlSessionFactory

SqlSessionFactory 一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例。 使用 SqlSessionFactory 的最佳实践是在应用运行期间不要重复创建多次,多次重建 SqlSessionFactory 被视为一种代码“坏习惯”。因此 SqlSessionFactory 的最佳作用域是应用作用域。 有很多方法可以做到,最简单的就是使用单例模式或者静态单例模式。

SqlSession

每个线程都应该有它自己的 SqlSession 实例。SqlSession 的实例不是线程安全的,因此是不能被共享的,所以它的最佳的作用域是请求或方法作用域。 绝对不能将 SqlSession 实例的引用放在一个类的静态域,甚至一个类的实例变量也不行。 也绝不能将 SqlSession 实例的引用放在任何类型的托管作用域中,比如 Servlet 框架中的 HttpSession。 如果你现在正在使用一种 Web 框架,考虑将 SqlSession 放在一个和 HTTP 请求相似的作用域中。 换句话说,每次收到 HTTP 请求,就可以打开一个 SqlSession,返回一个响应后,就关闭它。 这个关闭操作很重要,为了确保每次都能执行关闭操作,你应该把这个关闭操作放到 finally 块中。 下面的示例就是一个确保 SqlSession 关闭的标准模式:

try (SqlSession session = sqlSessionFactory.openSession()) {

 // 你的应用逻辑代码

}

在所有代码中都遵循这种使用模式,可以保证所有数据库资源都能被正确地关闭。
CRUD

  1. Namespace
    namespace中的包名要和 Dao、Mapper 接口的包名一致

2.Id:对应的namespace中的方法名
3.parameterType :Sql语句执行的参数
4.resultType :Sql语句执行的返回值
public interface UserMapper {
//查询所有用户
public List getUserList();
//根据id查询用户
User getUserById(int id);
//insert 一个用户
void insertUser(User user);
//修改一个用户
void updateUser(User user);
//删除一个用户
void deleted(Integer id);
}



查询语句


      添加语句

    insert  into User  (name,pwd) values (#{name} , #{pwd})

       修改语句

    update User set  name = #{name} , pwd = #{pwd}  where  id = #{id}

       删除语句

    delete from  User    where  id = #{id}

注意点 ; 增删改需要提交事务分析错误

  1. 标签不要匹配错
  2. Resource绑定mapper,需要使用路径
  3. 程序配置文件必须符合规范
  4. NullpointerException 没有注册到资源
  5. 输出的xml文件中存在中文乱码问题
  6. Maven资源没有导出问题

万能Map

图片.png

图片.png

图片.png

Map 传递参数,直接在sql中取出key即可 resultType=“map”
对象传递参数,直接在sql中取对象的属性即可 resultType=“Object”
只有一个基本类型参数的情况下,可以直接在sql中取到
多个参数用Map,或者注解

配置解析

MyBatis 的配置文件包含了会深深影响 MyBatis 行为的设置和属性信息。 配置文档的顶层结构如下:

  • configuration(配置)

  • properties(属性)

  • settings(设置)

  • typeAliases(类型别名)

  • typeHandlers(类型处理器)

  • objectFactory(对象工厂)

  • plugins(插件)

  • environments(环境配置)

  • environment(环境变量)

  • transactionManager(事务管理器)

  • dataSource(数据源)

  • databaseIdProvider(数据库厂商标识)

  • mappers(映射器)

环境配置(environments)

MyBatis 可以配置成适应多种环境,这种机制有助于将 SQL 映射应用于多种数据库之中, 现实情况下有多种理由需要这么做。例如,开发、测试和生产环境需要有不同的配置;或者想在具有相同 Schema 的多个生产数据库中使用相同的 SQL 映射。还有许多类似的使用场景。

不过要记住:尽管可以配置多个环境,但每个 SqlSessionFactory 实例只能选择一种环境。

Mybatis默认事务管理器是jdbc,连接池: pooled

属性(properties)

这些属性可以在外部进行配置,并可以进行动态替换。你既可以在典型的 Java 属性文件中配置这些属性,也可以在 properties 元素的子元素中设置。jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/e3mall-32?characterEncoding=utf-8
jdbc.username=root
jdbc.password=123456

图片.png

图片.png

图片.png

可以直接引入外部文件
可以在其中增加一些属性配置
如果两个文件有同一个字段,优先使用外部配置文件
类型别名(typeAliases)
类型别名可为 Java 类型设置一个缩写名字。 它仅用于 XML 配置,意在降低冗余的全限定类名书写。



resultType="User">
也可以指定一个包名,MyBatis 会在包名下面搜索需要的 Java Bean


每一个在包 domain.blog 中的 Java Bean,在没有注解的情况下,会使用 Bean 的首字母小写的非限定类名来作为它的别名
resultType="user"若有注解,则别名为其注解值。见下面的例子:
@Alias("author")
public class Author {
...
}
设置(settings)这是 MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为。 下表描述了设置中各项设置的含义、默认值等。


图片.png

图片.png
图片.png

映射器(mappers)
既然 MyBatis 的行为已经由上述元素配置完了,我们现在就要来定义 SQL 映射语句了。
但首先,我们需要告诉 MyBatis 到哪里去找到这些语句。
在自动查找资源方面,Java 并没有提供一个很好的解决方案,所以最好的办法是直接告诉 MyBatis 到哪里去找映射文件。
你可以使用相对于类路径的资源引用,或完全限定资源定位符(包括 file:/// 形式的 URL),或类名和包名等

MapperRegistry :注册绑定我们的Mapper文件


图片.png

图片.png

图片.png

注意点 –方式二和方式三
接口和他的Mapper配置文件必须同名
接口和他的Mapper配置文件在同一包下

生命周期和作用域
图片.png

理解我们之前讨论过的不同作用域和生命周期类别是至关重要的,因为错误的使用会导致非常严重的并发问题。
SqlSessionFactoryBuilder
一旦创建了 SqlSessionFactory,就不再需要它了。

方法作用域(也就是局部方法变量)
SqlSessionFactory(相当于数据库连接池)
一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例。
最简单的就是使用单例模式或者静态单例模式。
应用作用域
SqlSession(连接到连接池的一个请求)
SqlSession 的实例不是线程安全的,因此是不能被共享的
请求或方法作用域。
用完关闭,否则资源被占用


图片.png

这里面的每个mapper,就代表一个具体的业务
ResultMap结果集映射
解决属性名与数据库字段名不一致的问题
public class User {
private Integer id;
private String username;
private String password;


图片.png

解决办法
1.起别名
select id,name ,pwd from User where id = #{id}
改为 select id,name as username,pwd as password from User where id = #{id}

  1. 用resultMap

数据库字段 : id name pwd
实体类参数: id username ppassword

Column 列 (数据库中的字段) property 属性 (实体类中的属性)






resultMap 元素是 MyBatis 中最重要最强大的元素。
它可以让你从 90% 的 JDBC ResultSets 数据提取代码中解放出来,并在一些情形下允许你进行一些 JDBC 不支持的操作。
实际上,在为一些比如连接的复杂语句编写映射代码的时候,一份 resultMap 能够代替实现同等功能的数千行代码。
ResultMap 的设计思想是,对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。

上述语句只是简单地将所有的列映射到 HashMap 的键上,这由 resultType 属性指定。
虽然在大部分情况下都够用,但是 HashMap 并不是一个很好的领域模型。
你的程序更可能会使用 JavaBean 或 POJO(Plain Old Java Objects,普通老式 Java 对象)作为领域模型。MyBatis 对两者都提供了支持。

没有一个需要显式配置 ResultMap,这就是 ResultMap 的优秀之处——你完全可以不用显式地配置它们。 虽然上面的例子不用显式配置 ResultMap。


如何数据库字段和实体类属性一致 可以省略,因为mybatis会自动生成

            日志
    日志工厂
图片.png

SLF4J
LOG4J 【掌握】
LOG4J2
JDK_LOGGING
COMMONS_LOGGING
STDOUT_LOGGING 【掌握】
NO_LOGGING

在mybatis-config.xml配置文件中配置





STDOUT_LOGGING :标准日志输出

图片.png

Log4j

Log4j是Apache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台、文件、GUI组件,甚至是套接口服务器、NT的事件记录器、UNIX Syslog守护进程等;我们也可以控制每一条日志的输出格式;通过定义每一条日志信息的级别,我们能够更加细致地控制日志的生成过程。最令人感兴趣的就是,这些可以通过一个配置文件来灵活地进行配置,而不需要修改应用的代码。

  1. 导包

  2. 写配置文件

log4j.properties

### 设置### log4j.rootLogger = debug,stdout,D,E ### 输出信息到控制抬 ### log4j.appender.stdout = org.apache.log4j.ConsoleAppender log4j.appender.stdout.Target = System.out log4j.appender.stdout.layout = org.apache.log4j.PatternLayout log4j.appender.stdout.layout.ConversionPattern = [%-5p] %d{yyyy-MM-dd HH:mm:ss,SSS} method:%l%n%m%n ### 输出DEBUG 级别以上的日志到=E://logs/error.log ### log4j.appender.D = org.apache.log4j.DailyRollingFileAppender log4j.appender.D.File = E://logs/log.log log4j.appender.D.Append = true log4j.appender.D.Threshold = DEBUG log4j.appender.D.layout = org.apache.log4j.PatternLayout log4j.appender.D.layout.ConversionPattern = %-d{yyyy-MM-dd HH:mm:ss} [ %t:%r ] - [ %p ] %m%n ### 输出ERROR 级别以上的日志到=E://logs/error.log ### log4j.appender.E = org.apache.log4j.DailyRollingFileAppender log4j.appender.E.File =E://logs/error.log log4j.appender.E.Append = true log4j.appender.E.Threshold = ERROR log4j.appender.E.layout = org.apache.log4j.PatternLayout log4j.appender.E.layout.ConversionPattern = %-d{yyyy-MM-dd HH:mm:ss} [ %t:%r ] - [ %p ] %m%n

3.在mybatis-config.xml配置文件中配置

**


简单使用

1导入包 org.apache.log4j.Logger;

2.日志对象

static Logger logger = Logger.getLogger(mybatilsTest.class); //mybatilsTest

3.日志级别

public void logtset(){ logger.info("info进入"); logger.debug("debug进入"); ------调试的时候可以用 logger.error("错误啦");
}

图片.png

分页
Sql
Select * from XXX LIMIT startIndex , pageSize;

使用mybatis实现分页,核心sql

  1. 接口
    //分页
    public List getlimitUser(Map map);

  2. Mapper.xml






  1. 测试
    @Test
    public void limit(){
    SqlSession sqlSession = MybatisUtils.getSqlSession();
    UserMapper mapper = sqlSession.getMapper(UserMapper.class);

    Map map=new HashMap();

    map.put("startIndex",0);
    map.put("pageSize",2);

    List list = mapper.getlimitUser(map);
    for (User user : list) {
    System.out.println(user);
    }

}

            分页插件
           Pagehelper
图片.png

1). 使用 Maven

在 pom.xml 中添加如下依赖:


    com.github.pagehelper
    pagehelper
    最新版本

2. 配置拦截器插件

1. 在 MyBatis 配置 xml 中配置拦截器插件

2. 在 Spring 配置文件中配置拦截器插件

使用 spring 的属性配置方式,可以使用 plugins 属性像下面这样配置:

params=value1

https://pagehelper.github.io/docs/howtouse/

使用注解开发

对于像 BlogMapper 这样的映射器类来说,还有另一种方法来完成语句映射。 它们映射的语句可以不用 XML 来配置,而可以使用 Java 注解来配置。比如,上面的 XML 示例可以被替换成如下的配置:

package org.mybatis.example;

public interface BlogMapper {

@Select("SELECT * FROM blog WHERE id = #{id}")

Blog selectBlog(int id);

}

使用注解来映射简单语句会使代码显得更加简洁,但对于稍微复杂一点的语句,Java 注解不仅力不从心,还会让你本就复杂的 SQL 语句更加混乱不堪。 因此,如果你需要做一些很复杂的操作,最好用 XML 来映射语句。


图片.png

图片.png

本质: 反射机制实现
底层: 动态代理

Mybatis工作原理
图片.png

在工具类可以设置默认提交事务
图片.png

图片.png

@Param

1.关于@Param
@Param是MyBatis所提供的(org.apache.ibatis.annotations.Param),作为Dao层的注解,作用是用于传递参数,从而可以与SQL中的的字段名相对应,一般在2=<参数数<=5时使用最佳。


图片.png
  1. 将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号。如:order by #user_id#,如果传入的值是111,那么解析成sql时的值为order by "111", 如果传入的值是id,则解析成的sql为order by "id".

  1. user_id$,如果传入的值是111,那么解析成sql时的值为order by user_id, 如果传入的值是id,则解析成的sql为order by id.
  2. 方式能够很大程度防止sql注入。

4.$方式无法防止Sql注入。

5..

MyBatis排序时使用order by 动态参数时需要注意,用$而不是#

字符串替换
默认情况下,使用#{}格式的

//方法存在多个参数,所以参数前面必须加上@Param(“xxx”)注解


//根据多条件查询
public User getUserByIdByname(@Param("id")Integer id , @Param("name")String name);
//根据条件查询
User wangwu = mapper.getUserByIdByname(2, "wangwu");
System.out.println(wangwu);

                    注解完成CRUD



public interface UserMapper {

//查询所有用户
@Select("select * from User")
public List getUserList();

//根据条件查询
@Select("select * from User where id =#{id} and name=#{name}")
public User getUserByIdByname(@Param("id")Integer id ,@Param("name")String name);

//insert xxx into
@Insert(" insert  into User  (name,pwd) values (#{name} , #{pwd})")
public void insertUser(User user);

//update setxxx
@Update(" update User set  name = #{name} , pwd = #{pwd}  where  id = #{id} ")
public void updateUserById(User user);

//deleted form xxx
@Delete("delete from  User    where  id = #{id}")
public void deleteById(Integer id);

}

@Test
public void selectTest(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);

/*      //查询所有用户
    List userList = mapper.getUserList();

    for (User user : userList) {
        System.out.println(user);
    }*/

   /* //根据条件查询
    User wangwu = mapper.getUserByIdByname(2, "wangwu");
    System.out.println(wangwu);*/

   /* //insert xxx into
    User user=new User();
    user.setName("添加");
    user.setPwd("21212");
    mapper.insertUser(user);*/

        /*//update setxxx
     User user=new User(12,"修改","修改密码");
    mapper.updateUserById(user);
    */

     /*   //deleted form xxx
       mapper.deleteById(7);*/
}

}

                    复杂查询         
                    多对一处理
    多个学生对应一个老师

对于学生而言, 关联……多个学生,关联一个老师 【多对一】

对于老师而言, 集合……一个老师可以有多个学生。 【一对多】

        实体类

public class Student implements Serializable {

private Integer id;
private String name;

//------private  Integer tid;
//学生需要关联一个老师   tid
private Teacher teacher;

public class Teacher {

private Integer id;
private String  name;

create table teacher(

id int(10) NOT NULL,
name VARCHAR(20)  NOT null,

primary key(id)

)
ENGINE=INNODB
DEFAULT CHARSET=utf8

INSERT INTO teacher (id,name) VALUES (2,'李老师');

create table student(

id int(10) NOT NULL,
name VARCHAR(20)  NOT null,
tid int(10) NOT 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,'小红',2);
INSERT INTO student (id,name,tid) VALUES (3,'小南',1);
INSERT INTO student (id,name,tid) VALUES (4,'小球',2);


图片.png

图片.png
图片.png

图片.png

· association – 一个复杂类型的关联;许多结果将包装成这种类型

  • 嵌套结果映射 – 关联可以是 resultMap 元素,或是对其它结果映射的引用

· collection – 一个复杂类型的集合

  • 嵌套结果映射 – 集合可以是 resultMap 元素,或是对其它结果映射的引用

//查询所有学生信息,以及对应的老师信息

SELECT s.id,s.name,t.name from student s

JOIN teacher t

on t.id = s.tid;

SELECT s.id,s.name,t.name from student s ,teacher t

where t.id = s.tid;

**

mybatis的javaType和ofType

都是指定对象的类型 不同的是当使用反向查询select从另一个maper文件中取出数据时必须用ofType

都可以为collection和association是指定对象的类型,

都不是必须写的, 只有反向select时需要ofType;

按照查询嵌套处理


**

@Test public void test(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); StudentMapper mapper = sqlSession.getMapper(StudentMapper.class); List studnetList = mapper.getStudnetList(); for (Student student : studnetList) { System.out.println(student);
}
}

按照结果嵌套处理

** 

@Test public void test(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); StudentMapper mapper = sqlSession.getMapper(StudentMapper.class); List studnetList = mapper.getStudnetList2(); for (Student student : studnetList) { System.out.println(student);
}
}

一对多处理

一个老师拥有多个学生

对于老师就是一对多关系

实体类

 public class Teacher { private Integer id; private String name; //一个老师拥有多个学生 private List students;

}

public class Student implements Serializable { private Integer id; private String name; private Integer tid;

**  ** 

**

  

@Test public void test(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); TeacherMapper mapper = sqlSession.getMapper(TeacherMapper.class); Teacher teacherById = mapper.getTeacherAndStudentById(1); System.out.println(teacherById); //输出结果 Teacher{id=1, name='王老师',
// students=[Student{id=1, name='
小明', tid=1}, Student{id=3, name='小南', tid=1}]} MybatisUtils.close();

}


图片.png

图片.png

你可能感兴趣的:(Mybatis基础-复习-面试)