親子関係に ElwInfo と ElwUnderlyingAsset の 2 つのクラスがあります。テーブル名は ELW_INFO、ELW_UNDERLYING_ASSET であり、ELW_UNDERLYING_ASSET は列 INSTRUMENT_CD によって ELW_INFO を参照します。
次のように、ファイル「Elw.hbm.xml」に両方のクラスをマップしました
<class name="ElwInfo" table="ELW_INFO" >
    <id name="instrumentCode" column="INSTRUMENT_CD" ><generator class="assigned"/></id>
    <property name="instrumentName" column="INSTRUMENT_NM"/>
    <property name="ccyCode"        column="CCY_CD"/>
    <one-to-one name="instrumentCommon" 
                class="InstrumentCommon"
                cascade="save-update" lazy="false" />
    <one-to-one name="instrumentClass" 
                class="InstrumentClass"
                cascade="none" lazy="false" />
    <property name="issueDate"      column="ISSUE_DT"/>
    <property name="maturityDate"   column="MRTY_DT"/>
    <property name="convertRatio"   column="CONVERT_RATIO"/>
    <property name="strike"         column="STRIKE"/>
    <property name="koBarrier"      column="KO_BARRIER"/>
    <property name="minRebateRate"  column="MIN_REBATE_RT"/>
    <property name="addRebate"      column="ADD_REBATE"/>
    <many-to-one class="com.fistglobal.riskcraft.market.web.domain.instrument2.common.ExchangeInfo" 
                 name="exchangeInfo" column="EXCH_CD" lazy="false" cascade="none" 
                 unique="true" />
    <many-to-one class="com.fistglobal.riskcraft.market.web.domain.instrument2.common.ElwTypeInfo" 
                 name="elwType" column="ELW_TYPE_CD" lazy="false" cascade="none" 
                 unique="true" />
    <many-to-one class="com.fistglobal.riskcraft.market.web.domain.instrument2.common.OptionTypeInfo" 
                 name="optionType" column="OPTION_TYPE_CD" lazy="false" cascade="none" 
                 unique="true" />
    <many-to-one class="com.fistglobal.riskcraft.market.web.domain.instrument2.common.ExerciseTypeInfo" 
                 name="exerciseType" column="EXERCISE_TYPE_CD" lazy="false" cascade="none" 
                 unique="true" />
    **<bag name="underlyingAssetList" cascade="save-update" table="ELW_UNDERLYING_ASSET" lazy="false" >
        <key column="INSTRUMENT_CD"/>
        <one-to-many class="ElwUnderlyingAsset" not-found="ignore"/>
    </bag>**
</class>
<class name="ElwUnderlyingAsset" table="ELW_UNDERLYING_ASSET" >
    <composite-id name="id" class="ElwUnderlyingAsset$Id">
        <key-property name="instrumentCode" column="INSTRUMENT_CD"/>
        <key-property name="underlyingAssetCode" column="UNDERLYING_ASSET_CD"/>
        <generator class="assigned"/>
    </composite-id>
    <property name="qty"   column="UNDERLYING_ASSET_QTY"/>
</class>
<class name="ElwData" table="ELW_DATA">
    <composite-id name="key" class="InstrumentDataKey">
        <key-property name="dt"             column="DT"/>
        <key-property name="instrumentCode" column="INSTRUMENT_CD"/>
    </composite-id>
    <property name="barrierHitYn"   column="BARRIER_HIT_YN"/>
    <property name="barrierHitDate" column="BARRIER_HIT_DT"/>
    <property name="amOrPm"         column="AM_OR_PM"/>
    <property name="maxPrice"       column="MAX_PRICE"/>
    <property name="minPrice"       column="MIN_PRICE"/>
</class>
2つのクラスは次のとおりです
public class ElwInfo extends InstrumentInfo {
private ExchangeInfo exchangeInfo;
private List<ElwUnderlyingAsset> underlyingAssetList;
private ElwTypeInfo elwType;
private OptionTypeInfo optionType;
private ExerciseTypeInfo exerciseType;
private String issueDate;
private String maturityDate;
private Double convertRatio;
private Double strike;
private Double koBarrier;
private Double minRebateRate;
private Double addRebate;
public ElwInfo() {
    exchangeInfo = new ExchangeInfo();
    underlyingAssetList = new ArrayList<ElwUnderlyingAsset>();
    elwType = new ElwTypeInfo();
    optionType = new OptionTypeInfo();
    exerciseType = new ExerciseTypeInfo();
}
public ExchangeInfo getExchangeInfo() {
    return exchangeInfo;
}
public void setExchangeInfo(ExchangeInfo exchangeInfo) {
    this.exchangeInfo = exchangeInfo;
}
public List<ElwUnderlyingAsset> getUnderlyingAssetList() {
    return underlyingAssetList;
}
public void setUnderlyingAssetList(List<ElwUnderlyingAsset> underlyingAssetList) {
    this.underlyingAssetList = underlyingAssetList;
}
public ElwTypeInfo getElwType() {
    return elwType;
}
public void setElwType(ElwTypeInfo elwType) {
    this.elwType = elwType;
}
public OptionTypeInfo getOptionType() {
    return optionType;
}
public void setOptionType(OptionTypeInfo optionType) {
    this.optionType = optionType;
}
public ExerciseTypeInfo getExerciseType() {
    return exerciseType;
}
public void setExerciseType(ExerciseTypeInfo execType) {
    this.exerciseType = execType;
}
public String getIssueDate() {
    return issueDate;
}
public void setIssueDate(String issueDate) {
    this.issueDate = issueDate;
}
public String getMaturityDate() {
    return maturityDate;
}
public void setMaturityDate(String maturityDate) {
    this.maturityDate = maturityDate;
}
public Double getConvertRatio() {
    return convertRatio;
}
public void setConvertRatio(Double convertRatio) {
    this.convertRatio = convertRatio;
}
public Double getStrike() {
    return strike;
}
public void setStrike(Double strike) {
    this.strike = strike;
}
public Double getKoBarrier() {
    return koBarrier;
}
public void setKoBarrier(Double koBarrier) {
    this.koBarrier = koBarrier;
}
public Double getMinRebateRate() {
    return minRebateRate;
}
public void setMinRebateRate(Double minRebateRate) {
    this.minRebateRate = minRebateRate;
}
public Double getAddRebate() {
    return addRebate;
}
public void setAddRebate(Double addRebate) {
    this.addRebate = addRebate;
}
}
public class ElwUnderlyingAsset {
public static class Id implements Serializable {
    private static final long serialVersionUID = -8296040137055492597L;
    private String instrumentCode;
    private String underlyingAssetCode;
    public String getInstrumentCode() {
        return instrumentCode;
    }
    public void setInstrumentCode(String instrumentCode) {
        this.instrumentCode = instrumentCode;
    }
    public String getUnderlyingAssetCode() {
        return underlyingAssetCode;
    }
    public void setUnderlyingAssetCode(String underlyingAssetCode) {
        this.underlyingAssetCode = underlyingAssetCode;
    }
    public boolean equals(Object o) {
        if (o != null && o instanceof Id) {
            Id that = (Id)o;
            return this.instrumentCode.equals(that.instrumentCode) && this.underlyingAssetCode.equals(that.underlyingAssetCode);
        } else {
            return false;
        }
    }
    public int hashCode() {
        if (instrumentCode!= null && underlyingAssetCode != null)
            return instrumentCode.hashCode() + underlyingAssetCode.hashCode();
        else
            return -1;
    }
    public String toString() {
        return ToStringBuilder.reflectionToString(this, ToStringStyle.MULTI_LINE_STYLE);
    }
}
public ElwUnderlyingAsset() {
    this.id = new Id();
}
private Id id;
private Double qty;
public Id getId() {
    return id;
}
public void setId(Id id) {
    this.id = id;
}
public Double getQty() {
    return qty;
}
public void setQty(Double qty) {
    this.qty = qty;
}
public String toString() {
    return ToStringBuilder.reflectionToString(this, ToStringStyle.MULTI_LINE_STYLE);
}
}
私が電話するとき
ElwInfo newInfo = .....// info instance is constructed purely from http request parameters .
getHibernateTemplate().saveOrUpdate(newInfo);
次の生成された SQL と Oracle Native Exception が発生します。生成された SQL は非常にばかげており、何が間違っていたのかわかりません。
Hibernate: 
update
    ELW_UNDERLYING_ASSET 
set
    INSTRUMENT_CD=null 
where
    INSTRUMENT_CD=?
私が期待しているのは
Hibernate: 
update
    ELW_UNDERLYING_ASSET 
set
    QTY = ?
where
    INSTRUMENT_CD=?
    and UNDERLYING_ASSET_CD=?
テーブル ELW_UNDERLYING_ASSET には複合キー (INSTRUMENT_CD、UNDERLYING_ASSET_CD) があるためです。
クライアント コードを次のように変更すると、
ElwInfo newInfo = .....// info instance is constructed purely from http request 
ElwInfo info = (ElwInfo)getHibernateTemplate().get(entityInfo.getClass(),entityInfo.getInstrumentCode());
info.setAddRebate(0.0);
info.getUnderlyingAssetList().clear();
info.getUnderlyingAssetList().addAll(newInfo.getUnderlyingAssetList());
getHibernateTemplate().saveOrUpdate(info);
次の生成された SQL を取得します。
Hibernate: 
update
    ELW_INFO 
set
    INSTRUMENT_NM=?,
    CCY_CD=?,
    ISSUE_DT=?,
    MRTY_DT=?,
    CONVERT_RATIO=?,
    STRIKE=?,
    KO_BARRIER=?,
    MIN_REBATE_RT=?,
    ADD_REBATE=?,
    EXCH_CD=?,
    ELW_TYPE_CD=?,
    OPTION_TYPE_CD=?,
    EXERCISE_TYPE_CD=? 
where
    INSTRUMENT_CD=?
Hibernate: 
update
    ELW_UNDERLYING_ASSET 
set
    INSTRUMENT_CD=null 
where
    INSTRUMENT_CD=? 
    and UNDERLYING_ASSET_CD=?
最初に生成された sql は問題ありませんが、前のものとは少し異なりますが、2 番目のものはまだ間違っています。
あらかじめご了承ください。