Mybatis工作原理浅析

Mybatis官方网址

https://mybatis.org/spring/(mybatis-spring)
https://mybatis.org/mybatis-3/(mybatis)

Mybatis最核心的两点

要学习Mybatis的工作原理,可以从搞清楚以下两个问题入手:

一、Mybatis是怎么实现的返回Mapper接口的对象完成对数据库的增删改查操作(Mapper是咱们自己定义的接口,里面定义了一些对数据库增删改查的方法)

public interface UserMapper {
    @Select("select user_name,email from icm_user")
    public List<Map<String,String>> query();
}
SqlSession session = sqlSessionFactory.openSession()) {
UserMapper userMapper = session.getMapper(UserMapper .class);
List<Map<String,String>>  list = mapper.query();

二、第一点并不难,应该是返回的动态代理的对象,Mybatis底层源码这块也是利用的JDK动态代理技术返回的Mapper代理对象,但是这个动态代理的对象是怎么交给Spring容器管理的哪?把一个【类】交给spring容器和把一个【对象】交给spring容器是有区别的,一个【类】交给spring管理可以在类上面加一个@Component注解,这样的话对象的产生过程就交给spring来管理了。
如何把一个对象怎么放到spring容器中那?就是对象的产生过程自己可以控制,有一下两种方式:
(1)@Bean注解
(2)实现FactoryBean接口,在getObject()方法里面返回对象,这样的话这个对象也是被spring容器管理的,实现了FactoryBean接口的Bean,根据该Bean的Id从BeanFactory中获取的实际上是FactoryBean的getObject()返回的对象,而不是FactoryBean本身, 如果要获取FactoryBean对象,可以在id前面加一个&符号来获取。
(3)调用AnnotationConfigApplicationContext对象的getBeanFactory().registerSingleton()方法(不常用)

在Mybatis的启动类上加一个@MapperScan(“com.xx.xx.mapper”)或者在xml配置文件中配置要扫描的Mapper,指定一下要扫描的包,就可以把UserMapper通过@Autowired注解自动注入了其他业务类作为属性,这个UserMapper是动态代理对象肯定是没有疑问的了,但是这个动态代理对象是动态产生的,Mybatis又是怎么做到的把这个动态代理对象放到spring容器中的,下面用代码来简单模拟一下这个过程的原理。

(1)AppConfig 类,在main方法中初始化spring容器,调用容器getBean方法拿到UserMapper代理类对象,调用query方法完成查询数据库的操作,其中MyScan是自己定义的注解(类似mybatis的MapperScan注解的作用),还把SqlSessionFactoryBean 、DataSource 加到了spring容器中
(2)自定义MyMapperFactoryBean(类似mybatis的MapperFactoryBean)实现FactoryBean接口(利用了spring的扩展点),在getObject方法里面通过jdk动态代理返回代理对象,其中属性1、SqlSessionFactory sqlSessionFactory,可以从spring容器中拿到,2、Class mapperInterface可以通过构造方法传入
(3)MyInvocationHandler,实现了InvocationHandler接口,重写invoke方法,里面完成真正的查询数据库的逻辑
(4)MyImportBeanDefinitionRegister ,实现了ImportBeanDefinitionRegistrar 接口(利用了spring的扩展点),在这个类中可以拿到MyMapperFactoryBean的BeanDefinition(如果不知道什么是BeanDefinition,可以先看一下这篇文章),进而调用构造方法给属性赋值,就是把要代理的接口传过去beanDefinition.getConstructorArgumentValues().addGenericArgumentValue()
根据MyScan(自己定义的注解)上要扫描的包,获取该包下的所有类,拿到类的名字,第一个字符转成小写作为bean的名字
(5)MyScan 注解中用@import导入MyImportBeanDefinitionRegister

到此就模拟mybatis,完成了把动态代理对象加到spring容器中,就是利用了jdk动态代理技术和spring提供的扩展点FactoryBean和ImportBeanDefinitionRegistrar。

AppConfig 类

@Configuration
@MyScan("com.mi.asop.demo.mapper")
public class AppConfig {
public static void main(String[] args) {
    AnnotationConfigApplicationContext ac = new
            AnnotationConfigApplicationContext(AppConfig.class);
    UserMapper userMapper = (UserMapper)ac.getBean("userMapper");
    List<Map<String,String>> resultList = userMapper.query();
    resultList.forEach((map->{
        System.out.println(map.get("user_name"));
        System.out.println(map.get("user_account"));
    }));
}
@Bean
public SqlSessionFactoryBean sqlSessionFactoryBean(DataSource dataSource) throws Exception {
    SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
    sqlSessionFactoryBean.setDataSource(dataSource);
    return sqlSessionFactoryBean;
}
@Bean
public DataSource dataSource(){
    DriverManagerDataSource driverManagerDataSource = new DriverManagerDataSource();
    driverManagerDataSource.setDriverClassName("com.mysql.jdbc.Driver");
    driverManagerDataSource.setUsername("xxxx");
    driverManagerDataSource.setPassword("xxxxx");
    driverManagerDataSource.setUrl("jdbc:mysql://xxxx:3308/xxxx?   useUnicode=true&characterEncoding=utf8&useSSL=false");
    return driverManagerDataSource;
}
}

UserMapper接口类

public interface UserMapper {
    @Select("select user_name,user_account from user_info")
    public List<Map<String,String>> query();
}

MyMapperFactoryBean 类

public class MyMapperFactoryBean implements FactoryBean {
@Autowired
SqlSessionFactory sqlSessionFactory;
Class mapperInterface;
public MyMapperFactoryBean(Class mapperInterface){
    this.mapperInterface = mapperInterface;
}
@Override
public Object getObject() throws Exception {
    Class[] clazz = new Class[]{mapperInterface};
    Object mapper = Proxy.newProxyInstance(MyMapperFactoryBean.class.getClassLoader(),clazz,new MyInvocationHandler(sqlSessionFactory));
    return mapper;
}
@Override
public Class<?> getObjectType() {
    return mapperInterface;
}
}

MyInvocationHandler

public class MyInvocationHandler implements InvocationHandler {
private SqlSessionFactory sqlSessionFactory;
public MyInvocationHandler(SqlSessionFactory sqlSessionFactory){
    this.sqlSessionFactory = sqlSessionFactory;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    String sql = method.getAnnotation(Select.class).value()[0];
    System.out.println("获取要执行的sql语句:"+sql);
    Connection connection = null;
    Statement statement = null;
    SqlSession sqlSession = null;
    List<Map<String,String>> resultList = new ArrayList<>();

    try{
        sqlSession = sqlSessionFactory.openSession();
        connection = sqlSession.getConnection();
        statement = connection.createStatement();
        ResultSet resultSet = statement.executeQuery(sql);
        while(resultSet.next()){
            Map<String,String> map = new HashMap<>();
            map.put("user_name",resultSet.getString("user_name"));
            map.put("user_account",resultSet.getString("user_account"));
            resultList.add(map);
        }
    }catch (Exception e){
        e.printStackTrace();
    }finally {
        if(statement!=null){
            statement.close();
        }
        if(connection!=null){
            connection.close();
        }
        if(sqlSession!=null){
            sqlSession.close();
        }
    }
    return resultList;
}

}

MyScan 注解类

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
@Import(MyImportBeanDefinitionRegister.class)
public @interface MyScan {
    String[] value() default {};

}

MyImportBeanDefinitionRegister 类

public class MyImportBeanDefinitionRegister implements ImportBeanDefinitionRegistrar {
@Override
public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry)     {
    Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(MyScan.class.getName());

    String[] basePackages = ( String[])annotationAttributes.get("value");
    /**
     * 拿到注解上的扫描的包的名称
     * com.mi.asop.demo.mapper
     */
    String packageName =  basePackages[0];
    /**
     * 得到包下面所有的类
     */
    String packagePath = "";
    String[] names = packageName.split("\\.");
    for(String name:names){
        packagePath+=name+"/";
    }

    String classPath = this.getClass().getResource("/").getPath()+packagePath;
    File file = new File(classPath);
    String[] list = file.list();
    for(String fileName:list){
        //com.mi.asop.demo.mapper.userMapper
        fileName = fileName.substring(0,fileName.lastIndexOf("."));
        String first = fileName.substring(0, 1);
        String after = fileName.substring(1);
        first = first.toLowerCase();
        String beanName = first+after;
          BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(MyMapperFactoryBean.class);
        AbstractBeanDefinition beanDefinition = builder.getBeanDefinition();
        beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(packageName+"."+fileName);
        beanDefinitionRegistry.registerBeanDefinition(beanName,beanDefinition);
    }
}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MyBatis 是一种基于 Java 的持久化框架,它的主要作用是简化与数据库的交互。MyBatis 工作原理如下: 1. 配置文件解析:MyBatis 配置文件是一个 XML 文件,它包含了数据库连接信息、Mapper 映射文件信息等。在 MyBatis 启动时,会将配置文件读取到内存中,并解析成相应的对象。 2. SqlSession 的创建:SqlSessionMyBatis 中用于与数据库进行交互的核心类,它封装了 JDBC 操作。在应用程序中需要执行 SQL 语句时,首先需要创建 SqlSession 对象。 3. Mapper 映射文件解析:Mapper 映射文件是 MyBatis 中用于定义 SQL 语句的 XML 文件,它包含了 SQL 语句、参数信息、返回值信息等。在 SqlSession 中执行 SQL 语句时,会根据 Mapper 映射文件中定义的 SQL 语句进行操作。 4. SQL 语句执行:当 SqlSession 接收到应用程序传递的 SQL 语句后,会根据 Mapper 映射文件中定义的 SQL 语句进行操作,包括 SQL 语句的解析、参数绑定、SQL 执行等。 5. 结果集映射:当 SQL 语句执行完毕后,MyBatis 会将查询结果映射成 Java 对象,并返回给应用程序。MyBatis 支持将结果集映射为单个 Java 对象、Java 对象列表、Map 等。 6. 事务管理:在 MyBatis 中,事务是通过 SqlSession 进行管理的。当应用程序需要执行一系列 SQL 语句时,可以通过 SqlSession 对象开启事务,执行完毕后再提交或回滚事务。 总的来说,MyBatis工作原理主要包括配置文件解析、SqlSession 的创建、Mapper 映射文件解析、SQL 语句执行、结果集映射和事务管理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值