5

我一直在尝试TypeHandler在 MyBatis 中进行自定义,以便对于null数据库中的列,MyBatis 返回Null Object 模式的实现,而不是null在域类中。

谷歌上寻求帮助后,我找到了优秀的项目mybatis-koans,即koan 19,它用我使用的相同方法解决了这个问题,即扩展BaseTypeHandler<T>(是抽象的)。在这一点上,我有一个TypeHandler类似于EmailTypeHandlerkoan的具体内容:

/**
 * Acts as a factory method to return the appropriate implementation of an Email.
 * Returns a Null object if the email value in the database was null/empty
 */
public class EmailTypeHandler extends BaseTypeHandler<Email> {

  @Override
  public Email getNullableResult(ResultSet rs, String colName) throws SQLException {
    return createEmail(rs.getString(colName));
  }

  @Override
  public Email getNullableResult(ResultSet rs, int colNum) throws SQLException {
    return createEmail(rs.getString(colNum));
  }


  private Email createEmail(String s) {
    System.out.println(s);
    if (s == null || s.equals("")) {
      return new NullEmail();
    } else {
      return new EmailImpl(s);
    }
  }

  @Override
  public Email getNullableResult(CallableStatement arg0, int arg1) throws SQLException {
    return null;
  }

  @Override
  public void setNonNullParameter(PreparedStatement ps, int colNum,
                                  Email e, JdbcType t) throws SQLException {
  }
}

不幸的是,作者( midpeter444 )似乎面临同样的问题:当数据库中的值为 时null,仍然返回 null 而不是具体制作的对象TypeHandler

4

2 回答 2

3

我在发布问题后立即看到了解决方案。在BaseTypeHandler代码中:

[...]
  public T getResult(ResultSet rs, String columnName) throws SQLException {
    T result = getNullableResult(rs, columnName);
    if (rs.wasNull()) {
      return null;
    } else {
      return result;
    }
  }

  public T getResult(ResultSet rs, int columnIndex) throws SQLException {
    T result = getNullableResult(rs, columnIndex);
    if (rs.wasNull()) {
      return null;
    } else {
      return result;
    }
  }

  public T getResult(CallableStatement cs, int columnIndex) throws SQLException {
    T result = getNullableResult(cs, columnIndex);
    if (cs.wasNull()) {
      return null;
    } else {
      return result;
    }
  }

  public abstract void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;

  public abstract T getNullableResult(ResultSet rs, String columnName) throws SQLException;

  public abstract T getNullableResult(ResultSet rs, int columnIndex) throws SQLException;

  public abstract T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException;
[...]

显然,将在数据库中的值为 时BaseTypeHandler返回,就像在这种情况下一样。因此,解决方案是创建一个具体的返回适当的实现(在这种情况下,当数据库中的值为 null 时为 NullObject 实现)而不进行子类化。nullnullcs.WasNull()trueTypeHandler<T>BaseTypeHandler

已编辑(根据 quux00评论):当然,我们可以重写getResult的方法BaseTypeHandler,因为它的方法提供的功能setParameters可能很有用。

于 2013-02-14T12:13:59.007 回答
2

感谢您的回答!它帮助我了解了我的问题以及如何克服它。我希望结果永远不会为空。

通过发布 的代码BaseTypeHandler,它确认返回 null 取决于ResultSet.wasNull而不是返回的结果本身getNullableResult。所以我覆盖getResultBaseTypeHandler. 在自定义实现中getResultsuper.getResult调用 , 然后检查 null 。下面是一个例子:

public Money getResult ( ResultSet rs, String columnName ) throws SQLException
{
    Money result = super.getResult ( rs, columnName );
    return result == null ? Money.ZERO : result;
}

public Money getResult ( ResultSet rs, int columnIndex ) throws SQLException
{
    Money result = super.getResult ( rs, columnIndex );
    return result == null ? Money.ZERO : result;
}

public Money getResult ( CallableStatement cs, int columnIndex ) throws SQLException
{
    Money result = super.getResult ( cs, columnIndex );
    return result == null ? Money.ZERO : result;
}
于 2016-02-25T16:08:11.733 回答