当前位置:Gxlcms > mysql > mybatis使用小贴士

mybatis使用小贴士

时间:2021-07-01 10:21:17 帮助过:36人阅读

分享了以下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结果,更要牢记这个注意项。

五、依据字符串是否为空,动态组织sql语句

mabatis xml配置sql,对字符串的处理能力不够强大,对于判空这样常见的需求,还是得通过para!= null and para != ''来完成,没有相关的内建函数。 sql语句

六、使用自定义的类型转换器

假设有一个字段,数据库字段类型,是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


……

mapper文件B


忘记密码?

登录