久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

(三)MyBatis從入門到入土——使用詳解

 Coder編程 2021-05-05


MyBatis使用詳解

上篇我們手動開發(fā)了一個MyBatis項目,,但是我們僅僅是編寫了代碼,對于整個項目是如何運行以及每個代碼的意義都沒有仔細(xì)的分析和說明,,那么接下來我們就開始分析每個代碼的意義以及如何編寫這個代碼

配置MyBatis全局配置文件

要使用Mybatis來操作數(shù)據(jù)庫,,那么當(dāng)然就需要配置數(shù)據(jù)庫相關(guān)信息,這件需要在mybatis全局配置文件中進行,。即全局配置的xml文件,,其對整個MyBatis進行事務(wù)的支持、數(shù)據(jù)庫的配置等信息的配置,。我們一般放在main/resource文件中,,如下所示

<configuration>
    <!-- 環(huán)境配置,可以配置多個環(huán)境 -->
    <environments default="chat01">
        <!--
            environment用來對某個環(huán)境進行配置
            id:環(huán)境標(biāo)識,,唯一
         -->
        <environment id="chat01">
            <!-- 事務(wù)管理器工廠配置 -->
            <transactionManager type="org.apache.ibatis.transaction.jdbc.JdbcTransactionFactory"/>
            <!-- 數(shù)據(jù)源工廠配置,,使用工廠來創(chuàng)建數(shù)據(jù)源 -->
            <dataSource type="org.apache.ibatis.datasource.pooled.PooledDataSourceFactory">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost:3306/mybatisdemo?characterEncoding=UTF-8"/>
                <property name="username" value="root"/>
                <property name="password" value="123456"/>
            </dataSource>
        </environment>
    </environments>
</configuration>

configuration元素

這個是mybatis全局配置文件的根元素,每個配置文件只有一個

environments元素

用來配置mybatis的環(huán)境信息,,用來配置多個環(huán)境的,,具體的一個環(huán)境使用environment元素進行配置,environment元素有個id用來標(biāo)識某個具體的環(huán)境,。

environments元素有個default屬性,,用來指定默認(rèn)使用哪個環(huán)境,如上面默認(rèn)使用的是chat01,。

environment元素

用來配置具體的環(huán)境信息,,這個元素下面有兩個子元素:transactionManager和dataSource

  • transactionManager元素

用來配置事務(wù)工廠的,有個type屬性,,type的值必須是org.apache.ibatis.transaction.TransactionFactory接口的實現(xiàn)類,,用來創(chuàng)建事務(wù)管理器對象的,TransactionFactory接口默認(rèn)有2個實現(xiàn):

org.apache.ibatis.transaction.managed.ManagedTransactionFactory
org.apache.ibatis.transaction.jdbc.JdbcTransactionFactory

一般情況下我們使用org.apache.ibatis.transaction.jdbc.JdbcTransactionFactory這個,,mybatis和其他框架集成,,比如和spring集成,,事務(wù)交由spring去控制。

  • dataSource元素

這個用來配置數(shù)據(jù)源的,,type屬性的值必須為接口org.apache.ibatis.datasource.DataSourceFactory的實現(xiàn)類,,DataSourceFactory也是一個工廠,用來創(chuàng)建數(shù)據(jù)源javax.sql.DataSource對象的,,mybatis中這個接口默認(rèn)有3個實現(xiàn)類:

org.apache.ibatis.datasource.jndi.JndiDataSourceFactory
org.apache.ibatis.datasource.pooled.PooledDataSourceFactory
org.apache.ibatis.datasource.unpooled.UnpooledDataSourceFactory

我們使用第2個org.apache.ibatis.datasource.pooled.PooledDataSourceFactory,,這個用來創(chuàng)建一個數(shù)據(jù)庫連接池類型的數(shù)據(jù)源,可以實現(xiàn)數(shù)據(jù)庫連接共用,,減少連接重復(fù)創(chuàng)建銷毀的時間。
配置數(shù)據(jù)源需要指定數(shù)據(jù)庫連接的屬性信息,,比如:驅(qū)動,、連接db的url、用戶名,、密碼,,這個在dataSource元素下面的property中配置,property元素的格式:

<property name="屬性名稱" value="值"/>

創(chuàng)建Mapper xml文件

在mybatis中一般我們將一個表的所有sql操作寫在一個mapper xml中,,一般命名為XXXMapper.xml格式,。

內(nèi)容如下:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-////DTD Mapper 3.0//EN"
        "http:///dtd/mybatis-3-mapper.dtd">
<mapper namespace="zhonghu.mybatis.chat01.UserMapper">
</mapper>

mapper xml根元素為mapper,這個元素有個namespace屬性,,系統(tǒng)中會有很多表,,每個表對應(yīng)一個Mapper xml,為了防止mapper文件重復(fù),,我們需要給每個mapper xml文件需要指定一個namespace,,通過這個可以區(qū)分每個mapper xml文件,上面我們指定為zhonghu.mybatis.chat01.UserMapper,。

一會對user表的所有操作相關(guān)的sql,,我們都會寫在上面這個xml中。

mybatis全局配置文件中引入Mapper xml文件

user.xml我們寫好了,,如何讓mybatis知道這個文件呢,,此時我們需要在mybatis-config.xml全局配置文件中引入UserMapper.xml,在mybatis-config.xml加入下面配置:

<mappers>
        <mapper resource="mapper/user.xml"/>
    </mappers>

mappers元素下面有多個mapper元素,,通過mapper元素的resource屬性可以引入Mapper xml文件,,resource是相對于classes的路徑。

上面說的都是一些配置文件,,配置文件都o(jì)k了,,下面我們就需要將mybatis跑起來了,此時需要使用到mybatis中的一些java對象了,。

構(gòu)建SqlSessionFactory對象

//指定mybatis全局配置文件
String resource = "mybatis-config.xml";
//讀取全局配置文件
InputStream inputStream = Resources.getResourceAsStream(resource);
//構(gòu)建SqlSessionFactory對象
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

SqlSessionFactory是一個接口,,是一個重量級的對象,,SqlSessionFactoryBuilder通過讀取全局配置文件來創(chuàng)建一個SqlSessionFactory,創(chuàng)建這個對象是比較耗時的,,主要耗時在對mybatis全局配置文件的解析上面,,全局配置文件中包含很多內(nèi)容,SqlSessionFactoryBuilder通過解析這些內(nèi)容,,創(chuàng)建了一個復(fù)雜的SqlSessionFactory對象,,這個對象的生命周期一般和應(yīng)用的生命周期是一樣的,隨著應(yīng)用的啟動而創(chuàng)建,,隨著應(yīng)用的停止而結(jié)束,,所以一般是一個全局對象,一般情況下一個db對應(yīng)一個SqlSessionFactory對象,。

構(gòu)建SqlSession對象

SqlSession相當(dāng)于jdbc中的Connection對象,,相當(dāng)于數(shù)據(jù)庫的一個連接,可以用SqlSession來對db進行操作:如執(zhí)行sql,、提交事務(wù),、關(guān)閉連接等等,需要通過SqlSessionFactory來創(chuàng)建SqlSession對象,,SqlSessionFactory中常用的有2個方法來創(chuàng)建SqlSession對象,,如下:

//創(chuàng)建一個SqlSession,默認(rèn)不會自動提交事務(wù)
SqlSession openSession();
//創(chuàng)建一個SqlSession,autoCommit:指定是否自動提交事務(wù)
SqlSession openSession(boolean autoCommit);

SqlSession接口中很多方法,,直接用來操作db,,方法清單如下,大家眼熟一下:

<T> T selectOne(String statement);
<T> T selectOne(String statement, Object parameter);
<E> List<E> selectList(String statement);
<E> List<E> selectList(String statement, Object parameter);
<E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds);
<K, V> Map<K, V> selectMap(String statement, String mapKey);
<K, V> Map<K, V> selectMap(String statement, Object parameter, String mapKey);
<K, V> Map<K, V> selectMap(String statement, Object parameter, String mapKey, RowBounds rowBounds);
<T> Cursor<T> selectCursor(String statement);
<T> Cursor<T> selectCursor(String statement, Object parameter);
<T> Cursor<T> selectCursor(String statement, Object parameter, RowBounds rowBounds);
void select(String statement, Object parameter, ResultHandler handler);
void select(String statement, ResultHandler handler);
void select(String statement, Object parameter, RowBounds rowBounds, ResultHandler handler);
int insert(String statement);
int insert(String statement, Object parameter);
int update(String statement);
int update(String statement, Object parameter);
int delete(String statement);
int delete(String statement, Object parameter);
void commit();
void commit(boolean force);
void rollback();
void rollback(boolean force);
List<BatchResult> flushStatements();
void close();
void clearCache();
Configuration getConfiguration();
<T> T getMapper(Class<T> type);
Connection getConnection();

上面以select開頭的可以對db進行查詢操作,,insert相關(guān)的可以對db進行插入操作,,update相關(guān)的可以對db進行更新操作。

引入lombok支持(非必須)

Lombok能以簡單的注解形式來簡化java代碼,,提高開發(fā)人員的開發(fā)效率,。例如開發(fā)中經(jīng)常需要寫的javabean,都需要花時間去添加相應(yīng)的getter/setter,,也許還要去寫構(gòu)造器,、equals等方法,而且需要維護,,當(dāng)屬性多時會出現(xiàn)大量的getter/setter方法,,這些顯得很冗長也沒有太多技術(shù)含量,一旦修改屬性,,就容易出現(xiàn)忘記修改對應(yīng)方法的失誤,。

Lombok能通過注解的方式,在編譯時自動為屬性生成構(gòu)造器、getter/setter,、equals,、hashcode、toString方法,。出現(xiàn)的神奇就是在源碼中沒有g(shù)etter和setter方法,,但是在編譯生成的字節(jié)碼文件中有g(shù)etter和setter方法。這樣就省去了手動重建這些代碼的麻煩,,使代碼看起來更簡潔些,。

lombok的使用步驟

  • 先在idea中安裝lombok插件
    打開idea,點擊File->Settings->plugins,,然后搜索Lombok Plugin,,點擊安裝就可以了。

  • maven中引入lombok支持

<dependency>
   <groupId>org.projectlombok</groupId>
   <artifactId>lombok</artifactId>
   <version>1.18.10</version>
   <scope>provided</scope>
</dependency>
  • 代碼中使用lombok相關(guān)功能

引入logback(非必須)

為了方便查看mybatis運行過程中產(chǎn)生的日志,,比如:執(zhí)行的sql,、sql的參數(shù)、sql的執(zhí)行結(jié)果等等調(diào)試信息,,我們需要引入日志框架的支持,logback是一個很好的日志框架,,此處我們就使用這個

mybatis中集成logback步驟

  • maven中引入logback支持

<dependency>
   <groupId>ch.qos.logback</groupId>
   <artifactId>logback-classic</artifactId>
   <version>1.2.3</version>
</dependency>
  • src/main/resources中創(chuàng)建logback.xml文件:

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
   <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
       <encoder>
           <pattern>%d{mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
       </encoder>
   </appender>
   <logger name="zhonghu" level="debug" additivity="false">
       <appender-ref ref="STDOUT" />
   </logger>
</configuration>

logback.xml具體的寫法不是本文討論的范圍,,有興趣的朋友可以去研究一下logback具體的用法。

寫一個測試用例

上面說了這么多,,下面我們就寫一個測試類來演示一下

內(nèi)容如下:

package zhonghu.mybatis.chat01;

import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;
import java.io.InputStream;

@Slf4j
public class UserMapperTest {
    private SqlSessionFactory sqlSessionFactory;
    @Before
    public void before() throws IOException {
        //指定mybatis全局配置文件
        String resource = "mybatis-config.xml";
        //讀取全局配置文件
        InputStream inputStream = Resources.getResourceAsStream(resource);
        //構(gòu)建SqlSessionFactory對象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        this.sqlSessionFactory = sqlSessionFactory;
    }
    @Test
    public void sqlSession() {
        SqlSession sqlSession = this.sqlSessionFactory.openSession();
        log.info("{}", sqlSession);
    }

}

上面代碼中有個@Slf4j注解,,這個是lombok提供的,可以在這個類中生成下面代碼:

private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(UserTest.class);

運行一下上面的用例:sqlSession方法,,輸出如下:

37:52.473 [main] INFO  z.mybatis.chat01.UserMapperTest - org.apache.ibatis.session.defaults.DefaultSqlSession@2d127a61

至此我們就搭建了一個最小化的Mybatis項目,,后面就需要我們根據(jù)需要編寫我們的sql文件

使用SqlSesion執(zhí)行sql操作

SqlSession常見的用法

SqlSession相當(dāng)于一個連接,可以使用這個對象對db執(zhí)行增刪改查操作,,操作完畢之后需要關(guān)閉,,使用步驟:

  • 獲取SqlSession對象:通過該sqlSessionFactory.openSession方法獲取SqlSession對象

  • 對db進行操作:使用SqlSession對象進行db操作

  • 關(guān)閉SqlSession對象:sqlSession.close();

如下所示

//獲取SqlSession
SqlSession sqlSession = this.sqlSessionFactory.openSession();
try {
    //執(zhí)行業(yè)務(wù)操作,如:增刪改查
} finally {
    //關(guān)閉SqlSession
    sqlSession.close();
}

上面我們將SqlSession的關(guān)閉放在finally塊中,,確保close()一定會執(zhí)行,。

新增操作

需求:傳入UserModel對象,然后將這個對象的數(shù)據(jù)插入到user表中,。

創(chuàng)建一個UserModel

UserModel類,,代碼如下:

package zhonghu.mybatis.chat01;

import lombok.*;

@Getter
@Setter
@NoArgsConstructor
@AllArgsConstructor
@Builder
@ToString
public class UserModel {
    private Long id;
    private String name;
    private Integer age;
    private Double salary;
}

這個類的字段和user表對應(yīng)。

user.xml中定義插入操作

我們說過了,,對user表的所有sql操作,,我們都放在user.xml中,我們在user.xml中加入下面配置,使用insert元素定義插入操作:

<!-- insert用來定義一個插入操作
     id:操作的具體標(biāo)識
     parameterType:指定插入操作接受的參數(shù)類型
 -->
<insert id="insertUser" parameterType="zhonghu.mybatis.chat01.UserModel">
    <![CDATA[
    INSERT INTO user (id,name,age,salary) VALUES (#{id},#{name},#{age},#{salary})
     ]]>
</insert>
  • insert元素用來定義了一個對db的insert操作

  • id:是這個操作的一個標(biāo)識,,一會通過mybatis執(zhí)行操作的時候會通過這個namespace和id引用到這個insert操作,,

  • parameterType:用來指定這個insert操作接受的參數(shù)的類型,可以是:各種javabean,、map,、list、collection類型的java對象,,我們這個插入接受的是UserModel對象,。

  • insert元素內(nèi)部定義了具體的sql,可以看到是一個insert的sql,,向user表插入數(shù)據(jù),。

需要插入的值從UserModel對象中獲取,取UserModel對象的的字段,,使用#{字段}這種格式可以獲取到UserModel中字段的值,。

調(diào)用SqlSession.insert方法執(zhí)行插入操作

user插入的sql我們已經(jīng)在UserMapper中寫好,此時我們怎么調(diào)用呢,?

需要調(diào)用SqlSession.insert方法:

int insert(String statement, Object parameter)

這個方法有2個參數(shù):

  • statement:表示那個操作,,值為Mapper xml的namespace.具體操作的id,如需要調(diào)用UserMapper.xml中的insertUser操作,,這個值就是:

zhonghu.mybatis.chat01.UserMapper.insertUser
  • parameter:insert操作的參數(shù),,和Mapper xml中的insert中的parameterType指定的類型一致。

返回值為插入的行數(shù),。

UserTest類中新增一個測試用例:

  @Test
    public void insertUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(false);) {
            //創(chuàng)建UserModel對象
            UserModel userModel = UserModel.builder().id(69L).name("Java冢狐").age(30).salary(50000D).build();
            //執(zhí)行插入操作
            int result = sqlSession.insert("zhonghu.mybatis.chat01.UserMapper.insertUser", userModel);
            log.info("插入影響行數(shù):{}", result);
            //提交事務(wù)
            sqlSession.commit();
        }
    }

運行輸出如下:

05:15.831 [main] DEBUG z.m.chat01.UserMapper.insertUser - ==>  Preparing: INSERT INTO user (id,name,age,salary) VALUES (?,?,?,?) 
05:15.853 [main] DEBUG z.m.chat01.UserMapper.insertUser - ==> Parameters: 69(Long), Java冢狐(String), 30(Integer), 50000.0(Double)
05:15.951 [main] DEBUG z.m.chat01.UserMapper.insertUser - <==    Updates: 1
05:15.952 [main] INFO  z.mybatis.chat01.UserMapperTest - 插入影響行數(shù):1

輸出中打印了詳細(xì)的sql語句,,以及sql的參數(shù)信息,可以看到Mapper xml中的#{}被替換為了?,,這個使用到了jdbc中的PreparedStatement來對參數(shù)設(shè)置值,。

輸出中的第二行詳細(xì)列出了參數(shù)的值以及每個值的類型。

第三行輸出了insert的結(jié)果為1,,表示插入成功了1行記錄,。

上面代碼中創(chuàng)建SqlSession,我們使用的是sqlSessionFactory.openSession()創(chuàng)建的,,這個方法創(chuàng)建的SqlSession,,內(nèi)部事務(wù)是非自動提交的方式,所以需要我們手動提交:

增,、刪,、改操作都需要提交事務(wù)

sqlSession.commit();

如果想自動提交事務(wù),在上面在創(chuàng)建SqlSession的時候改為sqlSessionFactory.openSession(true),,指定事務(wù)為自動提交模式,,所以最后我們不需要手動提交事務(wù)了。

更新操作

需求:傳入UserModel對象,然后通過id更新數(shù)據(jù),。

UserMapper.xml中定義Update操作

使用update定義更新操作:

<!-- update用來定義一個更新操作
     id:操作的具體標(biāo)識
     parameterType:指定操作接受的參數(shù)類型
 -->
    <update id="updateUser" parameterType="zhonghu.mybatis.chat01.UserModel">
    <![CDATA[
    UPDATE user SET name = #{name},age = #{age},salary = #{salary} WHERE id = #{id}
    ]]>
</update>

寫法和insert操作的寫法類似,,指定id標(biāo)識、parameterType指定操作的參數(shù)類型,,元素體中是具體的sql語句,。

調(diào)用SqlSession.update方法執(zhí)行更新操作

需要調(diào)用SqlSession.update方法:

int update(String statement, Object parameter)

這個方法有2個參數(shù):

  • statement:表示哪個操作,值為Mapper xml的namespace.具體操作的id,,如需要調(diào)用UserMapper.xml中的updateUser操作,,這個值就是:

zhonghu.mybatis.chat01.UserMapper.updateUser
  • parameter:update操作的參數(shù),和Mapper xml中的update中的parameterType指定的類型一致,。

返回值為update影響行數(shù),。

UserTest類中新增一個測試用例:

    @Test
    public void updateUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            //創(chuàng)建UserModel對象
            UserModel userModel = UserModel.builder().id(1L).name("Java冢狐,你好").age(18).salary(5000D).build();
            //執(zhí)行更新操作
            int result = sqlSession.update("zhonghu.mybatis.chat01.UserMapper.updateUser", userModel);
            log.info("影響行數(shù):{}", result);
        }
    }

運行輸出:

12:17.143 [main] DEBUG z.m.chat01.UserMapper.updateUser - ==>  Preparing: UPDATE user SET name = ?,age = ?,salary = ? WHERE id = ? 
12:17.163 [main] DEBUG z.m.chat01.UserMapper.updateUser - ==> Parameters: Java冢狐,,你好(String), 18(Integer), 5000.0(Double), 1(Long)
12:17.258 [main] DEBUG z.m.chat01.UserMapper.updateUser - <==    Updates: 1
12:17.258 [main] INFO  z.mybatis.chat01.UserMapperTest - 影響行數(shù):1

刪除操作

需求:根據(jù)用戶的id刪除對應(yīng)的用戶記錄

UserMapper.xml中定義Delete操作

使用update元素定義刪除操作:

<!-- update用來定義一個刪除操作
     id:操作的具體標(biāo)識
     parameterType:指定操作接受的參數(shù)類型
 -->
<update id="deleteUser" parameterType="java.lang.Long">
    <![CDATA[
    DELETE FROM user WHERE id = #{id}
    ]]>
</update>

寫法和update操作的寫法類似,,指定id標(biāo)識、parameterType指定操作的參數(shù)類型,,用戶id為Long類型的,,元素體中是具體的delete語句。

調(diào)用SqlSession.update方法執(zhí)行更新操作

需要調(diào)用SqlSession.delete方法:

int delete(String statement, Object parameter)

這個方法有2個參數(shù):

  • statement:表示哪個操作,,值為Mapper xml的namespace.具體操作的id,,如需要調(diào)用UserMapper.xml中的deleteUser操作,這個值就是:

com.javacode2018.chat02.UserMapper.
  • parameter:delete操作的參數(shù),,和Mapper xml中的delete中的parameterType指定的類型一致。

返回值為delete影響行數(shù),。

UserTest類中新增一個測試用例:

    @Test
    public void deleteUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            //定義需要刪除的用戶id
            Long userId = 1L;
            //執(zhí)行刪除操作
            int result = sqlSession.delete("zhonghu.mybatis.chat01.UserMapper.deleteUser", userId);
            log.info("影響行數(shù):{}", result);
        }
    }

運行輸出:

14:26.711 [main] DEBUG z.m.chat01.UserMapper.deleteUser - ==>  Preparing: DELETE FROM user WHERE id = ? 
14:26.729 [main] DEBUG z.m.chat01.UserMapper.deleteUser - ==> Parameters: 1(Long)
14:26.811 [main] DEBUG z.m.chat01.UserMapper.deleteUser - <==    Updates: 1
14:26.812 [main] INFO  z.mybatis.chat01.UserMapperTest - 影響行數(shù):1

執(zhí)行查詢

select語句有恩多屬性可以詳細(xì)的配置每一條SQL語句

  • SQL語句返回值類型【完整的類名或者別名】

  • 傳入SQL語句的參數(shù)類型【建議使用萬能的map】

  • 命名空間中的唯一的標(biāo)識符

  • 接口中的方法名與映射文件中的SQL語句ID一一對應(yīng)

需求:查詢所有用戶信息

UserMapper.xml中定義Select操作

    <!-- select用來定義一個查詢操作
     id:操作的具體標(biāo)識
     resultType:指定查詢結(jié)果保存的類型
 -->
    <select id="getUserList" resultType="zhonghu.mybatis.chat01.UserModel">
    <![CDATA[
    SELECT * FROM user
    ]]>
</select>

寫法和update操作的寫法類似,,指定id標(biāo)識、parameterType指定操作的參數(shù)類型,,resultType指定查詢結(jié)果的類型,,元素體中是具體的select語句。

調(diào)用SqlSession.select方法執(zhí)行更新操作

UserTest添加一個用例:

  @Test
    public void getUserList() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            //執(zhí)行查詢操作
            List<UserModel> userModelList = sqlSession.selectList("zhonghu.mybatis.chat01.UserMapper.getUserList");
            log.info("結(jié)果:{}", userModelList);
        }
    }

多插入幾行,,然后運行上面的用例,,輸出如下:

16:00.798 [main] DEBUG z.m.chat01.UserMapper.getUserList - ==>  Preparing: SELECT * FROM user 
16:00.817 [main] DEBUG z.m.chat01.UserMapper.getUserList - ==> Parameters: 
16:00.829 [main] DEBUG z.m.chat01.UserMapper.getUserList - <==      Total: 16
16:00.829 [main] INFO  z.mybatis.chat01.UserMapperTest - 結(jié)果:[UserModel(id=2, name=修改冢狐, age=23, salary=50000.0), UserModel(id=3, name=修改冢狐, age=24, salary=6666.66), UserModel(id=4, name=Mybatis-1, age=19, salary=10000.0), UserModel(id=5, name=Java冢狐-2, age=25, salary=20000.0), UserModel(id=6, name=Mybatis-2, age=20, salary=20000.0), UserModel(id=7, name=Java冢狐-3, age=26, salary=30000.0), UserModel(id=8, name=Mybatis-3, age=21, salary=30000.0), UserModel(id=9, name=Java冢狐-4, age=27, salary=40000.0), UserModel(id=10, name=Mybatis-4, age=22, salary=40000.0), UserModel(id=11, name=Java冢狐-5, age=28, salary=50000.0), UserModel(id=12, name=Mybatis-5, age=23, salary=50000.0), UserModel(id=13, name=Java冢狐, age=1, salary=0.0), UserModel(id=14, name=冢狐, age=23, salary=50000.0), UserModel(id=59, name=Java冢狐, age=30, salary=50000.0), UserModel(id=69, name=Java冢狐, age=30, salary=50000.0), UserModel(id=89, name=Java冢狐, age=30, salary=50000.0)]

Mapper接口的使用

為什么需要Mapper接口

上面我們講解了對一個表的增刪改查操作,都是通過調(diào)用SqlSession中的方法來完成的,,大家再來看一下SqlSession接口中剛才用到的幾個方法的定義:

int insert(String statement, Object parameter);
int update(String statement, Object parameter);
int delete(String statement, Object parameter);
<E> List<E> selectList(String statement);

這些方法的特點我們來看一下:

  • 調(diào)用這些方法,,需要明確知道statement的值,statement的值為namespace.具體操作的id,,這些需要打開Mapper xml中去查看了才知道,,寫起來不方便

  • parameter參數(shù)都是Object類型的,我們根本不知道這個操作具體類型是什么,需要查看Mapper xml才知道,,隨便傳遞個值,,可能類型不匹配,但是只有在運行的時候才知道有問題

  • selectList方法返回的是一個泛型類型的,,通過這個方法我們根本不知道返回的結(jié)果的具體類型,,也需要去查看Mapper xml才知道

這些都是使用過程中不方便的情況,想要方便的使用,,就需要用到mybatis中的Mapper接口,,我們可以定義一個interface,然后和Mapper xml關(guān)聯(lián)起來,,Mapper xml中的操作和Mapper接口中的方法會進行綁定,,當(dāng)我們調(diào)用Mapper接口的方法的時候,會間接調(diào)用到Mapper xml中的操作,,接口的完整類名需要和Mapper xml中的namespace一致,。

Mapper接口的用法(三步)

步驟1:定義Mapper接口

去看一下,user.xml中的namespace,,是:

<mapper namespace="zhonghu.mybatis.chat01.UserMapper">

我們創(chuàng)建的接口完整的名稱需要和上面的namespace的值一樣,,下面我們創(chuàng)建一個接口UserMapper,如下:

UserMapper.xml中有4個操作,,我們需要在UserMapper接口中也定義4個操作,,和UserMapper.xml的4個操作對應(yīng),如下:

package zhonghu.mybatis.chat01;
import java.util.List;

public interface UserMapper {
    int insertUser(UserModel model);
    int updateUser(UserModel model);
    int deleteUser(Long userId);
    List<UserModel> getUserList();
}

UserMapper接口中定義了4個方法,,方法的名稱需要和UserMapper.xml具體操作的id值一樣,,這樣調(diào)用UserMapper接口中的方法的時候,才會對應(yīng)的找到UserMapper.xml中具體的操作,。

比如調(diào)用UserMapper接口中的insertUser方法,,mybatis查找的規(guī)則是:通過接口完整名稱.方法名稱去Mapper xml中找到對應(yīng)的操作。

步驟2:通過SqlSession獲取Mapper接口對象

SqlSession中有個getMapper方法,,可以傳入接口的類型,,獲取具體的Mapper接口對象,如下:

/**
   * Retrieves a mapper.
   * @param <T> the mapper type
   * @param type Mapper interface class
   * @return a mapper bound to this SqlSession
   */
  <T> T getMapper(Class<T> type);

如獲取UserMapper接口對象:

UserMapper mapper = sqlSession.getMapper(UserMapper.class);

步驟3:調(diào)用Mapper接口的方法對db進行操作

如調(diào)用UserMapper接口的insert操作:

@Test
public void insertUser() {
    try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        //創(chuàng)建UserModel對象
        UserModel userModel = UserModel.builder().id(System.currentTimeMillis()).name("Java冢狐").age(30).salary(50000D).build();
        //執(zhí)行插入操作
        int insert = mapper.insertUser(userModel);
        log.info("影響行數(shù):{}", insert);
    }
}

案例:使用Mapper接口來實現(xiàn)增刪改查

創(chuàng)建一個測試類,,代碼如下:

package zhonghu.mybatis.chat01;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Before;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;

@Slf4j
public class UserMapperTest {
    private SqlSessionFactory sqlSessionFactory;
    @Before
    public void before() throws IOException {
        //指定mybatis全局配置文件
        String resource = "mybatis-config.xml";
        //讀取全局配置文件
        InputStream inputStream = Resources.getResourceAsStream(resource);
        //構(gòu)建SqlSessionFactory對象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        this.sqlSessionFactory = sqlSessionFactory;
    }
    @Test
    public void insertUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            UserMapper mapper = sqlSession.getMapper(UserMapper.class);
            //創(chuàng)建UserModel對象
            UserModel userModel = UserModel.builder().id(System.currentTimeMillis()).name("Java冢狐").age(30).salary(50000D).build();
            //執(zhí)行插入操作
            int insert = mapper.insertUser(userModel);
            log.info("影響行數(shù):{}", insert);
        }
    }
    @Test
    public void updateUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            UserMapper mapper = sqlSession.getMapper(UserMapper.class);
            //創(chuàng)建UserModel對象
            UserModel userModel = UserModel.builder().id(1L).name("Java冢狐,,你好").age(18).salary(5000D).build();
            //執(zhí)行更新操作
            int result = mapper.updateUser(userModel);
            log.info("影響行數(shù):{}", result);
        }
    }
    @Test
    public void deleteUser() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            UserMapper mapper = sqlSession.getMapper(UserMapper.class);
            //定義需要刪除的用戶id
            Long userId = 1L;
            //執(zhí)行刪除操作
            int result = mapper.deleteUser(userId);
            log.info("影響行數(shù):{}", result);
        }
    }
    @Test
    public void getUserList() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            UserMapper mapper = sqlSession.getMapper(UserMapper.class);
            //執(zhí)行查詢操作
            List<UserModel> userModelList = mapper.getUserList();
            userModelList.forEach(item -> {
                log.info("{}", item);
            });
        }
    }
}

大家認(rèn)真看一下上面的代碼,這次我們使用了UserMapper來間接調(diào)用UserMapper.xml中對應(yīng)的操作,,可以去運行一下感受一下效果,。

Mapper接口使用時注意的幾點

  • Mapper接口的完整類名必須和對應(yīng)的Mapper xml中的namespace的值一致

  • Mapper接口中方法的名稱需要和Mapper xml中具體操作的id值一致

  • Mapper接口中方法的參數(shù)、返回值可以不和Mapper xml中的一致

Mapper接口的原理

這個使用java中的動態(tài)代理實現(xiàn)的,,mybatis啟動的時候會加載全局配置文件mybatis-config.xml,,然后解析這個文件中的mapper元素指定的UserMapper.xml,,會根據(jù)UserMapper.xml的namespace的值創(chuàng)建這個接口的一個動態(tài)代理,具體可以去看一下mybatis的源碼,,主要使用java中的Proxy實現(xiàn)的,,使用java.lang.reflect.Proxy類中的newProxyInstance方法,我們可以創(chuàng)建任意一個接口的一個代理對象:

public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)

我們使用Proxy來模仿Mapper接口的實現(xiàn):

package zhonghu.mybatis.chat01;

import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.List;

@Slf4j
public class UserMapperTest {
    public static class UserMapperProxy implements InvocationHandler {
        private SqlSession sqlSession;
        private Class<?> mapperClass;
        public UserMapperProxy(SqlSession sqlSession, Class<?> mapperClass) {
            this.sqlSession = sqlSession;
            this.mapperClass = mapperClass;
        }
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            log.debug("invoke start");
            String statement = mapperClass.getName() + "." + method.getName();
            List<Object> result = sqlSession.selectList(statement);
            log.debug("invoke end");
            return result;
        }
    }
    private SqlSessionFactory sqlSessionFactory;
    @Before
    public void before() throws IOException {
        //指定mybatis全局配置文件
        String resource = "mybatis-config.xml";
        //讀取全局配置文件
        InputStream inputStream = Resources.getResourceAsStream(resource);
        //構(gòu)建SqlSessionFactory對象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        this.sqlSessionFactory = sqlSessionFactory;
    }
    @Test
    public void test1() {
        try (SqlSession sqlSession = this.sqlSessionFactory.openSession(true);) {
            UserMapper userMapper = (UserMapper) Proxy.newProxyInstance(UserMapperTest.class.getClassLoader(), new Class[]{UserMapper.class}, new UserMapperProxy(sqlSession, UserMapper.class));
            log.info("{}", userMapper.getUserList());
        }
    }
}

上面代碼中:UserMapper是沒有實現(xiàn)類的,,可以通過Proxy.newProxyInstance給UserMapper接口創(chuàng)建一個代理對象,,當(dāng)調(diào)用UserMapper接口的方法的時候,會調(diào)用到UserMapperProxy對象的invoke方法,。

運行一下test1用例,,輸出如下:

29:37.847 [main] DEBUG z.m.chat01.UserMapper.getUserList - ==>  Preparing: SELECT * FROM user 
29:37.865 [main] DEBUG z.m.chat01.UserMapper.getUserList - ==> Parameters: 
29:37.878 [main] DEBUG z.m.chat01.UserMapper.getUserList - <==      Total: 16
29:37.878 [main] DEBUG z.mybatis.chat01.UserMapperTest - invoke end
29:37.878 [main] INFO  z.mybatis.chat01.UserMapperTest - [UserModel(id=2, name=修改冢狐, age=23, salary=50000.0), UserModel(id=3, name=修改冢狐, age=24, salary=6666.66), UserModel(id=4, name=Mybatis-1, age=19, salary=10000.0), UserModel(id=5, name=Java冢狐-2, age=25, salary=20000.0), UserModel(id=6, name=Mybatis-2, age=20, salary=20000.0), UserModel(id=7, name=Java冢狐-3, age=26, salary=30000.0), UserModel(id=8, name=Mybatis-3, age=21, salary=30000.0), UserModel(id=9, name=Java冢狐-4, age=27, salary=40000.0), UserModel(id=10, name=Mybatis-4, age=22, salary=40000.0), UserModel(id=11, name=Java冢狐-5, age=28, salary=50000.0), UserModel(id=12, name=Mybatis-5, age=23, salary=50000.0), UserModel(id=13, name=Java冢狐, age=1, salary=0.0), UserModel(id=14, name=冢狐, age=23, salary=50000.0), UserModel(id=59, name=Java冢狐, age=30, salary=50000.0), UserModel(id=69, name=Java冢狐, age=30, salary=50000.0), UserModel(id=89, name=Java冢狐, age=30, salary=50000.0)]

注意上面輸出的invoke start和invoke end,可以看到我們調(diào)用userMapper.getUserList時候,,被UserMapperProxy#invoke方法處理了,。

Mybatis中創(chuàng)建Mapper接口代理對象使用的是下面這個類,大家可以去研究一下:

public class MapperProxyFactory<T> {
  private final Class<T> mapperInterface;
  private final Map<Method, MapperMethod> methodCache = new ConcurrentHashMap<Method, MapperMethod>();
  public MapperProxyFactory(Class<T> mapperInterface) {
    this.mapperInterface = mapperInterface;
  }
  public Class<T> getMapperInterface() {
    return mapperInterface;
  }
  public Map<Method, MapperMethod> getMethodCache() {
    return methodCache;
  }
  @SuppressWarnings("unchecked")
  protected T newInstance(MapperProxy<T> mapperProxy) {
    return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
  }
  public T newInstance(SqlSession sqlSession) {
    final MapperProxy<T> mapperProxy = new MapperProxy<T>(sqlSession, mapperInterface, methodCache);
    return newInstance(mapperProxy);
  }
}

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,,所有內(nèi)容均由用戶發(fā)布,,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式,、誘導(dǎo)購買等信息,,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,,請點擊一鍵舉報,。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多