MyBatis简单实例

时间:2024-04-08 14:36:14

---直接贴代码

(1)User.Java

package me.gacl.domain;

/**

* @author gacl

* users表所对应的实体类

*/

public class User {

//实体类的属性和表的字段名称一一对应

private int id;

private String name;

private int age;

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 int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

@Override

public String toString() {

return "User [id=" + id + ", name=" + name + ", age=" + age + "]";

}  }

(2)UserMapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
 <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
 <!-- 为这个mapper指定一个唯一的namespace,namespace的值习惯上设置成包名+sql映射文件名,这样就能够保证namespace的值是唯一的
 例如namespace="me.gacl.mapping.userMapper"就是me.gacl.mapping(包名)+userMapper(userMapper.xml文件去除后缀)
  -->
 <mapper namespace="me.gacl.mapping.userMapper">
     <!-- 在select标签中编写查询的SQL语句, 设置select标签的id属性为getUser,id属性值必须是唯一的,不能够重复
     使用parameterType属性指明查询时使用的参数类型,resultType属性指明查询返回的结果集类型
     resultType="me.gacl.domain.User"就表示将查询结果封装成一个User类的对象返回
     User类就是users表所对应的实体类
     -->
     <!--
         根据id查询得到一个user对象
      -->
     <select id="getUser" parameterType="int"
         resultType="me.gacl.domain.User">
         select * from t_user where id=#{id}
     </select>
 </mapper>

(3)conf.xml

<?xml version="1.0" encoding="UTF-8"?>
 <!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="oracle.jdbc.driver.OracleDriver" />
                 <property name="url" value="jdbc:oracle:thin:@132.224.24.118:1521:jsfx" />
                 <property name="username" value="sett_analyse" />
                 <property name="password" value="g4el_yj_ha" />
             </dataSource>
         </environment>
     </environments>
     <mappers>
        <!-- 注册userMapper.xml文件,
        userMapper.xml位于me.gacl.mapping这个包下,所以resource写成me/gacl/mapping/userMapper.xml-->
                 <mapper resource="me/gacl/mapping/userMapper.xml"/>
                      </mappers>
    
 </configuration>

(4)Test1.java

package me.gacl.test;

import java.io.IOException; import java.io.InputStream; import java.io.Reader; import me.gacl.domain.User; 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 Test1 {

public static void main(String[] args) throws IOException {

//mybatis的配置文件

String resource = "conf.xml";

//使用类加载器加载mybatis的配置文件(它也加载关联的映射文件)

InputStream is = Test1.class.getClassLoader().getResourceAsStream(resource);

//构建sqlSession的工厂

SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(is);

//使用MyBatis提供的Resources类加载mybatis的配置文件(它也加载关联的映射文件)

//Reader reader = Resources.getResourceAsReader(resource);

//构建sqlSession的工厂

//SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(reader);

//创建能执行映射文件中sql的sqlSession

SqlSession session = sessionFactory.openSession();

/**          * 映射sql的标识字符串,

* me.gacl.mapping.userMapper是userMapper.xml文件中mapper标签的namespace属性的值,

* getUser是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL

*/

String statement = "me.gacl.mapping.userMapper.getUser";//映射sql的标识字符串

//执行查询返回一个唯一user对象的sql

User user = session.selectOne(statement, 1);

System.out.println(user);

} }

Mybatis访问数据库的步骤:

(1)加载配置文件,

(2)获取SqlSessionFactory工厂

(3)获取sqlSession

(4)获取执行器Executor

(5)Mapper statement对象(包含输入输出)

---总结

(1)映射文件中resultType对应的是单条记录的类型,所以就算返回的是一个对象的集合,这里也和返回一个对象的类型一样

(2)${}表示拼接 like '%${value}%',但是这种写法会引起sql注入,例如1=1 or

(3)在配置文件中配置别名typeAlias      mapper.xml中就可以使用别名,批量别名用package

注意:

在配置文件中引用映射文件时有3中方法:

(1)通过Resource加载单个映射文件

(2)通过mapper接口加载单个文件

<mapper class=...../>

(3)批量加载mapper映射文件

<package name=..../>

第(2)(3)中方式规定:指定mapper接口类名和mapper.xml映射文件名保持一致,且在同一个目录下,前提是使用mapper代理的方法

2、mybatis可以针对parameterType或resultType指定的类型定义一些别名,在mapper.xml中通过别名定义,方便开发

默认支持的别名:

类似String  string

Integer integer

单个别名定义:

<typeAliases>

<typeAlias tpye="全路径类名"  alias="别名"/>

</typeAliases>

批量定义别名

<package name=""/>//别名就是类名,大小写都可以

resultMap可以实现延迟加载,resultType无法实现延迟加载。

一对一关联的时候,如果关联表的字段有一样的情况下,<resultMap>标签的列名column写之前,要在select语句中对相同列名定义别名,这是

column用别名代替,不能按照数据库表字段映射;

例如:

<association property="user" javaType="me.gacl.domain.User">
   <id column="user_id" property="id" />
   <result column="userName" property="userName" />
   </association>
   
 </resultMap>
  
   <select id="findCustomeUser"  resultMap="bbb">
  
   select user.id,user.name as userName, customers.id,customers.name,customers.user_id from
   customers,user where customers.user_id = user.id
  
   </select>

如果不加别名,会导致user对象的username属性取值不正确

延迟加载:association、collection自带延迟加载的功能

setting配置:

<settings>
     <setting name="lazyLoadingEnabled" value="true"/>
     <setting name="aggressiveLazyLoading" value="false"/>
     </settings>

java文件中写法:

@Test
 public void queryCustomerUserLazy(){
  
  SqlSession sqlSession = sessionFactory.openSession();
  UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
     Customers customers = userMapper.queryCustomerUserLazy(48);
     //当调用customer.getUser()方法后才可以加载user的值
     User user = customers.getUser();
     System.out.println(user);
  
 }
 Mapper.xml:

<!-- 实现延迟加载 -->
   <resultMap type="me.gacl.domain.Customers" id="cc">
   <id column="id" property="id"/>
   <result column="name" property="name"/>
   <result column="user_id" property="user_id"/>
   <association property="user" javaType="me.gacl.domain.User"
    select="me.gacl.mapping.UserMapper.queryUser" column="user_id">
    </association>
   </resultMap>
  <select id="queryCustomerUserLazy" parameterType="int" resultMap="cc">
  select * from customers where id=#{id}
  </select>
  <select id="queryUser" parameterType="int"  resultMap="dd">
  select * from user where id = #{value}
  </select>
  <resultMap type="me.gacl.domain.User" id="dd">
  <id column="id" property="id"/>
  <result column="username" property="name"/>
  </resultMap>

延迟加载的思考:

不使用mybatis提供的association、collection中延迟加载的功能,如何实现延迟加载呢?

实现方法如下:

定义2个mapper方法,分别调用

什么是查询 缓存:

mybatis提供查询缓存,用户减轻数据的压力,提供数据库性能

mybatis提供一级缓存、二级缓存

修改、添加、删除后commit会清空一级缓存区域

二级缓存是mapper级别的缓存,除了在conf.xml配置文件中添加配置:

<setting name="cacheEnabled" value="true"/>

在映射文件中也要开启:

<mapper namespace="......">

<cache/>//开启本mapper的namespace下的 二级缓存

可能需要对pojo实现序列化 具体不懂

这样配置后再每一个statement中 useCache默认是true 开启二级缓存

刷新缓存:flushCache="true"

总结:一般下执行完commit操作都需要刷新缓存,flushCache=true表示刷新缓存,这样可以避免数据库脏读。