0

hibernateとpostgresで最初のステップを踏み、エンティティクラスの古いスタイルのxml構成を作成し、(Manning Java Persistence with Hib)の本で説明されているように、いくつかのdao実装を作成し、HibernateUtilクラスを作成してSessionFactoryを作成しました。いくつかのエンティティとdaoを介して保存しようとしました。

このコードでは、Customerクラスに一連の'Order'があります。これを構成するために、以下に示すようにhbm.xmlファイルを作成しました。

demo()メソッドで、エンティティインスタンスを作成し、daoを介してsaveOrUpdate()を呼び出しました。正常に実行され、dbに2つの顧客と注文が作成されます。ただし、注文を顧客に追加しようとすると、RuntimeExceptionが発生し、トランザクションがロールバックされます。アプリをデバッグしようとすると、

org.hibernate.context.ThreadLocalSessionContext.TransactionProtectionWrapper.invoke()---> lang.reflect.Method.invoke()---> java.lang.reflect.InvocationTargetException

なぜこれが発生するのか混乱しています。addOrdersToCustomers()を呼び出さない場合、このエラーは発生しません。そのメソッドを呼び出さない場合、Customerオブジェクトに注文のセットがないことを意味しませんか?データベースでは、createOrders()メソッドがCustomerを設定するため、OrderテーブルにはCustomerのIDに正常に設定されたFKcustomer_idがありますOrderのフィールドをcustomerインスタンスに追加しますが、CustomerのSetordersフィールドは更新されません。

これを修正する方法はありますか?ありがとう、

ジム

public class Main {
    CustomerDao custdao;
    OrderDao orderdao;  
    Customer mark,jon;
    Order order1,order2,order3; 
    SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd");

        public void demo()throws ParseException{
             custdao = Daofactory.getCustomerDao();
         orderdao = Daofactory.getOrderDao();
             createCustomers();
             createOrders();
             //addOrdersToCustomers();//uncommenting this causes RuntimeException
             updateEntities();
        }
        private  void updateEntities() {
            Transaction tx = null;
            Session session = HibernateUtil.getCurrentSession();
            logger.info("got session:"+session);
            try{
                tx = session.beginTransaction();
                logger.info("got transaxion:"+tx);

                custdao.saveOrUpdateCustomer(mark);         
                custdao.saveOrUpdateCustomer(jon);          
                orderdao.saveOrUpdateOrder(order1);     
                        tx.commit();
                 }catch(Exception e){
                    tx.rollback();
                 }
        }

       private void addOrdersToCustomers() {
        mark.getOrders().add(order1);
       }
       private void createCustomers() {
        mark = new Customer();
        mark.setName("mark");
        mark.setEmailAddress("mark@home");
        mark.setAddress("121,3rd avenue");
        mark.setCity("San Diego");
        mark.setState("CA");
        mark.setCountry("U.S.A");

        jon = new Customer();
        jon.setName("jon");
        jon.setEmailAddress("jon@home");
        jon.setAddress("1121 vpura");
        jon.setCity("bangalore");
        jon.setState("karnataka");
        jon.setCountry("india");

    }
    private void createOrders() throws ParseException {
        order1 = new Order();
        order1.setCustomer(mark);
        order1.setOrderNumber(Long.parseLong("111111111"));
        order1.setOrderDate(sdf.parse("2001/01/02"));
                ...  
    }
       ...
       public static void main(String[] args) throws ParseException {
        new Main().demo();

        }
    }

マッピングは次のとおりです。

public class Customer {
    private Long customer_id;
    ...        
    private Set<Order> orders;
    public Customer() {
        super();
        orders = new HashSet<Order>();
    }
    ...
}

<hibernate-mapping package="org.me.hibernatestore">
 <class name="Customer" table="CUSTOMER">
  <id column="CUSTOMER_ID" name="customer_id"  type="java.lang.Long">
  <generator class="native"/>
  </id>  
...      
    <set name="orders" table="ORDERS" lazy="false" cascade="delete-orphan">
        <key column="CUSTOMER_ID"/>
        <one-to-many class="Order" />
    </set>    
 </class>
</hibernate-mapping>

Order.java

public class Order {
    private Long order_id;
    ...
    private Customer customer;
    ...
}

Order.hbm.xml

...
<class name="Order" table="ORDERS">
    <id name="order_id" column="ORDER_ID" type="long">
        <generator class="native"/>
    </id>
    ...
    <many-to-one name="customer" class="Customer" column="CUSTOMER_ID" lazy="false" />
...

daoの実装には基本クラスがあります

public class BaseDaoImpl<T, ID extends Serializable> implements BaseDao<T,ID>{
    private Class<T> persistentClass;
    private Session session;    
    public BaseDaoImpl() {
        this.persistentClass = (Class<T>)(((ParameterizedType)this.getClass().getGenericSuperclass()).getActualTypeArguments()[0]);
    }   
    public Session getSession() {
        if (session == null){           
            return HibernateUtil.getCurrentSession();
        }       
        return session;
    }
       ...
}

HibernateUtil.java

public class HibernateUtil {    
    private static SessionFactory sessionFactory;
    static {
        try {
            sessionFactory = new Configuration().configure().buildSessionFactory();         
        }catch (Throwable ex) {
            throw new ExceptionInInitializerError(ex);
        }
    }
    public static SessionFactory getSessionFactory() {
        return sessionFactory;
    }
    public static Session getCurrentSession(){
        Session session = sessionFactory.getCurrentSession();       l
        return session;
    }
4

1 に答える 1

1

スタックトレースは理由を明確に示しています。一時的なオブジェクトへの参照を使用してオブジェクトを保存しようとしています。save()追加したOrderオブジェクトを呼び出す前に呼び出すか、クラスのセットにCustomer(少なくとも)永続カスケードを追加する必要があります。あなたのxmlの場合、これは修正されますordersCustomer

<set name="orders" table="ORDERS" lazy="false" cascade="delete-orphan">

<set name="orders" table="ORDERS" lazy="false" cascade="delete-orphan,persist">
于 2011-07-27T14:23:41.907 に答える