JPA学习笔记五 单向多对一

建表语句,虽然可以自动建表,但我就是想皮一下,开心

CREATE TABLE t_user (
    id INT NOT NULL auto_increment,
    `name` VARCHAR (20),
    PRIMARY KEY (id)
) ENGINE = INNODB DEFAULT charset = utf8;

CREATE TABLE t_orders (
    id INT NOT NULL auto_increment,
    `name` VARCHAR (20),
    uid INT,
    PRIMARY KEY (id),
    CONSTRAINT t_user_id FOREIGN KEY (uid) REFERENCES t_user (id)
) ENGINE = INNODB DEFAULT CHARSET = utf8;


persistence.xml文件

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0"
             xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
    <persistence-unit name="jpa" transaction-type="RESOURCE_LOCAL">

        <!--
        配置用什么ORM框架
        1. 实际上配置的是 javax.persistence.spi.PersistenceProvider 接口的实现类
        2. 如果JPA项目中只有一个JPA的实现产品,则可以不配置该节点
         -->
        <provider>org.hibernate.ejb.HibernatePersistence</provider>

        <!-- 添加持久化类 -->
        <class>com.yjj.entity.Order</class>
        <class>com.yjj.entity.User</class>

        <properties>
            <!-- 配置数据源信息 -->
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
            <property name="javax.persistence.jdbc.url" value="jdbc:mysql:///jpa"/>
            <property name="javax.persistence.jdbc.user" value="root"/>
            <property name="javax.persistence.jdbc.password" value="123456"/>

            <!-- 配置JPA实现产品的属性,即hibernate的属性 -->
            <property name="hibernate.format_sql" value="true"/><!-- 是否格式化sql语句 -->
            <property name="hibernate.show_sql" value="true"/> <!-- 是否在控制台打印sql语句 -->
            <property name="hibernate.hbm2ddl.auto" value="update"/>
        </properties>
    </persistence-unit>
</persistence>

实体类
一的一方表中就只有id和name,那么属性也就只有id和name

package com.yjj.entity;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Table;
/**
 * 单向多对一
 * t_user表字段,id,name
 */
@Table(name = "t_user")
@Entity
public class User {
    @Id
    @GeneratedValue
    private Integer id;

    private String name;

    public Integer getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

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

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }
}


多的一方表中有id,name,uid(外键对应t_user表的id),那么属性就有id,name,把uid换成对应的实体类User,维护一个User对象

package com.yjj.entity;

import javax.persistence.*;
/**
 * 单向多对一
 * t_orders表字段,id,name,uid
 */
@Table(name = "t_orders")
@Entity
public class Order {
    @Id
    @GeneratedValue
    private Integer id;
    private String name;
    @JoinColumn(name = "uid")
    @ManyToOne
    private User user;

    public Integer getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

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

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    @Override
    public String toString() {
        return "Order{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", user=" + user +
                '}';
    }
}


测试类

      User user =new User();
        user.setName("殷俊杰");
        Order order=new Order();
        order.setName("订单一");
        order.setUser(user);
        Order order1=new Order();
        order1.setName("订单二");
        order1.setUser(user);

        manager.persist(user);
        manager.persist(order);
        manager.persist(order1);
        

输出

Hibernate: 
    insert 
    into
        t_user
        (name) 
    values
        (?)
Hibernate: 
    insert 
    into
        t_orders
        (name, uid) 
    values
        (?, ?)
Hibernate: 
    insert 
    into
        t_orders
        (name, uid) 
    values
        (?, ?)

单向多对一保存的时候必须先保存一方,否则会出现多余的update语句,从而影响性能
试一下先保存多的一方,再保存一的一方

        manager.persist(order);
        manager.persist(order1);
        manager.persist(user);

输出

Hibernate: 
    insert 
    into
        t_orders
        (name, uid) 
    values
        (?, ?)
Hibernate: 
    insert 
    into
        t_orders
        (name, uid) 
    values
        (?, ?)
Hibernate: 
    insert 
    into
        t_user
        (name) 
    values
        (?)
Hibernate: 
    update
        t_orders 
    set
        name=?,
        uid=? 
    where
        id=?
Hibernate: 
    update
        t_orders 
    set
        name=?,
        uid=? 
    where
        id=?

结论:如果先插入order,它的USER_ID字段会先为空,在插入user之后,再进行update操作,USER_ID。因此,在多对一映射关系的情况下,建议先插入“一”的对象(user),再插入“多”的对象(order),减少sql语句可以提高效率
查找

        Order order=manager.find(Order.class,1);
        System.out.println(order);

输出

Hibernate: 
    select
        order0_.id as id1_1_0_,
        order0_.name as name2_1_0_,
        order0_.uid as uid3_1_0_,
        user1_.id as id1_3_1_,
        user1_.name as name2_3_1_ 
    from
        t_orders order0_ 
    left outer join
        t_user user1_ 
            on order0_.uid=user1_.id 
    where
        order0_.id=?
Order{id=1, name='订单一', user=User{id=1, name='殷俊杰'}}

@ManyToOne(fetch=FetchType.LAZY)在ManyToOne注解中可加入FetchType.LAZY延迟加载,在用到一的对象时才真正查询一的对象
删除

 Order order=entityManager.find(Order.class,1);
 entityManager.remove(order);

输出

Hibernate: 
    select
        order0_.id as id1_0_0_,
        order0_.ordername as ordernam2_0_0_,
        order0_.userid as userid3_0_0_,
        user1_.id as id1_3_1_,
        user1_.name as name2_3_1_ 
    from
        t_order order0_ 
    left outer join
        user user1_ 
            on order0_.userid=user1_.id 
    where
        order0_.id=?
Hibernate: 
    delete 
    from
        t_order 
    where
        id=?

不会关联删除user因为还有其他订单关联着user,如果先删除user的话会报错,因为外键关联。
更新

 Order order=entityManager.find(Order.class,2);
 order.getUser().setName("霸波尔奔");
 transaction.commit();

输出

Hibernate: 
    select
        order0_.id as id1_0_0_,
        order0_.ordername as ordernam2_0_0_,
        order0_.userid as userid3_0_0_,
        user1_.id as id1_3_1_,
        user1_.name as name2_3_1_ 
    from
        t_order order0_ 
    left outer join
        user user1_ 
            on order0_.userid=user1_.id 
    where
        order0_.id=?
Hibernate: 
    update
        user 
    set
        name=? 
    where
        id=?

更新时不用persist再保存,提交事务时会自动将缓存保存到数据库,但是一定要提交事务。
很烦,不想写注释,反正自己看
参考文章
http://blog.csdn.net/je_ge/article/details/53493897

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

推荐阅读更多精彩内容