zoukankan      html  css  js  c++  java
  • Mybatis总结

    总结的另外和一个关于Mybatis的博文:http://www.cnblogs.com/DreamDrive/p/4091004.html

     1.namespace的作用

     命名空间除了对sql进行隔离,mybatis中对命名空间有特殊的作用,用于定义mapper接口地址。

     问题:

     没有使用接口编程,java是面向接口编程语言,对数据库的操作应该定义一些操作接口,如:用户添加、用户删除、用户查询等,调用dao接口完成数据库操作。

    上边代码,该调用selectOne、selectList,完全由人工判断不方便,selectList和selectOne方法参数是一个object类型,如果程序员在编码如果设置参数错误不会在编译阶段报错。
    设想:
    能否mybatis封装 方法实现体,
    方法名:和mapper.xml中的statement的id保持一致。
    方法输入参数类型:和mapper.xml中parameterType指定类型一致。
    方法返回值:可以根据dao接口的 返回值的类型决定是调用selectOne还是selectList,类型和mapper.xml中的resulttype类型保持 一致
    采用mybatis提供动态代理方式生成接口实现对象。
    mapper.xml知道mapper.java,对应关系通过namespace进行配置。

    改为mapper 接口实现:

    第一步:定义mapper.xml

    Mapper.xml文件不变还用原来的。

    第二步:定义mapper 接口
    //mapper接口类路径和mapper.xml中的namespace一致
    public interface UserMapper {
        //根据mapper.xml中定义statement来方法
        //方法名:和mapper.xml中的statement的id一致
        //输入参数:和mapper.xml中的parametertype一致
        //输入结果:和mapper.xml中的resulttype一致
        public  User findUserById(int id)throws Exception;
        //自定义条件查询用户信息
        public List<User> findUserList(User user)throws Exception;
        //查询用户列表输出map
        public List<Map> findUserListReturnMap(User user)throws Exception;
        //查询用户列表使用resultmap
        public List<User> findUserListResultMap(User user)throws Exception;
        //插入用户
        public void insertUser(User user)throws Exception;
        //查询用户列表的总数
        public int findUserCount(User user)throws Exception;
        //查询用户传map
        public List<User>  findUserListByMap(Map map)throws Exception;
    }

    接口定义有如下特点:

    1、 Mapper接口方法名和mapper.xml中定义的每个sqlid相同

    2、 Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql parameterType的类型相同

    3、 Mapper接口方法的输出参数类型和mapper.xml中定义的每个sqlresultType的类型相同

    第三步:修改namespace

    Mapper.xml映射文件中的namepace改为如下:

    <mapper namespace="cn.itcast.mybatis.mapper.UserMapper">

    修改后namespace即是mapper接口的地址。

    第四步:通过mapper接口调用statement
    public class UserMapperTest {
        SqlSessionFactory sqlSessionFactory;
        @Before
        public void setUp() throws Exception {
            // 创建会话工厂
            // 创建的会话工厂SqlsessionFactory
            String resource = "SqlMapConfig.xml";
            InputStream inputStream = Resources.getResourceAsStream(resource);
            // 创建的会话工厂SqlsessionFactory
            sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        }
        @Test
        public void testFindUserById() throws Exception {
                SqlSession sqlSession = sqlSessionFactory.openSession();
                //生成mapper接口的代理对象
                UserMapper userMapper =  sqlSession.getMapper(UserMapper.class);
                //调用 mapper接口的方法
                User user = userMapper.findUserById(1);
                System.out.println(user);
        }
        @Test
        public void testInsertUser() throws Exception {
                SqlSession sqlSession = sqlSessionFactory.openSession();
                //生成mapper接口的代理对象
                UserMapper userMapper =  sqlSession.getMapper(UserMapper.class);
                //构造查询条件
                User user_insert  = new User();
                user_insert.setUsername("张三四");
                user_insert.setSex("1");
                //调用 mapper接口的方法
                userMapper.insertUser(user_insert);
                //提交事务
                sqlSession.commit();
                sqlSession.close();
        }    
    }    
     session.getMapper(UserMapper.class)生成一个代理对象作为UserMapper的接口实现对象。

    总结:
    mapper接口动态代理对象生成规则:
    1、mapper接口类路径和mapper.xml中的namespace一致
    2、方法名:和mapper.xml中的statement的id一致
    3、输入参数:和mapper.xml中的parametertype一致
    4、输入结果:和mapper.xml中的resulttype一致
    两个文件:mapper.xml(mapper映射文件)和mapper.java(mapper接口文件)

    2.SqlMapConfig.xml

    配置内容

    SqlMapConfig.xml中配置的内容和顺序如下:

    properties(属性)

    settings(全局配置参数)

    typeAliases(类型别名)

    typeHandlers(类型处理器)

    objectFactory(对象工厂)

    plugins(插件)

    environments(环境集合属性对象)

      environment(环境子属性对象)

        transactionManager(事务管理)

        dataSource(数据源)

    mappers(映射器)

    properties(属性)

    SqlMapConfig.xml可以引用java属性文件中的配置信息如下:

    classpath下定义db.properties文件,

    jdbc.driver=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://localhost:3306/mybatis
    jdbc.username=root
    jdbc.password=mysql

     SqlMapConfig.xml引用如下:

    <properties resource="db.properties" />
        <environments default="development">
            <environment id="development">
                <transactionManager type="JDBC" />
                <dataSource type="POOLED">
                    <property name="driver" value="${jdbc.driver}" />
                    <property name="url" value="${jdbc.url}" />
                    <property name="username" value="${jdbc.username}" />
                    <property name="password" value="${jdbc.password}" />
                </dataSource>
            </environment>
        </environments>

    mappers(映射器)
    Mapper配置的几种方法:
    <mapper resource=" " />
    使用相对于类路径的资源
    如:<mapper resource="sqlmap/user.xml" />

    <mapper url=" " />
    使用完全限定路径
    如:<mapper url="file:///D:workspace_spingmvcmybatis_01configsqlmapuser.xml" />

    <mapper class=" " />
    使用mapper接口类路径
    如:<mapper class="cn.itcast.mybatis.mapper.UserMapper"/>

    注意:此种方法要求mapper接口名称和mapper映射文件名称相同,且放在同一个目录中。

    <package name=""/>
    注册指定包下的所有mapper接口
    如:<package name="cn.itcast.mybatis.mapper"/>
    注意:此种方法要求mapper接口名称和mapper映射文件名称相同,且放在同一个目录中。

     3. Mapper.xml

     Mapper.xml映射文件中定义了操作数据库的sql,每个sql是一个statement,映射文件是mybatis的核心。

     parameterType (输入类型)

    #{}与${}

    #{}实现的是向prepareStatement中的预处理语句中设置参数值,sql语句中#{}表示一个占位符即?

    <!-- 根据id查询用户信息 -->
    
    <select id="selectUserById"  parameterType="int" resultType="user">
    
    select * from user where id = #{id}
    
    </select>

    使用占位符#{}可以有效防止sql注入,在使用时不需要关心参数值的类型,mybatis会根据参数值的类型调用不同的statement设置参数值的方法。可以想象为:如果参数值是一个字符串则自动映射生成的sql中参数值两边自动有单引号,如果参数值是一个数字型则自动映射生成的sql中参数值两边没有单引号。

    注意:当传递单个值时#{}中的参数名称通常和mapper接口的形参名称相同,也可以设置成任意值。

    ${}#{}不同,${}是将参数值不加修饰的拼在sql中,相当中用jdbcstatement拼接sql,使用${}不能防止sql注入,但是有时用${}会非常方便,如下的例子:

    <!-- 根据名称模糊查询用户信息 -->
    
    <select id="selectUserByName" parameterType="string" resultType="user">
    
       select * from user where username like '%${value}%'
    
    </select>

    如果本例子使用#{}则传入的字符串中必须有%号,而%是人为拼接在参数中,显然有点麻烦,如果采用${}sql中拼接为%的方式则在调用mapper接口传递参数就方便很多。

    //如果使用占位符号则必须人为在传参数中加%

    List<User> list = userMapper.selectUserByName("%管理员%");

    //如果使用${}原始符号则不用人为在参数中加%

    List<User> list = userMapper.selectUserByName("管理员");

    再比如order by排序,如果将列名通过参数传入sql,根据传的列名进行排序,应该写为:

    ORDER BY ${columnName} 

    如果使用#{}将无法实现此功能。

    注意:${}不能防止sql注入,对系统安全性有很大的影响,如果使用${}建议传入参数尽量不让用户自动填写,即使要用户填写也要对填写的数据进行校验,保证安全性。

    另外,当传递单个值时${}中填写的参数名称经过测试填写value不错报。

    传递pojo对象

    Mybatis使用ognl表达式解析对象字段的值,如下例子:

    <!—传递pojo对象综合查询用户信息 -->
        <select id="selectUserByUser" parameterType="user" resultType="user">
           select * from user where id=#{id} and username like '%${username}%'
        </select>

    上边大括号标注的是user对象中的字段名称。

    测试:

    public void testselectUserByUser()throws Exception{
            //获取session
            SqlSession session = sqlSessionFactory.openSession();
            //获限mapper接口实例
            UserMapper userMapper = session.getMapper(UserMapper.class);
            //构造查询条件user对象
            User user = new User();
            user.setId(1);
            user.setUsername("管理员");
            //传递user对象查询用户列表
            List<User> list = userMapper.selectUserByUser(user);
            //关闭session
            session.close();
        }

     异常测试:

    Sql中字段名输入错误后测试,username输入dusername测试结果报错:

    org.apache.ibatis.exceptions.PersistenceException: 
    ### Error querying database.  Cause: org.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'dusername' in 'class cn.itcast.mybatis.po.User'
    ### Cause: org.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'dusername' in 'class cn.itcast.mybatis.po.User'

     传递hashmap

    Sql映射文件定义如下:

    <!-- 传递hashmap综合查询用户信息 -->

    <select id="selectUserByHashmap" parameterType="hashmap" resultType="user">

       select * from user where id=#{id} and username like '%${username}%'

    </select>

     上边红色标注的是hashmapkey

     测试:

    public void testselectUserByHashmap()throws Exception{
            //获取session
            SqlSession session = sqlSessionFactory.openSession();
            //获限mapper接口实例
            UserMapper userMapper = session.getMapper(UserMapper.class);
            //构造查询条件Hashmap对象
            HashMap<String, Object> map = new HashMap<String, Object>();
            map.put("id", 1);
            map.put("username", "管理员");
            
            //传递Hashmap对象查询用户列表
            List<User> list = userMapper.selectUserByHashmap(map);
            //关闭session
            session.close();
        }

    异常测试:

    传递的map中的keysql中解析的key不一致。

    测试结果没有报错,只是通过key获取值为空。

    resultType(输出类型)

    输出简单类型

    参考getnow输出日期类型,看下边的例子输出整型:

     Mapper.xml文件

    <!-- 获取用户列表总数 -->
    
    <select id="selectUserCount" parameterType="user" resultType="int">
    
       select count(1) from user
    
    </select>

    Mapper接口

    public int selectUserCount(User user) throws Exception;

    调用:

    public void testselectUserCount() throws Exception{
            //获取session
            SqlSession session = sqlSessionFactory.openSession();
            //获取mapper接口实例
            UserMapper userMapper = session.getMapper(UserMapper.class);
        
            User user = new User();
            user.setUsername("管理员");
    
            //传递Hashmap对象查询用户列表
            int count = userMapper.selectUserCount(user);
            
            //使用session实现
            //int count = session.selectOne("cn.itcast.mybatis.mapper.UserMapper.selectUserCount", user);
            //关闭session
            session.close();
        }

    总结:

    输出简单类型必须查询出来的结果集有一条记录,最终将第一个字段的值转换为输出类型。

    使用sessionselectOne可查询单条记录。

    Sql片段

    Sql中可将重复的sql提取出来,使用时用include引用即可,最终达到sql重用的目的,如下:

    <!-- 传递pojo综合查询用户信息 -->
        <select id="selectUserByUser" parameterType="user" resultType="user">
            select * from user 
            <where> 
            <if test="id!=null and id!=''">
            and id=#{id}
            </if>
            <if test="username!=null and username!=''">
            and username like '%${username}%'
            </if>
            </where>
        </select>

    把where条件抽取出来:

    <sql id="query_user_where">
        <if test="id!=null and id!=''">
            and id=#{id}
        </if>
        <if test="username!=null and username!=''">
            and username like '%${username}%'
        </if>
        </sql>

    使用include引用:

    <select id="selectUserByUser" parameterType="user" resultType="user">
            select * from user 
            <where> 
            <include refid="query_user_where"/>
            </where>
        </select>

    注意:如果引用其它mapper.xmlsql片段,则在引用时需要加上namespace,如下:

    <include refid="namespace.sql片段”/>

    resultMap

    当输出pojo的字段和sql查询出来的字段名称不对应时而还想用这个pojo类作为输出类型这时就需要使用resultMap了。

    另外,resultMap也解决了一对一关联查询、一对多关联查询等常见需求。

    创建Person类:

    public class Person {
        private int id;
        private String name;// 用户姓名,名称和User表的字段名称不一样
        private String sex;// 性别
        private Date birthday;// 出生日期
        private String addr;// 地址,名称和User表的字段名称不一样
        private String detail;// 详细信息
        private Float score;// 成绩
    get/set。。。。

    定义resultMap

    mapper.xml文件中定义resultMap

    <!-- resultMap定义 -->
        <resultMap type="cn.itcast.mybatis.po.Person" id="personmap">
           <id property="id" column="id"/>
           <result property="name" column="username" />
           <result property="addr" column="address" />
        </resultMap>

    <id />:此属性表示查询结果集的唯一标识,非常重要。如果是多个字段为复合唯一约束则定义多个<id />

    Property:表示person类的属性。

    Column:表示sql查询出来的字段名。

    Columnproperty放在一块儿表示将sql查询出来的字段映射到指定的pojo类属性上。

    <result />:普通结果,即pojo的属性。

    这里只将sql查询出来的字段与pojo属性名不一致的进行了定义,通过后边的测试pojo属性名和sql字段相同的自动进行映射。

    Mapper.xml定义

    <!-- 获取用户列表返回resultMap -->
        <select id="selectUserListResultMap" resultMap="personmap">
            select * from user
        </select>

    使用resultMap指定上边定义的personmap

    Mapper接口定义

    public List<Person> selectUserListResultMap() throws Exception;

    实际返回的类型是Person类型。

    测试

    public void testselectUserListResultMap() throws Exception{
            //获取session
            SqlSession session = sqlSessionFactory.openSession();
            //获限mapper接口实例
            UserMapper userMapper = session.getMapper(UserMapper.class);
        
            User user = new User();
            user.setUsername("管理员");
    
            //查询用户列表返回resultMap
            List<Person> list = userMapper.selectUserListResultMap();
            System.out.println(list);
            //关闭session
            session.close();
        }
  • 相关阅读:
    CoreText实现图文混排
    iOS中的数据的存储方式
    获得自定义的所有相簿
    swift基础语法(31- swift可选类型)
    swift内存管理(30- Swift内存管理)
    swift基础语法(29- 析构方法)
    swift基础语法(27- 构造方法,带参数的构造方法,常量存储属性与构造方法,结构体构造方法)
    iOS --旋转动画
    iOS--UILabel上画横线
    iOS--(转)集成银联3.3.0
  • 原文地址:https://www.cnblogs.com/DreamDrive/p/6925629.html
Copyright © 2011-2022 走看看