文章項目工程結(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ò)展)拧略。