mybatis原始dao開發(fā)和Mapper代理方法開發(fā)

文章項目工程結(jié)構(gòu)見mybatis入門之增刪改查

一 丁恭、SqlSession使用范圍

1.1 SqlSessionFactoryBuilder

通過SqlSessionFactoryBuilder創(chuàng)建會話工廠SqlSessionFactory
將SqlSessionFactoryBuilder當(dāng)成一個工具類使用即可瞎访,不需要使用單例管理SqlSessionFactoryBuilder赔退。
在需要創(chuàng)建SqlSessionFactory時候澜术,只需要new一次SqlSessionFactoryBuilder即可。

1.2 SqlSessionFactory

通過SqlSessionFactory創(chuàng)建SqlSession盹舞,使用單例模式管理sqlSessionFactory(工廠一旦創(chuàng)建像樊,使用一個實例)眷唉。

將來mybatis和spring整合后予颤,使用單例模式管理sqlSessionFactory囤官。

1.3 SqlSession

SqlSession是一個面向用戶(程序員)的接口,可以直接調(diào)用方法執(zhí)行對應(yīng)的sql
SqlSession中提供了很多操作數(shù)據(jù)庫的方法:如:selectOne(返回單個對象)冬阳、selectList(返回單個或多個對象)、党饮。

SqlSession是線程不安全的肝陪,在SqlSesion實現(xiàn)類中除了有接口中的方法(操作數(shù)據(jù)庫的方法)還有數(shù)據(jù)域?qū)傩浴?/p>

SqlSession最佳應(yīng)用場合在方法體內(nèi),定義成局部變量使用刑顺。

二 氯窍、原始dao開發(fā)方法(程序員需要寫dao接口和dao實現(xiàn)類)

2.1 思路

程序員需要寫dao接口和dao實現(xiàn)類。
需要向dao實現(xiàn)類中注入SqlSessionFactory蹲堂,在方法體內(nèi)通過SqlSessionFactory創(chuàng)建SqlSession

2.2 dao接口

在項目下新建package cn.euegene.dao,并且在該包下新建UserDao接口

package cn.eugene.dao;

import cn.eugene.po.User;

public interface UserDao {
    
    public User selectUserById(int id);
    public void insertUser(User user);
    public void deleteUser(int id);

}

2.3 dao接口實現(xiàn)類

在package—— cn.euegene.dao下新建UserDao實現(xiàn)類UserDaoImpl

package cn.eugene.dao;

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;

import cn.eugene.po.User;

public class UserDaoImpl implements UserDao {

   //通過dao向?qū)崿F(xiàn)類中注入SqlSessionFactory
   private SqlSessionFactory sqlSessionFactory;
   public UserDaoImpl(SqlSessionFactory sqlSessionFactory){
       this.sqlSessionFactory =sqlSessionFactory;
   }
   @Override
   public User selectUserById(int id) {
       // TODO Auto-generated method stub
       SqlSession sqlSession = sqlSessionFactory.openSession();
       //執(zhí)行查詢操作
       User user = sqlSession.selectOne("test.findUserById",id);
       //釋放資源
       sqlSession.close();
       return user;
   }

   @Override
   public void insertUser(User user) {
       // TODO Auto-generated method stub
       SqlSession sqlSession = sqlSessionFactory.openSession();
       //執(zhí)行插入操作
       sqlSession.insert("test.insertUser",user);
       //提交代碼
       sqlSession.commit();
       //釋放資源
       sqlSession.close();
       
   }

   @Override
   public void deleteUser(int id) {
       // TODO Auto-generated method stub
       SqlSession sqlSession = sqlSessionFactory.openSession();
       //執(zhí)行刪除操作
       sqlSession.delete("test.updateUser",id);
       //提交代碼
       sqlSession.commit();
       //釋放資源
       sqlSession.close();
   }

}

2.4 測試代碼

項目要引入Junit測試狼讨,測試dao接口的 實現(xiàn)類

package cn.eugene.dao;

import static org.junit.Assert.*;

import java.io.InputStream;

import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Before;
import org.junit.Test;

import cn.eugene.po.User;

public class UserDaoImplTest {
    private SqlSessionFactory sqlSessionFactory;
    @Before
    public void setUp() throws Exception{
        
        //創(chuàng)建sqlSessionFactory
        String resource = "SqlMapConfig.xml";
        InputStream inputstream = Resources.getResourceAsStream(resource);
        //創(chuàng)建會話工廠,傳入mybatis的配置文件信息
        sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputstream);
        
    }

    @Test
    public void testSelectUserById() {
        
        UserDao userDao = new UserDaoImpl(sqlSessionFactory);
        User user = userDao.selectUserById(1);
        System.out.println(user);
        
    }

}

2.5 總結(jié)原始 dao開發(fā)問題

1柒竞、dao接口實現(xiàn)類方法中存在大量模板方法政供,設(shè)想能否將這些代碼提取出來,大大減輕程序員的工作量。

2布隔、調(diào)用sqlsession方法時將statement的id硬編碼了离陶,如sqlSession.selectOne("test.findUserById",id)

3、調(diào)用sqlsession方法時傳入的變量衅檀,由于sqlsession方法使用泛型招刨,即使變量類型傳入錯誤,在編譯階段也不報錯哀军,不利于程序員開發(fā)沉眶。

三、mapper代理方法(程序員只需要編寫mapper接口杉适,想當(dāng)于dao接口)

3.1 思路(mapper代理開發(fā)規(guī)范)

程序員還需要編寫mapper.xml映射文件

程序員編寫mapper接口需要遵循一些開發(fā)規(guī)范沦寂,mybatis可以自動生成mapper接口實現(xiàn)類代理對象。

開發(fā)規(guī)范:

1淘衙、在mapper.xml中namespace等于mapper接口地址

<mapper namespace="cn.eugene.mapper.UserMapper">

2传藏、mapper.java接口中的方法名和mapper.xml中statement的id一致

3、mapper.java接口中的方法輸入?yún)?shù)類型和mapper.xml中statement的parameterType指定的類型一致彤守。

4毯侦、mapper.java接口中的方法返回值類型和mapper.xml中statement的resultType指定的類型一致。

    <select id="findUserById" parameterType="int" resultType="cn.eugene.po.User">
        select * from user where id = #{id}
    </select>

總結(jié):
以上開發(fā)規(guī)范主要是對下邊的代碼進(jìn)行統(tǒng)一生成:

User user = sqlSession.selectOne("test.findUserById", id);
sqlSession.insert("test.insertUser", user);
具垫。侈离。。筝蚕。

3.2 mapper接口

在項目下新建包cn.enugene.mapper卦碾,新建接口UserMapper

package cn.eugene.mapper;

import cn.eugene.po.User;

public interface UserMapper {
    
    public  User findUserById(int id) throws Exception;
    
    public void deleteUser(int id) throws Exception;
    
    public void updateUser(User user) throws Exception;

}

3.3 mapper.xml (sql映射文件)

在config下新建file floder ——mapper,新建userMapper2.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="cn.eugene.mapper.UserMapper">
    <!-- 根據(jù)id查詢用戶信息 -->
    <select id="findUserById" parameterType="int" resultType="cn.eugene.po.User">
        select * from user where id = #{id}
    </select>
    
     <!-- 根據(jù)用戶名查詢用戶信息 -->
     <select id="findUserByName" parameterType="java.lang.String" resultType="cn.eugene.po.User">
        select * from user where userName like '%${value}%'
    </select>
    <insert id="insertUser" parameterType="cn.eugene.po.User">
        insert into user(userName,userAge,userAddress)VALUES(#{userName},#{userAge},#{userAddress})
    </insert>

    <!-- 根據(jù)id刪除用戶 -->
    <delete id="deleteUser" parameterType="int">
        delete from user where id = #{id}
    </delete>
    
    <update id="updateUser" parameterType="cn.eugene.po.User">
        update user set userName=#{userName},userAge=#{userAge},userAddress=#{userAddress} where id= #{id}
    </update>
</mapper>

3.4 在配置文件中加載sql映射文件

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

3.5 測試

同樣使用junit對mapper接口進(jìn)行測試

package cn.eugene.mapper;

import static org.junit.Assert.*;

import java.io.InputStream;

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 cn.eugene.po.User;

public class UserMapperTest {
    private SqlSessionFactory sqlSessionFactory;
    @Before
    public void setUp() throws Exception{
        
        //創(chuàng)建sqlSessionFactory
        String resource = "SqlMapConfig.xml";
        InputStream inputstream = Resources.getResourceAsStream(resource);
        //創(chuàng)建會話工廠起宽,傳入mybatis的配置文件信息
        sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputstream);      
    }

    @Test
    public void testFindUserById() throws Exception {
        
        SqlSession sqlSession = sqlSessionFactory.openSession();
        //mybatis自動生成接口實現(xiàn)類代理對象
        UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
        User user = userMapper.findUserById(1);
        System.out.println(user);
    }

}

四 洲胖、一些問題總結(jié)

4.1 代理對象內(nèi)部調(diào)用selectOne或selectList

如果mapper方法返回單個pojo對象(非集合對象),代理對象內(nèi)部通過selectOne查詢數(shù)據(jù)庫坯沪。

如果mapper方法返回集合對象绿映,代理對象內(nèi)部通過selectList查詢數(shù)據(jù)庫。

4.2 mapper接口方法參數(shù)只能有一個是否影響系統(tǒng) 開發(fā)

mapper接口方法參數(shù)只能有一個腐晾,系統(tǒng)是否不利于擴(kuò)展維護(hù)叉弦。

系統(tǒng) 框架中,dao層的代碼是被業(yè)務(wù)層公用的藻糖。
即使mapper接口只有一個參數(shù)淹冰,可以使用包裝類型的pojo滿足不同的業(yè)務(wù)方法的需求。

注意:持久層方法的參數(shù)可以包裝類型巨柒、map樱拴。凝颇。。疹鳄,service方法中建議不要使用包裝類型(不利于業(yè)務(wù)層的可擴(kuò)展)拧略。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市瘪弓,隨后出現(xiàn)的幾起案子垫蛆,更是在濱河造成了極大的恐慌,老刑警劉巖腺怯,帶你破解...
    沈念sama閱讀 212,029評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件袱饭,死亡現(xiàn)場離奇詭異,居然都是意外死亡呛占,警方通過查閱死者的電腦和手機(jī)虑乖,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,395評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來晾虑,“玉大人疹味,你說我怎么就攤上這事≈钠” “怎么了糙捺?”我有些...
    開封第一講書人閱讀 157,570評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長笙隙。 經(jīng)常有香客問我洪灯,道長,這世上最難降的妖魔是什么竟痰? 我笑而不...
    開封第一講書人閱讀 56,535評論 1 284
  • 正文 為了忘掉前任签钩,我火速辦了婚禮,結(jié)果婚禮上坏快,老公的妹妹穿的比我還像新娘铅檩。我一直安慰自己,他們只是感情好假消,可當(dāng)我...
    茶點故事閱讀 65,650評論 6 386
  • 文/花漫 我一把揭開白布柠并。 她就那樣靜靜地躺著岭接,像睡著了一般富拗。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上鸣戴,一...
    開封第一講書人閱讀 49,850評論 1 290
  • 那天啃沪,我揣著相機(jī)與錄音,去河邊找鬼窄锅。 笑死创千,一個胖子當(dāng)著我的面吹牛缰雇,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播追驴,決...
    沈念sama閱讀 39,006評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼械哟,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了殿雪?” 一聲冷哼從身側(cè)響起暇咆,我...
    開封第一講書人閱讀 37,747評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎丙曙,沒想到半個月后爸业,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,207評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡亏镰,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,536評論 2 327
  • 正文 我和宋清朗相戀三年扯旷,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片索抓。...
    茶點故事閱讀 38,683評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡钧忽,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出逼肯,到底是詐尸還是另有隱情惰瓜,我是刑警寧澤,帶...
    沈念sama閱讀 34,342評論 4 330
  • 正文 年R本政府宣布汉矿,位于F島的核電站崎坊,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏洲拇。R本人自食惡果不足惜奈揍,卻給世界環(huán)境...
    茶點故事閱讀 39,964評論 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望赋续。 院中可真熱鬧男翰,春花似錦、人聲如沸纽乱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,772評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽鸦列。三九已至租冠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間薯嗤,已是汗流浹背顽爹。 一陣腳步聲響...
    開封第一講書人閱讀 32,004評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留骆姐,地道東北人镜粤。 一個月前我還...
    沈念sama閱讀 46,401評論 2 360
  • 正文 我出身青樓捏题,卻偏偏與公主長得像,于是被迫代替她去往敵國和親肉渴。 傳聞我的和親對象是個殘疾皇子公荧,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,566評論 2 349

推薦閱讀更多精彩內(nèi)容