mybatis百科-列對映類ResultMapping
阿新 • • 發佈:2018-12-16
ResultMapping
物件是 mybatis 的 <resultMap> 節點在 ResultMap
物件中基礎組成部分.
ResultMapping
物件記錄了結果集中一列與隊友JavaBean中一個屬性的對應關係。
1 成員變數
// Configuration 物件, 看過前面原始碼的應該知道這個物件的含義 private Configuration configuration; // 對應相應 JavaBean 中的成員變數 private String property; // 對應節點的 column 屬性, 對應檢索出來的列名(別名) private String column; // 對應節點的 javaType 屬性 private Class<?> javaType; // 對應節點的 jdbcType 屬性, 表示對映列的JDBC屬性 private JdbcType jdbcType; // 型別處理器 private TypeHandler<?> typeHandler; // 對應另一個 resultMap 的 id, 負責將結果集中的一部分對映成其他物件。 private String nestedResultMapId; // private String nestedQueryId; // 對應節點的 notNullColumns 屬性拆分後的結果 private Set<String> notNullColumns; // 對應節點的 columnPrefix 屬性 private String columnPrefix; // 處理後的標記, 有兩種:id和constructor private List<ResultFlag> flags; // private List<ResultMapping> composites; // 對應節點的 resultSet 屬性 private String resultSet; // 對應節點的 foreignColumn 屬性 private String foreignColumn; // 是否延遲載入, 對應節點的 fetchType 屬性 private boolean lazy;
2 建構函式
ResultMapping() {
}
就是一個空的建構函式
3 其他函式
3.1 setter 和 getter 函式
ResultMapping
物件建立使用的是建造者模式, 因此,只有部分成員變數含有 setter 函式。而除了 Configuration 物件, 其他都含有 getter 函式。
3.2 equals 和 hashCode 函式
該函式重寫了 equals 方法。
@Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } ResultMapping that = (ResultMapping) o; if (property == null || !property.equals(that.property)) { return false; } return true; }
而 equals 重寫, 則基本上 hashCode 也要重寫。 以 property 或 column 的 hashCode 作為其 hashCode
@Override
public int hashCode() {
if (property != null) {
return property.hashCode();
} else if (column != null) {
return column.hashCode();
} else {
return 0;
}
}
3.3 toString 函式
@Override public String toString() { final StringBuilder sb = new StringBuilder("ResultMapping{"); //sb.append("configuration=").append(configuration); // configuration doesn't have a useful .toString() sb.append("property='").append(property).append('\''); sb.append(", column='").append(column).append('\''); sb.append(", javaType=").append(javaType); sb.append(", jdbcType=").append(jdbcType); //sb.append(", typeHandler=").append(typeHandler); // typeHandler also doesn't have a useful .toString() sb.append(", nestedResultMapId='").append(nestedResultMapId).append('\''); sb.append(", nestedQueryId='").append(nestedQueryId).append('\''); sb.append(", notNullColumns=").append(notNullColumns); sb.append(", columnPrefix='").append(columnPrefix).append('\''); sb.append(", flags=").append(flags); sb.append(", composites=").append(composites); sb.append(", resultSet='").append(resultSet).append('\''); sb.append(", foreignColumn='").append(foreignColumn).append('\''); sb.append(", lazy=").append(lazy); sb.append('}'); return sb.toString(); }
恨我們平常寫的基本差不多。
4 內部類 Builder
ResultMapping 是使用建造者模式來進行建立的。
4.1 成員變數
private ResultMapping resultMapping = new ResultMapping();
4.2 建構函式
就是給部分屬性賦值。
public Builder(Configuration configuration, String property, String column, TypeHandler<?> typeHandler) {
this(configuration, property);
resultMapping.column = column;
resultMapping.typeHandler = typeHandler;
}
public Builder(Configuration configuration, String property, String column, Class<?> javaType) {
this(configuration, property);
resultMapping.column = column;
resultMapping.javaType = javaType;
}
public Builder(Configuration configuration, String property) {
resultMapping.configuration = configuration;
resultMapping.property = property;
resultMapping.flags = new ArrayList<ResultFlag>();
resultMapping.composites = new ArrayList<ResultMapping>();
resultMapping.lazy = configuration.isLazyLoadingEnabled();
}
4.3 建造者模式相關函式
除了 configuration, property, column, 其他成員變數都有類似如下的函式: 賦值後返回 Builder 物件本身。
public Builder javaType(Class<?> javaType) {
resultMapping.javaType = javaType;
return this;
}
建造物件的函式
public ResultMapping build() {
// 返回不可更改的 List
resultMapping.flags = Collections.unmodifiableList(resultMapping.flags);
// 返回不可更改的 List
resultMapping.composites = Collections.unmodifiableList(resultMapping.composites);
resolveTypeHandler();
// 校驗
validate();
return resultMapping;
}
校驗
// 對我們寫的配置進行校驗
private void validate() {
// 不可同時定義 nestedQueryId 和 nestedResultMapId
if (resultMapping.nestedQueryId != null && resultMapping.nestedResultMapId != null) {
throw new IllegalStateException("Cannot define both nestedQueryId and nestedResultMapId in property " + resultMapping.property);
}
// nestedQueryId 、 nestedResultMapId 和 typeHandler 不能同時為 null
if (resultMapping.nestedQueryId == null && resultMapping.nestedResultMapId == null && resultMapping.typeHandler == null) {
throw new IllegalStateException("No typehandler found for property " + resultMapping.property);
}
// Issue #4 and GH #39: column is optional only in nested resultmaps but not in the rest
if (resultMapping.nestedResultMapId == null && resultMapping.column == null && resultMapping.composites.isEmpty()) {
throw new IllegalStateException("Mapping is missing column attribute for property " + resultMapping.property);
}
if (resultMapping.getResultSet() != null) {
int numColumns = 0;
if (resultMapping.column != null) {
numColumns = resultMapping.column.split(",").length;
}
int numForeignColumns = 0;
if (resultMapping.foreignColumn != null) {
numForeignColumns = resultMapping.foreignColumn.split(",").length;
}
if (numColumns != numForeignColumns) {
throw new IllegalStateException("There should be the same number of columns and foreignColumns in property " + resultMapping.property);
}
}
}