1>.学过hibernate的都知道用<id>来配置主键生成策略,显然,它只能配置主键是单列的表,对于联合主键的情况<id>就无能为力了。今天翻到个复合主键映射的资料,觉得还是有一定价值的,就写了下来,以备后用。
    2>.hibernate处理复合主键映射是基于这样一种做法:把目标类中所有主键关联属性抽出来,单独写成一个类(我暂且叫它主键类),目标类就只需持有主键类对象,而不必再包含各个主键属性;在映射文件中使用<composite-id>标签来配置主键对象并指出关联属性,普通属性照常配置;hibernate只会创建一张表在,并且把主键设置为主键类各属性的联合主键,存储和加载数据时,会自然的把关联对象各属性跟表中的主键字段对应起来。在这个操作过程中,它要求主键类必须是序列化的,并且要覆盖equals方法,最好覆盖hashCode方法。(我试验过了,不覆盖equals和hashCode,结果没有出现任何问题,而且从输出的sql语句看,也没有影响数据库操作,不知道它是否还有更深的含义?还希望高手给指点下)
    3>.例子。一个规模较大公司的部门表(hibernate_dept_compositePK),由所在区域(area),部门名(name),本部门人数(empCount),组建时间(birthday)等字段组成,我们使用所在区域和部门名做联合主键:
1.目标类:Department.java
public class Department { 
  /** 把主键关联属性抽象出来单独写成一个类 */ 
  //private String area; 
  //private String name; 
  /**把主键类对象作为成员变量*/ 
  private DepartmentPK departmentPK; 
  private int empCount; 
  private Date birthday;

//  public String getArea() { 
//    return area; 
//  } 
// 
//  public void setArea(String area) { 
//    this.area = area; 
//  } 
// 
//  public String getName() { 
//    return name; 
//  } 
// 
//  public void setName(String name) { 
//    this.name = name; 
//  } 
    
  public int getEmpCount() { 
    return empCount; 
  }

public void setEmpCount(int empCount) { 
    this.empCount = empCount; 
  }

public Date getBirthday() { 
    return birthday; 
  }

public void setBirthday(Date birthday) { 
    this.birthday = birthday; 
  }

public DepartmentPK getDepartmentPK() { 
    return departmentPK; 
  }

public void setDepartmentPK(DepartmentPK departmentPK) { 
    this.departmentPK = departmentPK; 
  }

}

2.主键类:DepartmentPK.java
public class DepartmentPK implements Serializable {

private static final long serialVersionUID = -288002855915204255L; 
  private String area; 
  private String name; 
  /** 
    * 覆盖hashCode方法(根据area和name判断) 
    */ 
  //@Override 
  public int hashCode() { 
    final int prime = 31; 
    int result = 1; 
    result = prime * result + ((area == null) ? 0 : area.hashCode()); 
    result = prime * result + ((name == null) ? 0 : name.hashCode()); 
    return result; 
  } 
  /** 
    * 覆盖equals(根据area和name判断) 
    */ 
  @Override 
  public boolean equals(Object obj) { 
    if (this == obj) 
      return true; 
    if (obj == null) 
      return false; 
    if (getClass() != obj.getClass()) 
      return false; 
    final DepartmentPK other = (DepartmentPK) obj; 
    if (area == null) { 
      if (other.area != null) 
        return false; 
    } else if (!area.equals(other.area)) 
      return false; 
    if (name == null) { 
      if (other.name != null) 
        return false; 
    } else if (!name.equals(other.name)) 
      return false; 
    return true; 
  }

public String getArea() { 
    return area; 
  }

public void setArea(String area) { 
    this.area = area; 
  }

public String getName() { 
    return name; 
  }

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

3.映射文件Department.hbm.xml
<?xml version="1.0"?> 
<!DOCTYPE hibernate-mapping PUBLIC    
  "-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
  "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> 
<hibernate-mapping> 
  <class name="com.yangfei.hibernate.compositePk.entity.Department" table="hibernate_dept_compositePK"> 
    <!-- 联合主键 --> 
    <!-- name指的是主键对象属性 --> 
    <composite-id name="departmentPK"> 
      <!-- 这里是主键关联属性 --> 
      <key-property name="area" /> 
      <key-property name="name" /> 
    </composite-id> 
    <!-- 其它属性 --> 
    <property name="empCount" length="4" /> 
    <property name="birthday" type="date" /> 
  </class> 
</hibernate-mapping>
4.hibernate配置文件hibernate.cfg.xml
<?xml version='1.0' encoding='UTF-8'?> 
<!DOCTYPE hibernate-configuration PUBLIC 
                    "-//Hibernate/Hibernate Configuration DTD 3.0//EN" 
                    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">

<!-- Generated by MyEclipse Hibernate Tools.                                     --> 
<hibernate-configuration> 
        <session-factory> 
                <property name="dialect">org.hibernate.dialect.Oracle9Dialect</property> 
                <property name="connection.url">jdbc:oracle:thin:@127.0.0.1:1521:orcl10</property> 
                <property name="connection.username">scott</property> 
                <property name="connection.password">yf123</property> 
                <property name="connection.driver_class">oracle.jdbc.driver.OracleDriver</property> 
                <property name="hibernate.show_sql">true</property> 
            
          <mapping resource="com/yangfei/hibernate/compositePk/entity/Department.hbm.xml"/> 
        </session-factory> 
</hibernate-configuration>

5.测试类:DepartmentTest.java
public class DepartmentTest extends TestCase { 
  /** 
    * 测试插入数据 
    */ 
  public void save() { 
    Session session = HibernateUtils.getSession(); 
    Transaction t = session.beginTransaction(); 
    try { 
      Department dept = new Department(); 
      /** 生成主键对象 */ 
      DepartmentPK deptPK = new DepartmentPK(); 
      deptPK.setArea("北京"); 
      deptPK.setName("研发部"); 
      dept.setDepartmentPK(deptPK);

dept.setEmpCount(100); 
      dept.setBirthday(new Date());

session.save(dept); 
      t.commit(); 
    } catch (HibernateException e) { 
      e.printStackTrace(); 
      t.rollback(); 
    } finally { 
      HibernateUtils.closeSession(session); 
    } 
  } 
  /** 
    * 测试加载数据 
    */ 
  public void load() { 
    Session session = HibernateUtils.getSession(); 
    Transaction t = session.beginTransaction(); 
    try { 
      /** 生成主键对象 */ 
      DepartmentPK deptPK = new DepartmentPK(); 
      deptPK.setArea("北京"); 
      deptPK.setName("研发部");

Department dept=(Department)session.load(Department.class, deptPK); 
      System.out.println(dept.getDepartmentPK().getArea()+","+dept.getDepartmentPK().getName()+","+dept.getEmpCount()+","+dept.getBirthday()); 
    } catch (HibernateException e) { 
      e.printStackTrace(); 
      t.rollback(); 
    } finally { 
      HibernateUtils.closeSession(session); 
    } 
  } 
    
  /** 
    * 测试修改数据 
    */ 
  public void update() { 
    Session session = HibernateUtils.getSession(); 
    Transaction t = session.beginTransaction(); 
    try { 
      /** 生成主键对象 */ 
      DepartmentPK deptPK = new DepartmentPK(); 
      deptPK.setArea("北京"); 
      deptPK.setName("研发部");

Department emp=(Department)session.load(Department.class, deptPK); 
      System.out.println(emp.getDepartmentPK().getArea()+","+emp.getDepartmentPK().getName()+","+emp.getEmpCount()+","+emp.getBirthday()); 
      emp.setEmpCount(100); 
      session.saveOrUpdate(emp); 
        
      /** 生成主键对象 */ 
      DepartmentPK deptPK2 = new DepartmentPK(); 
      deptPK2.setArea("北京"); 
      deptPK2.setName("研发部"); 
      Department dept=(Department)session.load(Department.class, deptPK2); 
      System.out.println(dept.getDepartmentPK().getArea()+","+dept.getDepartmentPK().getName()+","+dept.getEmpCount()+","+dept.getBirthday()); 
      t.commit(); 
    } catch (HibernateException e) { 
      e.printStackTrace(); 
      t.rollback(); 
    } finally { 
      HibernateUtils.closeSession(session); 
    } 
  } 
    
  /** 
    * 测试删除数据 
    */ 
  public void delete() { 
    Session session = HibernateUtils.getSession(); 
    Transaction t = session.beginTransaction(); 
    try { 
      /** 生成主键对象 */ 
      DepartmentPK deptPK = new DepartmentPK(); 
      deptPK.setArea("北京"); 
      deptPK.setName("研发部"); 
        
      Department dept=(Department)session.load(Department.class, deptPK); 
      session.delete(dept); 
      t.commit(); 
    } catch (HibernateException e) { 
      e.printStackTrace(); 
      t.rollback(); 
    } finally { 
      HibernateUtils.closeSession(session); 
    } 
  } 
}

4.一般来说,复合主键映射用起来是很复杂的,无论是开发时还是升级时。所以,人们往往宁愿选择增加个主键字段,也不尽量不采用它。
     本文转自NightWolves 51CTO博客,原文链接:http://blog.51cto.com/yangfei520/310555,如需转载请自行联系原作者

Hibernate的复合主键映射相关推荐

  1. NHibernate Mapping Attribute实现复合主键映射

    当我在学习使用NHibernate Mapping Attribute的时候,遇到复合主键映射问题,查询了很多资料,也向朋友请教过,由于相关资料不多,中间走了些弯路,好在最终较好解决了,现记录分享如下 ...

  2. Hibernate复合主键映射

    目录: 1. 实现方式一:将复合主键对应的属性与实体其他普通属性放在一起 2. 实现方式二:将主键属性提取到一个主键类中,实体类只需包含主键类的一个引用 在日常开发中会遇到这样一种情况,数据库中的某张 ...

  3. hibernate中表的复合主键映射表

    成绩表Cj:学号和课程号是复合主键 映射文件Cj.hbm.xml: <?xml version="1.0"?> <!DOCTYPE hibernate-mappi ...

  4. mapper中mysql复合语句_MyBatis定义复合主键

    前为别名后为主查询getXXX语句中字段 SELECT colid,colname FROM table1 SELECT * FROM table2 WHERE id = #{id} AND name ...

  5. hibernate之初学复合主键

    Jar包什么的就看前面几篇文章,接下来介绍复合主键知识 首先是hibernate的配置文件hibernate,cfg.xml基本没有什么变化 <!DOCTYPE hibernate-config ...

  6. Oracle-No.04 Oracle视图加主键解决hibernate复合主键问题

    2019独角兽企业重金招聘Python工程师标准>>> Oracle视图默认是没有主键的,因此在用hibernate反向生成java类时,会自动采用复合主键,多生成一个类名+ID的类 ...

  7. Hibenate主键映射和Hibernate常用的标示生成器

    l  Hibenate主键映射 自然主键:充当主键的字段具有一定的含义,是构成记录的组成部分. 代理主键:充当主键的字段本身没有什么含义 Hibernate常用的标示生成器 标示生成器名称 描述 As ...

  8. 对逻辑主键、业务主键和复合主键的思考

    这几天对逻辑主键.业务主键和复合主键进行了一些思考,也在网上搜索了一下相关的讨论,相关讨论可以看最下面的参考链接.下面是自己基于 SQL Server 做的一些总结,其他数据库(Oracle.MySQ ...

  9. SpringData JPA复合主键

    一.JPA和复合主键的简介        JPA全称Java Persistence API,是一组用于将数据存入数据库的类和方法的集合.JPA通过JDK5.0注解或XML描述对象-关系表的映射关系, ...

最新文章

  1. 吴恩达 coursera ML 第十七课总结+作业答案
  2. hosts文件中同一个域名两个IP的解析顺序
  3. logback-spring.xml配置文件
  4. 迟来总比没有好:SSE或服务器发送的事件现在已在JAX-RS中
  5. 跨站点脚本(XSS)和预防
  6. 在html中怎么写加起来的,css可以在html里面写吗?
  7. scrapy爬虫循环抓取同一个url,避免同一URL使爬虫关闭
  8. Android8. 0 学习(33)--- msensor移植
  9. Kafka监控KafkaOffsetMonitor【转】
  10. 使用GCC和GNU Binutils编写能在x86实模式运行的16位代码
  11. DOM判断节点类型分分钟搞定
  12. 如何在Mac上的“活动监视器”中运行系统诊断?
  13. 线性代数知识荟萃(4)——矩阵相抵
  14. Linux电源管理(1)_整体架构
  15. 科锐c语言,科锐C语言学习视频,资源教程下载
  16. cmake:cmake_minimum_required命令
  17. 2019级软件工程应用与实践-人工智能快递柜(代码分析9)
  18. 福建厦门:企业征信报告 可多渠道查询
  19. 解决:IOS无法使用微软账号登录,显示“无法登录,请使用IMAP并重新登录“
  20. Python语言程序设计(MOOC崇天)第三章基本数据类型学习笔记(天天向上的力量+文本进度条)

热门文章

  1. php文件包含绕过,文件包含漏洞(绕过姿势) | nmask's Blog
  2. linux命令-- pstack命令(跟踪进程栈)
  3. c++ 12.一维数组冒泡排序
  4. mmap mprotect详解
  5. linux查看上次重启日志和ip,Linux實時查看日志,訪問前10IP 和相關命令
  6. 5G NR Search space和CORESET
  7. Aptana:JavaScript开发利器
  8. jzoj_3385_黑魔法师之门
  9. [Material Design] 教你做一个Material风格、动画的button(MaterialButton)
  10. 【Centos7】安装memcached