Hibernate 级联属性
Hibernate 级联属性
萧小蚁 发表于2年前
Hibernate 级联属性
  • 发表于 2年前
  • 阅读 17
  • 收藏 0
  • 点赞 0
  • 评论 0

腾讯云 学生专属云服务套餐 10元起购>>>   

摘要: day03 完

级联状态

cascade(级联)

用来指定两个对象之间的操作联动关系。当主控方执行操作时,关联对象(被动方)是否同步执行同一操作。

常用可选值

none:所有情况下均不进行级联操作。这是默认值。

save-update: 在执行保存和更新时进行级联操作。

delete:在执行删除时进行级联操作。

all : 所有情况下均进行级联操作。

级联只对增、删、改起作用。

inverse="true"和cascade="all"同时使用:

级联delete操作可以正常进行, insert和update会级联操作,但不维护关联关系。

注意:如果inverse="true"和cascade="all"同时使用,为了在做insert和update操作时同样能够

进行级联操作,需要在被控端同时维护一对多和多对一两个关系。

package model;

import java.util.Set;

/**
 * @author sally
 * 双向的一对多或多对一 自身关联
 */
public class Category {

	private int id;
	private String name;
	//如果把Category看成是多的一端
	private Category parent;
	//如果把Category看成是少的一端,则需要对多的一端进行对象集合的引用
	private Set<Category> clist;
	
	public Category(){
		
	}

	public int getId() {
		return id;
	}

	public void setId(int id) {
		this.id = id;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public Category getParent() {
		return parent;
	}

	public void setParent(Category parent) {
		this.parent = parent;
	}

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC 
	"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
	"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
	<!-- name属性指定类名(全限定名) table指明表名,不指明table数据默认的表名和实体名一致 -->
    <class name="model.Category" table="cate_tab">
    	<!-- type指明当前字段的类型    name对应实体中的属性名 -->
        <id type="integer" name="id">
        	<!-- 提供ID自增的策略  native会根据数据库自行判断 -->
            <generator class="native"/>
        </id>
        <property name="name" type="string"></property>
        <set name="clist" inverse="false" cascade="all">
        	<key column="parent_id"></key>
        	<!-- 配置一对多的关联映射 -->
        	<one-to-many class="model.Category"/>
        </set>
        <many-to-one name="parent" column="parent_id"></many-to-one>
        
    </class>	
</hibernate-mapping>

<!DOCTYPE hibernate-configuration PUBLIC
	"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
	"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">

<hibernate-configuration>
	<session-factory>
		<!-- 配置连接数据库的参数 -->
		<!-- 配置数据库的方言 -->
		<property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
		<!-- 数据库驱动 -->
		<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
		<property name="hibernate.connection.url">jdbc:mysql:///2o2mzishen</property>
		<property name="hibernate.connection.username">root</property>
		<property name="hibernate.connection.password">root</property>
		 <property name="hibernate.show_sql">true</property>
		 <!-- 其它属性配置 -->
		<!-- 指明C3P0的提供者 -->
		<property name="hibernate.connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
		<!-- 连接池参数的配置 -->
		<property name="hibernate.c3p0.min_size">5</property> 
	    <property name="hibernate.c3p0.max_size">30</property> 
	    <property name="hibernate.c3p0.timeout">1800</property> 
	    <property name="hibernate.c3p0.max_statements">50</property>
	    
	    <!-- 打印SQL语句到控制台 -->
		<property name="hibernate.show_sql">true</property>
		<property name="hibernate.format_sql">true</property>
		<property name="hibernate.hbm2ddl.auto">update</property> 
		<!-- 注册实体的对象关系映射文件 -->
		
		<mapping resource="model/Category.hbm.xml"/>
		
	</session-factory>
</hibernate-configuration>

package util;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class HibernateUtils {

	private static SessionFactory factory;
	static{
		factory=new Configuration().configure().buildSessionFactory();
	}
	public static SessionFactory getFactory(){
		return factory;
	}
	
	public static Session getSession(){
		return factory.openSession();
	}
	
	public static void close(Session session){
		if(session!=null){
			session.close();
		}
		
	}
	
}

package test;


import java.util.HashSet;
import java.util.Set;

import model.Category;

import org.hibernate.Session;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.hibernate.tool.hbm2ddl.SchemaExport;
import org.junit.Test;

import util.HibernateUtils;

public class O2mTest {

	@Test
	public void createTable(){
		Configuration cfg=new Configuration().configure();
		SchemaExport se=new SchemaExport(cfg);
		se.create(true, true);
	}
	/**
	 * 先添加多的一端,再添加少的一端
	 */
	@Test
	public void save(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		//添加一级分类
		Category c1=new Category();
		c1.setName("计算机");
		session.save(c1);
		
		//添加二级分类
		Category c2=new Category();
		c2.setName("软件开发");
		c2.setParent(c2);
		session.save(c2);
		
		//添加二级分类
		Category c3=new Category();
		c3.setName("硬件维修");
		c3.setParent(c3);
		session.save(c3);
		
		ts.commit();
		HibernateUtils.close(session);
	}
	/**
	 * 先添加少的一端,再添加多的一端
	 */
	@Test
	public void save1(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		
		
		ts.commit();
		HibernateUtils.close(session);
	}
	/**
	 * Inverse = true,不指定cascade
    cascade的默认值为none, 当对一级分类进行保存操作时,二级分类什么都不做. 所以只保存了一级分类对象, 没有保存二级分类对象
	 */
	@Test
	public void save2(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		//添加一级分类
				Category c1=new Category();
				c1.setName("手机");
				
				//添加二级分类
				Category c2=new Category();
				c2.setName("小米");
				//添加二级分类
				Category c3=new Category();
				c3.setName("华为");
				
				Set<Category> c=new HashSet<>();
				c.add(c2);
				c.add(c3);
				c1.setClist(c);
				
				session.save(c1);
					
		ts.commit();
		HibernateUtils.close(session);
	}
	/**
	 * Inverse = true,cascade=all
    一级分类与二级分类对象,包扩外键[经过测试,发现外键没有成保存,值为null]都成功保存。
    
	 */
	@Test
	public void save4(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		//添加一级分类
				Category c1=new Category();
				c1.setName("手机");
				
				
				//添加二级分类
				Category c2=new Category();
				c2.setName("小米");
				//添加二级分类
				Category c3=new Category();
				c3.setName("华为");
				
				Set<Category> c=new HashSet<>();
				c.add(c2);
				c.add(c3);
				c1.setClist(c);
				
				session.save(c1);
					
		ts.commit();
		HibernateUtils.close(session);
	}
	
	/**
	 * Inverse = false,cascade=all
    一级分类与二级分类对象都能保存成功,包扩外键都成功保存。
    
	 */
	@Test
	public void save5(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		//添加一级分类
				Category c1=new Category();
				c1.setName("手机");
				
				
				//添加二级分类
				Category c2=new Category();
				c2.setName("小米");
				//添加二级分类
				Category c3=new Category();
				c3.setName("华为");
				
				Set<Category> c=new HashSet<>();
				c.add(c2);
				c.add(c3);
				c1.setClist(c);
				
				session.save(c1);
					
		ts.commit();
		HibernateUtils.close(session);
	}
	
	/**
	 * Inverse = false,不指定cascade=all
  报错。因为少的一端为主控方,负责维护关系,所以在插入少的一端对象后,会尝试修改并不存在的多的一端对象。  
	 */
	@Test
	public void save6(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		
		//添加一级分类
				Category c1=new Category();
				c1.setName("手机");
				
				
				//添加二级分类
				Category c2=new Category();
				c2.setName("小米");
				//添加二级分类
				Category c3=new Category();
				c3.setName("华为");
				
				Set<Category> c=new HashSet<>();
				c.add(c2);
				c.add(c3);
				c1.setClist(c);
				
				session.save(c1);
					
		ts.commit();
		HibernateUtils.close(session);
	}
	
	/**
	 * 
	 */
	@Test
	public void delete(){
		Session session=HibernateUtils.getSession();
		Transaction ts=session.beginTransaction();
		Category c=(Category)session.get(Category.class,7);
		session.delete(c);
		
		ts.commit();
		HibernateUtils.close(session);
	}
	
	
}


共有 人打赏支持
萧小蚁
粉丝 94
博文 278
码字总数 192339
×
萧小蚁
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: