Mybatis原生dao开发方法实现增删改查

思路

程序员需要写 dao 接口和 dao 实现类

需要向 dao 实现类中注入 SqlSessionFactory,在方法体内通过 SqlSessionFactory

 

项目准备

1、项目结构

① 新建 Java 项目

② 在 src 下新建 com.liuyanzhao.mybatis.po 包(用来放持久化类,和数据库对应)

和 com.liuyanzhao.mybatis.dao 包 (用来放 CURD 实现代码)

③ 在 src 下新建(或者拷贝)Configuration.xml (mybatis全局配置文件)和 userDao.xml (映射文件)

 

2、导入 两个必备的 jar 包

  • mysql-jdbc 驱动包: mysql-connector-java-5.1.41-bin.jar
  • mybatis 包:mybatis-3.4.4.jar

在项目中新建一个 lib 目录,将两个包拷贝到 lib 中,然后将两个包加入环境(build path)中

 

3、新建数据库和数据表

我这里在本地测试,主机 localhost,数据库用户名root,密码为空

① 新建数据库:mybatis

② 新建数据表:user表

表结构如下,五个字段(id,username,gender,birthday,address)

Mybatis原生dao开发方法实现增删改查

 

代码实现

1、Configuration.xml    mybatis的全局配置文件

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE configuration
  3.     PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.     "http://mybatis.org/dtd/mybatis-3-config.dtd">
  5. <configuration>
  6.   <!-- 和Spring整合后,environments配置将废除-->
  7.   <environments default="development">
  8.     <environment id="development">
  9.       <!--使用JDBC事务管理,事务控制由mybatis-->
  10.       <transactionManager type="JDBC">
  11.         <property name="" value=""/>
  12.       </transactionManager>
  13.       <!--数据库连接池-->
  14.       <dataSource type="UNPOOLED">
  15.         <property name="driver" value="com.mysql.jdbc.Driver"/>
  16.         <property name="url" value="jdbc:mysql://localhost:3306/mybatis?useUnicode=true&amp;characterEncoding=utf8"/>
  17.         <property name="username" value="root"/>
  18.         <property name="password" value=""/>
  19.       </dataSource>
  20.     </environment>
  21.   </environments>
  22.   <!--加载映射文件-->
  23.   <mappers>
  24.     <mapper resource="userDao.xml" />
  25.   </mappers>
  26. </configuration>

 

2、userDao.xml   映射文件

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <!--namespace命名空间,作用是对sql进行分类话管理,理解sql,理解sql隔离
  6.     注意:使用mapper代理方法开发,namespace有着特殊的作用
  7. -->
  8. <mapper namespace="test">
  9.     <!--在映射文件中配置很多sql语句-->
  10.     <!--
  11.     id:表示映射文件中的sql
  12.     将sql语句封装到mappedStatement对象中,所以称id为Statement的id
  13.     parameterType:指定输入参数的类型
  14.     resultType:指定就是单条记录所映射的java对象类型
  15.     #{}表示占位符,相当于?
  16.     #{id} 其中id表示接受输入的参数,参数名就是id,如果输入的参数是简单类型,#{}中的参数可以任意,可以value或者其他名称
  17.     ${}:表示拼接sql串,将接受到的参数的内容不加任何修饰直接拼接在sql中
  18.     ${}安全隐患:使用${}拼接sql,引起sql注入
  19.     ${value}:接受输入参数的内容,如果传入类型是简单类型,${}中只有使用value
  20.     -->
  21.     <select id="findUserById" parameterType="int" resultType="com.liuyanzhao.mybatis.po.User">
  22.         SELECT * FROM user WHERE id=#{value}
  23.     </select>
  24.     <!--根据用户名称模糊查询,可能返回多条-->
  25.     <select id="findUserByName" parameterType="java.lang.String" resultType="com.liuyanzhao.mybatis.po.User">
  26.         SELECT * FROM user WHERE username LIKE '%${value}%'
  27.     </select>
  28.     <!--#{}指定pojo的属性名,接收到pojo对象的属性值,Mybatis通过OGNL获取对象的属性值-->
  29.     <!--添加用户-->
  30.     <insert id="insertUser" parameterType="com.liuyanzhao.mybatis.po.User">
  31.      <!--
  32.         将插入数据的主键返回,返回到user对象中
  33.         SELECT LAST_INSERT_ID(); 得到insert进去的主键值,只适用自增主键
  34.     -->
  35.         <selectKey keyProperty="id" order="AFTER" resultType="java.lang.Integer">
  36.             SELECT LAST_INSERT_ID()
  37.         </selectKey>
  38.         INSERT INTO user(username,birthday,gender,address) VALUE(#{username},#{birthday},#{gender},#{address})
  39.     </insert>
  40.     <!--删除用户,根据id-->
  41.     <delete id="deleteUserById" parameterType="java.lang.Integer">
  42.         DELETE FROM user WHERE id=#{id}
  43.     </delete>
  44.     <!--更新用户,根据id
  45.     需要传入用户信息
  46.     parameterType指定user对象,包括id和更新信息。注意:id必须存在
  47.     #{id}:从输入的user对象中获取id的属性值
  48.     -->
  49.     <update id="updateUserById" parameterType="com.liuyanzhao.mybatis.po.User">
  50.         UPDATE user SET username=#{username},birthday=#{birthday},gender=#{gender},address=#{address} WHERE id=#{id}
  51.     </update>
  52. </mapper>

 

3、UserDao.java   dao接口

  1. package com.liuyanzhao.mybatis.dao;
  2. import com.liuyanzhao.mybatis.po.User;
  3. import java.util.List;
  4. /**
  5.  * Created by 言曌 on 2017/8/10.
  6.  */
  7. public interface UserDao {
  8.     //根据id查询用户信息
  9.     public User findUserById(int id) throws Exception;
  10.     //根据username查询用户信息
  11.     public List<User> findUserByName(String username) throws Exception;
  12.     //添加用户信息
  13.     public void insertUser(User user) throws Exception;
  14.     //删除用户信息
  15.     public void deleteUser(int id) throws Exception;
  16.     //更新用户信息
  17.     public void updateUser(User user) throws Exception;
  18. }

 

4、UserDaoImpl.java  dao接口的实现类

  1. package com.liuyanzhao.mybatis.dao;
  2. import com.liuyanzhao.mybatis.po.User;
  3. import org.apache.ibatis.session.SqlSession;
  4. import org.apache.ibatis.session.SqlSessionFactory;
  5. import java.util.List;
  6. /**
  7.  * Created by 言曌 on 2017/8/10.
  8.  */
  9. public class UserDaoImpl implements UserDao{
  10.     //向dao实现类注入SqlSessionFactory
  11.     //这里通过构造方法注入
  12.     private SqlSessionFactory sqlSessionFactory;
  13.     public UserDaoImpl(SqlSessionFactory sqlSessionFactory) {
  14.         this.sqlSessionFactory = sqlSessionFactory;
  15.     }
  16.     @Override
  17.     public User findUserById(int id) throws Exception {
  18.         SqlSession sqlSession = sqlSessionFactory.openSession();
  19.          //执行查询操作,赋值给user对象
  20.         User user = sqlSession.selectOne("test.findUserById",id);
  21.         //释放资源
  22.         sqlSession.close();
  23.         return user;
  24.     }
  25.     @Override
  26.     public List<User> findUserByName(String username) throws Exception {
  27.         SqlSession sqlSession = sqlSessionFactory.openSession();
  28.         //执行查询操作,存储到list中
  29.         List<User> list = sqlSession.selectList("test.findUserByName",username);
  30.         //释放资源
  31.         sqlSession.close();
  32.         return list;
  33.     }
  34.     @Override
  35.     public void insertUser(User user) throws Exception {
  36.         SqlSession sqlSession = sqlSessionFactory.openSession();
  37.         //执行插入操作
  38.         sqlSession.insert("test.insertUser",user);
  39.         //提交事务
  40.         sqlSession.commit();
  41.         //释放资源
  42.         sqlSession.close();
  43.     }
  44.     @Override
  45.     public void deleteUser(int id) throws Exception {
  46.         SqlSession sqlSession = sqlSessionFactory.openSession();
  47.         //执行删除操作
  48.         sqlSession.delete("test.deleteUserById",id);
  49.         //提交事务
  50.         sqlSession.commit();
  51.         //释放资源
  52.         sqlSession.close();
  53.     }
  54.     @Override
  55.     public void updateUser(User user) throws Exception {
  56.         SqlSession sqlSession = sqlSessionFactory.openSession();
  57.         //执行更新操作
  58.         sqlSession.update("test.updateUserById",user);
  59.         //提交事务
  60.         sqlSession.commit();
  61.         //释放资源
  62.         sqlSession.close();
  63.     }
  64. }

 

5、测试类 UserDaoImplTest.java

  1. package com.liuyanzhao.mybatis.test;
  2. import com.liuyanzhao.mybatis.dao.UserDao;
  3. import com.liuyanzhao.mybatis.dao.UserDaoImpl;
  4. import com.liuyanzhao.mybatis.po.User;
  5. import org.apache.ibatis.session.SqlSessionFactory;
  6. import org.apache.ibatis.io.Resources;
  7. import org.apache.ibatis.session.SqlSessionFactoryBuilder;
  8. import org.junit.Before;
  9. import org.junit.Test;
  10. import java.io.InputStream;
  11. import java.util.Date;
  12. import java.util.List;
  13. /**
  14.  * Created by Liu_Yanzhao on 2017/8/10.
  15.  */
  16. public class UserDaoImplTest {
  17.     private SqlSessionFactory sqlSessionFactory;
  18.     @Before
  19.     public void setUp() throws Exception {
  20.         String resource = "Configuration.xml";
  21.         InputStream inputStream = Resources.getResourceAsStream(resource);
  22.         sqlSessionFactory = new SqlSessionFactoryBuilder()
  23.             .build(inputStream);
  24.     }
  25.     //测试,根据id查询用户
  26.     @Test
  27.     public void testFindUserById() throws Exception {
  28.         UserDao userDao = new UserDaoImpl(sqlSessionFactory);
  29.         User user = userDao.findUserById(10);
  30.         System.out.println(user);
  31.     }
  32.     //测试,根据username查询用户
  33.     @Test
  34.     public void testFindUserByName() throws Exception {
  35.         UserDao userDao = new UserDaoImpl(sqlSessionFactory);
  36.         List<User> list = userDao.findUserByName("小明");
  37.         System.out.println(list);
  38.     }
  39.     //测试,添加信息
  40.     @Test
  41.     public void testInsertUser() throws Exception {
  42.         UserDao userDao = new UserDaoImpl(sqlSessionFactory);
  43.         //创建User对象,存储用户信息
  44.         User user = new User();
  45.         user.setUsername("龙小云");
  46.         user.setGender("女");
  47.         user.setBirthday(new Date());
  48.         user.setAddress("江西南昌");
  49.         //执行添加操作
  50.         userDao.insertUser(user);
  51.     }
  52.     //测试,删除信息
  53.     @Test
  54.     public void testDeleteUserByUser() throws Exception {
  55.         UserDao userDao = new UserDaoImpl(sqlSessionFactory);
  56.         //执行删除操作
  57.         userDao.deleteUser(10);
  58.     }
  59.     //测试,更新用户信息
  60.     @Test
  61.     public void testUpdateUser() throws Exception {
  62.         UserDao userDao = new UserDaoImpl(sqlSessionFactory);
  63.         //创建User对象,存储用户信息
  64.         User user = new User();
  65.         user.setId(8);
  66.         user.setUsername("令小冲");
  67.         user.setGender("男");
  68.         user.setBirthday(new Date());
  69.         user.setAddress("华山");
  70.         //执行更新操作
  71.         userDao.updateUser(user);
  72.     }
  73. }

 

总结原生 dao 开发问题

1、dao 接口实现类方法中存在大量模板方法,设想是否能将这些代码提取出来,从而减少程序员的工作量。

 

2、调用 sqlSession 方法时将 statement 的 id 硬编码(硬编码是指将可变变量用一个固定值来代替的方法)了。

如:

  1. User user = sqlSession.selectOne("test.findUserById",id);

在根据id查询用户的方法里,这句里的 "test.findUserById" 是一个固定的值。

 

3、调用 sqlSession 方法时传入的变量,由于 sqlSession 方法使用泛型,即使变量类型传入错误,在编译阶段也不会报错,不利于 程序员开发

如:

  1. User user = sqlSession.selectOne("test.findUserById",id);

还是 2  中的那一行代码,id 处接受一个对象值,但是我们要求是 一个整数,所以当我输入其他的,如字符串类型时,编译时不会报错,运行时才会有。

 

本文链接:https://liuyanzhao.com/5782.html

  • 微信
  • 赶快加我聊天吧
  • weinxin
  • 博客交流群
  • 海纳百川,大家来水
  • weinxin
言曌

发表评论

:?::razz::sad::evil::!::smile::oops::grin::eek::shock::???::cool::lol::mad::twisted::roll::wink::idea::arrow::neutral::cry::mrgreen: