前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >从org.springframework.dao.DuplicateKeyException说起

从org.springframework.dao.DuplicateKeyException说起

作者头像
春哥大魔王
发布2018-07-23 11:33:01
2K0
发布2018-07-23 11:33:01
举报

Spring MyBatis的异常处理

通常在dao层将所有异常转嫁到Spring的RuntimeException体系(org.springframework.dao.DataAccessException)中来。 如:

代码语言:javascript
复制
try {  
    vehicleDAO.insert(vehicle);  
} catch (DataAccessException e) {  
    SQLException sqle = (SQLException) e.getCause();  
    System.out.println("Error code: " + sqle.getErrorCode());  
    System.out.println("SQL state: " + sqle.getSQLState());  
} 

我们可以获取状态码和SQL状态。 这样就知道了这个错误的具体含义,比如104:唯一约束验证失败。这就是我们故意设置的重复主键问题。

Spring的JDBC模块为我们预定义了一些错误代码,它存储在org.springframework.jdbc.support包下的sql-error-codes.xml文件中,其中描述HSQL的内容为:

代码语言:javascript
复制
<bean id="HSQL" class="org.springframework.jdbc.support.SQLErrorCodes">  
    <property name="databaseProductName">  
        <value>HSQL Database Engine</value>  
    </property>  
    <property name="badSqlGrammarCodes">  
        <value>-22,-28</value>  
    </property>  
    <property name="duplicateKeyCodes">  
        <value>-104</value>  
    </property>  
    <property name="dataIntegrityViolationCodes">  
        <value>-9</value>  
    </property>  
    <property name="dataAccessResourceFailureCodes">  
        <value>-80</value>  
    </property>  </bean>  

上面我们已经知道在org.springframework.jdbc.support包下有sql-error-codes.xml文件,在Spring启动时会自动读取这个文件中的错误码,它为我们预分类了一些错误码,而我们可以加强它,来使用我们自定义的异常。

首先,定义一个异常类,我们就来自定义一下前面的-104错误,就是HSQL的重复键的问题:

代码语言:javascript
复制
package org.ourpioneer.vehicle.exception;  import org.springframework.dao.DataIntegrityViolationException;  public class VehicleDuplicateKeyException extends  
        DataIntegrityViolationException {  
    public VehicleDuplicateKeyException(String msg) {  
        super(msg);  
    }  
    public VehicleDuplicateKeyException(String msg, Throwable cause) {  
        super(msg, cause);  
    }  
} 

之后我们重新新建一个sql-error-codes.xml代码,并将它放到类路径的根目录下,这样Spring会发现它并使用我们自定义的文件,在配置中定义如下:

代码语言:javascript
复制
<bean id="HSQL" class="org.springframework.jdbc.support.SQLErrorCodes">  
        <property name="databaseProductName" value="HSQL Database Engine" />  
        <property name="useSqlStateForTranslation" value="false" />  
        <property name="customTranslations">  
            <list>  
                <ref local="vehicleDuplicateKeyTranslation" />  
            </list>  
        </property>  
    </bean>  
    <bean id="vehicleDuplicateKeyTranslation"  
    class="org.springframework.jdbc.support.CustomSQLErrorCodesTranslation">  
        <property name="errorCodes" value="-104" />  
        <property name="exceptionClass"  
    value="org.ourpioneer.vehicle.exception.VehicleDuplicateKeyException" />  </bean>  

之后就可以使用我们自己定义的异常类了。出现异常后,VehicleDuplicateKeyException就抛出了。 DataAccessException是RuntimeException,是无需检查的异常,不要求进行代码处理。

Spring的Dao抛上来的异常通常有:

  • CleanupFailureDataAccessException 一项操作成功地执行,但在释放数据库资源时发生异常(例如,关闭一个Connection)
  • DataAccessResourceFailureException 数据访问资源彻底失败,例如不能连接数据库
  • DataIntegrityViolationException Insert或Update数据时违反了完整性,例如违反了惟一性限制
  • DataRetrievalFailureException 某些数据不能被检测到,例如不能通过关键字找到一条记录
  • DeadlockLoserDataAccessException 当前的操作因为死锁而失败
  • IncorrectUpdateSemanticsDataAccessException Update时发生某些没有预料到的情况,例如更改超过预期的记录数。当这个异常被抛出时,执行着的事务不会被回滚
  • InvalidDataAccessApiusageException 一个数据访问的JAVA API没有正确使用,例如必须在执行前编译好的查询编译失败了
  • InvalidDataAccessResourceUsageException 错误使用数据访问资源,例如用错误的SQL语法访问关系型数据库
  • OptimisticLockingFailureException 乐观锁的失败。这将由ORM工具或用户的DAO实现抛出
  • TypemismatchDataAccessException Java类型和数据类型不匹配,例如试图把String类型插入到数据库的数值型字段中
  • UncategorizedDataAccessException 有错误发生,但无法归类到某一更为具体的异常中

这样服务层可以精确的捕获异常,或者向上继续抛出异常。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-06-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 春哥talk 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Spring MyBatis的异常处理
  • Spring的Dao抛上来的异常通常有:
相关产品与服务
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档