Hibernate注解

JPA

Java Persistence API----java持久化接口

Hibernate注解分类

  • 类级别注解

    • @Entity

      映射实体类

      @Entity(name = "tableName")

      name可选,对应数据库中的一个表,若表名与实体类名相同,则可以省略

      注意:使用@Entity时必须指定实体类的主键属性

    • @Table

      与@Entity配合使用,只能标注在实体class定义处,表示实体对应的数据库表的信息

      @Table(name = "", schema = "",catalog = "")

      name可选,映射表的名称,默认表名与实体名称一致,只有在不一致的情况下才需要指定表名

      catelog(目录)可选,表示Catalog名称,默认为Catalog("")

      sahema(模式)可选,表示Schema名称,默认为Schema("")

      ![](C:\Users\lirui\Desktop\java web\schema与catalog.png)

      ![](C:\Users\lirui\Desktop\java web\2.schema与catalog.png)

    • @Embeddable 嵌入类

      表示一个非Entity类,可以嵌入到另一个Entity类中作为属性而存在

  • 属性级别注解

    • 添加方式

      • 写在属性上面
      • 写在属性的get访问器上面
    • @Id

      必须,定义了映射到数据库表的主键属性,一个实体类可以有一个或多个属性被映射成主键

      注意:

      ​ 1.如果有多个属性定义为主键属性,该实体类必须实现Serializable接口

      ​ 2.字符串作为主键,长度不能过长,需要指定长度 @Column(length=8)

    • @SequenceGenerator

    • @GeneratedValue
      参考:http://blog.csdn.net/fancylovejava/article/details/7438660

      @GeneratedValue(strategy=GenerationType,generator = "")

      可选,用于定义主键生成策略

      strategy

      表示主键生成策略,取值有:

      ​ 1.GenerationType.AUTO:根据底层数据库自动选择(默认,mysql默认为自动增长型)

      ​ 2.GenerationType.INDENTITY:根据数据库Identity字段生成

      ​ 3.GenerationType.SEQUENCE:使用Sequence来决定主键的取值

      ​ 4.GenerationType.TABLE:使用指定表来决定主键取值,结合@TableGenerator使用

          @Id
          @TableGenerator(name="tab_cat_gen",allocationSize=1)
          @GeneratedValue(strategy=GenerationType.TABLE)
      

      generator

      表示主键的生成器名称,这个属性通常和ORM框架相关,例如Hibernate可以指定uuid等主键的生成方式

      主键为String,使用手工设置主键的方式,定义如下:

      @Id
      @GeneratedValue(generator = "生成器名称")
      @GenericGenerator(name = "生成器名称",strategy = "assigned")
      @Column(length = 8)
      
    • @Column

      可将属性映射到列,使用该注解来覆盖默认值,描述了数据库表中该字段的详细定义,这对于根据JPA注解生成的数据库表结构的工具非常有作用

      name:可选,表示数据库表中该字段的名称,默认情形下属性名称一致

      nullable:可选,表示该字段是否允许为null,默认为true

      unique:可选,表示该字段是否为唯一标识,默认为false

      length:可选,表示该字段大小,仅对String类型的字段有效,默认为255(若为主键不能使用默认值)

      insertable:可选,表示在ORM框架执行插入操作时,该字段是否应出现在Insetrt语句中,默认为true

      updateable:可选,表示在ORM框架执行更新操作时,该字段是否应出现在UPDATE语句中,默认为true。对于一经创建就不可以更改的字段,该属性非常有用

    • @Embedded

      用于注解属性,表示该属性的类是嵌入类,注意同时嵌入类也必须标注@Embeddable注解

    • @EmbeddedId

      使用嵌入式主键类实现复合主键

      注意:嵌入式主键类必须实现Serializable接口,必须有默认的public 无参构造方法,必须覆盖equals和hashCode方法,使用@Embeddable注解嵌入式主键类

    • @Lob

    • @Version

    • @Basic

    • @Transient

      可选,表示该属性并非一个到数据库表的字段的映射,ORM将忽略该属性,如果一个属性并非数据库表的字段映射,就务必标示为@Transient,否则ORM框架默认其注解为@Basic

  • 映射关系注解

    备注:cascade定义级联

    mappedBy定义外键关系维护,使用该注解的一方不进行外键维护,所以,所以推荐级联方定义在另一方,使另一方维护外键,进行级联

    • 一对一单向外键

      主表类A与从表类B的主键值相对应

      主表A:
      @OneToOne(cascade = CascadeType.ALL)
      @PrimaryKeyJoinColumn
      public B getB(){...}
      
      从表B:
      无
      

      主表A中有一个从表属性是B类型的b

      主表A:
      @OneToOne(cascade=CascadeType.All)
      @JoinColumn(name="pid",unique=true)
      public B getB(){...}
      
      从表B:
      无
      

      注意:保存时应先保存外键对象,再保存主表对象

      未测试

    • 一对一双向外键关联

      双向关联必须设置mappedBy属性,因为双向关联只能交一方控制,不可能双方都设置外间保存关联关系,否则双方都无法保存

      主表A中有一个从表属性是B类型的b,同时,从表B中有一个主表属性是A类型的a

      主表A:
      @OneToOne(cascade = CascadeType.ALL)
      @JoinColumn(name="主表外键")//这里指定的是数据库中的外键字段。
      public B getB(){...}
      
      从表B:
      @OneToOne(mappedBy = "b")
      public A getA(){...}
      

      注意:@JoinColumn是可选的。默认值是从表变量名+"_"+从表的主键(注意,这里加的是主键。而不是主键对应的变量)。

      未测试

    • 一对一单向外键联合主键

      • 创建主键类,主键类必须实现Serializable接口,重写hashCode()和equals()方法,并添加注解@Embeddable注解
      • 实体类中主键添加@EmbeddedId注解

      未测试

    • 多对一单向外键关联

      多方有一方的引用

      多方
      @ManyToOne(cascade = {CascadeType.ALL},fetch = FetchType.EAGER)
      @JoinColumn(name = "外键名",referencedColumnName = "关联表的列名")
      private One one;
      
      一方:
      无
      

      测试通过,cascade定义级联,可在多方添加cascade,添加后可以省略一方的保存,直接保存多方数据即可级联,也可不添加cascade

    • 一对多单向外键关联

      一方持有多方集合Set

      一方
      @OneToMany(cascade = {CascadeType.ALL},fetch = FetchType.LAZY)
      @JoinColumn(name = "外键名")
      private Set many=new HashSet<Many>();
      
      多方:
      无
      

      测试通过,cascade定义级联,可在一方添加cascade,添加后可以省略多方的保存,直接保存单方数据即可级联,也可不添加cascade

    • 一对多双向外键关联

      多方持有一方引用,一方持有多方集合Set

      一方
      @OneToMany(cascade = {CascadeType.ALL},fetch = FetchType.LAZY)
      @JoinColumn(name = "外键名")
      private Set many=new HashSet<Many>();
      
      多方
      @ManyToOne(cascade = {CascadeType.ALL},fetch = FetchType.EAGER)
      @JoinColumn(name = "外键名",referencedColumnName = "关联表的列名")
      private One one;
      
      //备注:不推荐此种写法,推荐下方写法,因为下方方法采用了关系维护,使用多方维护关系,会省略update语句,提高执行效率
      

       一方
       @OneToMany(mappedBy = "one")
       private Set many=new HashSet<Many>();
       
       多方
       @ManyToOne(cascade = CascadeType.ALL)
       @JoinColumn(name = "lkm_cust_id", referencedColumnName = "cust_id", nullable = false)
       private One one;
      

      测试通过,cascade定义级联,cascade可以添加在维护外键的一方(多方),或均不添加,根据添加情况可简化保存代码

    • 多对多单向外键关联

      其中有一个多方持有另一个多方的集合对象

      //关系属性
      //EAGER: 查询时关联的数据也同时取出放入内存中,因为在内存里,所以在session外也可以取

      @ManyToMany
      @JoinTable(name="中间表名"
                 ,joinColumns={@JoinColumn(name="自己的外键")} 
                 ,inverseJoinColumns={@JoinColumn(name="对方的外键")})
      private Set many1=new HashSet<Many1>();
      

      备注:JoinTable用于维护中间表,name中间表表名,两个外键: joinColumns=指向自己的外键 inverseJoinColumns=指向对方的外键

      测试通过

    • 多对多双向外键关联

      双方持有对方集合

      一方:
        @ManyToMany(mappedBy="")
        private Set many1=new HashSet<Many1>();
      
        另一方:
        @ManyToMany(cascade = CascadeType.ALL)
        @JoinTable(name="中间表名"
                   ,joinColumns={@JoinColumn(name="自己的外键")} 
                   ,inverseJoinColumns={@JoinColumn(name="对方的外键")})
        private Set many2=new HashSet<Many2>();
      

      测试通过,cascade定义级联,cascade可以添加在维护外键的一方

    • 中间表持有双方引用

      • 新建复合主键(参考建立联合主键的第三种方式)

        包含两个外键的普通javaBean对象(必须实现Serializable接口,必须有默认的public 无参构造方法,必须覆盖equals和hashCode方法)

      • 引用复合对象类@IdClass(复合对象.class)

      • 声明两个外键@Id注解

      • 中间表持有双方引用并注解

        @Entity
        @IdClass(SysUserRolePK.class)
        public class SysUserRole {
            private long roleId;
            private long userId;
        
            @Id
            @Column(name = "role_id")
            public long getRoleId() {...}
        
            public void setRoleId(long roleId) {...}
        
            @Id
            @Column(name = "user_id")
            public long getUserId() {...}
        
            public void setUserId(long userId) {...}
            
            @ManyToOne
          @JoinColumn(name = "role_id", referencedColumnName = "role_id", nullable =      false)
          public SysRole getSysRoleByRoleId() {
              return sysRoleByRoleId;
          }
        
          @ManyToOne
          @JoinColumn(name = "user_id", referencedColumnName = "user_id", nullable = false)
          public SysUser getSysUserByUserId() {
              return sysUserByUserId;
          }
        }
        

联合主键

必须实现Serializable接口,必须有默认的public 无参构造方法,必须覆盖equals和hashCode方法

  • 存在的意义(以下仅为猜测):

    数据库通过主键来确定是不是同一条数据,在内存中则需要将主键映射成实现equals和hashCode方法的类,通过hashcode来确定是不是同一条数据

  • 实现联合主键的三种方式

    • 将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并重写equals和hascode,再将该类注解为@Embeddable,最后在主类中(该类不包含联合主键类中的字段)保存该联合主键类的一个引用,并生成set和get方法,并将该引用注解为@Id

    • 将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并重写equals和hascode,最后在主类中(该类不包含联合主键类中的字段)保存该联合主键类的一个引用,并生成set和get方法,并将该引用注解为@EmbeddedId

    • 将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并要重写equals和hashcode.最后在主类中(该类包含联合主键类中的字段)将联合主键字段都注解为@Id,并在该类上方将上这样的注解:@IdClass(联合主键类.class)

      @Entity
      @IdClass(SysUserRolePK.class)
      public class SysUserRole {
          private long roleId;
          private long userId;
      
          @Id
          @Column(name = "role_id")
          public long getRoleId() {...}
      
          public void setRoleId(long roleId) {...}
      
          @Id
          @Column(name = "user_id")
          public long getUserId() {...}
      
          public void setUserId(long userId) {...}
      }
      
      public class SysUserRolePK implements Serializable {
          private long roleId;
          private long userId;
      
          @Column(name = "role_id")
          @Id
          //以上两行存在不存在均可
          public long getRoleId()  {...}
      
          public void setRoleId(long roleId) {...}
      
          @Column(name = "user_id")
          @Id
          //以上两行存在不存在均可
          public long getUserId()  {...}
      
          public void setUserId(long userId) {...}
      
          @Override
          public boolean equals(Object o) {...}
      
          @Override
          public int hashCode() {...}
      }
      
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容