乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

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

       Coder編程 2021-05-05


      MyBatis使用詳解

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

      配置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屬性,用來指定默認使用哪個環(huán)境,如上面默認使用的是chat01。

      environment元素

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

      • transactionManager元素

      用來配置事務(wù)工廠的,有個type屬性,type的值必須是org.apache.ibatis.transaction.TransactionFactory接口的實現(xiàn)類,用來創(chuàng)建事務(wù)管理器對象的,TransactionFactory接口默認有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中這個接口默認有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,默認不會自動提交事務(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

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

      輸出中的第二行詳細列出了參數(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語句有恩多屬性可以詳細的配置每一條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);
                  });
              }
          }
      }

      大家認真看一下上面的代碼,這次我們使用了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)購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
        轉(zhuǎn)藏 分享 獻花(0

        0條評論

        發(fā)表

        請遵守用戶 評論公約

        類似文章 更多