mybatis使用小贴士
程序员文章站
2024-02-12 23:28:40
...
分享了以下tips: 一、事务管理 二、xml配置sql代码段 三、#和$的区别 四、注意对、做转义 五、依据字符串是否为空,动态组织sql语句 六、使用自定义的类型转换器 七、resultMap的复用 一、事务管理 用户执行一个动作,后台需依次更新多个表,如果其中有一个
分享了以下tips:
一、事务管理
二、xml配置sql代码段
三、#和$的区别
四、注意对做转义
五、依据字符串是否为空,动态组织sql语句
六、使用自定义的类型转换器
七、resultMap的复用
一、事务管理
用户执行一个动作,后台需依次更新多个表,如果其中有一个更新失败,则要回滚之前的更新。这种情况,就是事务回滚。 要支持事务操作,需要:1、确保数据库表的类型是InnoDB,而不是MyISAM
(MyISAM不支持事务,这是一个坑,之前总结过 http://blog.csdn.net/lizeyang/article/details/9280253)2、所有更新操作完成后,再执行commit
例如:SqlSession session = getSqlSession(); int sqlResult = session.insert("insert a table", po); int sqlResult = session.update("update b table", po); //这时,前面的insert和update还没真正执行 session.commit(); //commit后,db才真正更新 session.close();
二、xml配置sql代码段
mybatis需要通过xml配置每次db操作的sql语句。如果多个sql语句中,包含了相同的sql语段,怎么办呢?复制粘贴?这显然不是一个好习惯。建议配置代码段,然后每个sql直接引用这个代码段。 语法:1、在xml中先用这个标签,包装代码段,配置id
from t_comment where refer_type = #{type} and refer_id = #{referId}
2、在需要这个代码段的地方,通过 语句,引入这个sql
例如
三、#和$的区别
当sql中有一些参数,需要调用时动态传入时,就需要在sql中写变量,调用时再传入变量。mybatis中,#和$,都是变量的修饰符。一般推荐用#。 #相当于jdbc中,对prepareStatement做set参数的操作,而$则相当于拼接了sql语句,再执行statament。prepareStatement、statament的区别,具体可以在网上搜下,这里就不赘述了。四、注意对做转义
其实,写xml文件,都得注意这个问题,要将>转义成> 要将来筛选sql结果,更要牢记这个注意项。五、依据字符串是否为空,动态组织sql语句
mabatis xml配置sql,对字符串的处理能力不够强大,对于判空这样常见的需求,还是得通过para!= null and para != ''来完成,没有相关的内建函数。六、使用自定义的类型转换器
假设有一个字段,数据库字段类型,是timestamp,当赋值到javabean时,希望变成对应的long值。怎么做呢?最原生态的一种选择,赋值给bean时,就是timestamp类型,之后再重新遍历数据,改成long值。这种做法,显然很搓,增加了业务逻辑的代码量,不利于转化逻辑复用。 其实,mybatis本身就支持自定义类型转换器,可以很好地支持这种需求。这里主要想告诉你,mybatis是有这种能力的,更详细的内容,可以再看官方教程。这里简单贴下我的示例。1、configuration xml文件配置
…… ……
2、自定义类型转换器
上面的handle为test.TimeTypeHandler,因此工程中,要写对应的test.TimeTypeHandler.java。转换逻辑,就在这个java文件中package test; public class TimeTypeHandler implements TypeHandler{ private final String TIME_TYPE = "yyyy-MM-dd HH:mm:ss"; public long strToLongTime(String dateStr) { if (null == dateStr) return 0L; SimpleDateFormat sdf = new SimpleDateFormat(TIME_TYPE); Date date = new Date(); try { date = sdf.parse(dateStr); } catch (ParseException e) { e.printStackTrace(); } if (date == null) return 0L; System.out.println(date); return date.getTime(); } public Long getResult(ResultSet arg0, String arg1) throws SQLException { String datestr = arg0.getString(arg1); return strToLongTime(datestr); } public Long getResult(ResultSet arg0, int arg1) throws SQLException { String datestr = arg0.getString(arg1); return strToLongTime(datestr); } public Long getResult(CallableStatement arg0, int arg1) throws SQLException { String datestr = arg0.getString(arg1); return strToLongTime(datestr); } public void setParameter(PreparedStatement arg0, int arg1, Long arg2, JdbcType arg3) throws SQLException { if (arg2 == null) { arg2 = 0L; } Date date = new Date(arg2); SimpleDateFormat sdf = new SimpleDateFormat(TIME_TYPE); String datetime = sdf.format(date); arg0.setString(arg1, datetime); } }
3、mapper xml配置
…… //这里定义了ExpDownloadPo这个sql查询结果,要映射到java bean :test.ExpDownloadPo …… //first_download_time这个数据,在赋值到firstDownloadTime时,要转转换,从jdbc的time型,转换成javaType的long型,具体怎么转换,因为前面第一点,定义了 ,因此会自动使用test.TimeTypeHandler这个转换器
七、resultMap的复用
我们一般会定义多个mapper.xml文件,将不同的功能,放到不同的xml中。如果mapper文件A需要使用mapper文件B中所定义的resultmap时,是可以直接使用的,请勿在文件B中重复定义resultmap。 具体使用方法是: 例如mapper A的namespace是com.blog.test,其中定义的resultmapid是testbean,那么,在文件B中,如果要用到这个resultmap,则用它的全名即可,也就是com.blog.test.testbean(namespace+resultmapid)mapper文件A
……