5

TypeHandlerMyBatis でカスタムを作成して、データベースの列に対して、MyBatis がドメイン クラスに含まれる代わりにNull オブジェクト パターンnullの実装を返すようにしようとしています。null

グーグルで助けを求めた後、私は優れたプロジェクトmybatis-koans、つまり、私が使用しているのと同じアプローチ、つまり拡張(抽象的)でこの問題に正確に対処するkoan 19に到達しました。BaseTypeHandler<T>この時点で、その koan にTypeHandler似た具象があります。EmailTypeHandler

/**
 * 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 ) も同じ問題に直面しているようです。DB の値が の場合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

EDITED ( quux00のコメントに続く): もちろん、そのメソッドによって提供される機能が役立つ場合は、 のgetResultメソッドをオーバーライドすることができます。BaseTypeHandlersetParameters

于 2013-02-14T12:13:59.007 に答える
2

ご回答有難うございます!自分の問題とそれを克服する方法を理解するのに役立ちました。結果がnullにならないようにしたかったのです。

のコードを投稿することで、から返された結果自体ではなく、BaseTypeHandlerに応じて null が返されていることが確認されました。そのため、 の具象サブクラスをオーバーライドしました。のカスタム実装では、が呼び出され、null がチェックされます。以下に例を示します。ResultSet.wasNullgetNullableResultgetResultBaseTypeHandlergetResultsuper.getResult

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 に答える