spring data jpa 多对多 ManyToMany

环境搭建

源码地址:gitee:https://gitee.com/ytfs-dtx/JPA

导入依赖

<properties>

        <spring.version>5.2.5.RELEASE</spring.version>

        <hibernate.version>5.4.10.Final</hibernate.version>

        <slf4j.version>1.7.30</slf4j.version>

        <log4j.version>2.12.1</log4j.version>

        <druid.version>1.1.21</druid.version>

        <mysql.version>5.1.6</mysql.version>

    </properties>

    <dependencies>

        <!-- spring beg -->

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-aop</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <dependency>

            <groupId>org.aspectj</groupId>

            <artifactId>aspectjweaver</artifactId>

            <version>1.9.5</version>

        </dependency>

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-context</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-context-support</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-test</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <!-- spring对orm框架的支持包-->

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-orm</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-beans</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <dependency>

            <groupId>org.springframework</groupId>

            <artifactId>spring-core</artifactId>

            <version>${spring.version}</version>

        </dependency>

        <!-- hibernate beg -->

        <dependency>

            <groupId>org.hibernate</groupId>

            <artifactId>hibernate-entitymanager</artifactId>

            <version>${hibernate.version}</version>

        </dependency>

        <dependency>

            <groupId>org.hibernate</groupId>

            <artifactId>hibernate-core</artifactId>

            <version>${hibernate.version}</version>

        </dependency>

        <dependency>

            <groupId>org.hibernate.validator</groupId>

            <artifactId>hibernate-validator</artifactId>

            <version>6.1.2.Final</version>

            <exclusions>

                <exclusion>

                    <artifactId>classmate</artifactId>

                    <groupId>com.fasterxml</groupId>

                </exclusion>

            </exclusions>

        </dependency>

        <!-- hibernate end -->

        <dependency>

            <groupId>mysql</groupId>

            <artifactId>mysql-connector-java</artifactId>

            <version>${mysql.version}</version>

        </dependency>

        <dependency>

            <groupId>com.alibaba</groupId>

            <artifactId>druid</artifactId>

            <version>${druid.version}</version>

        </dependency>

        <!-- spring data jpa 的坐标 -->

        <dependency>

            <groupId>org.springframework.data</groupId>

            <artifactId>spring-data-jpa</artifactId>

            <version>2.2.6.RELEASE</version>

            <exclusions>

                <exclusion>

                    <artifactId>slf4j-api</artifactId>

                    <groupId>org.slf4j</groupId>

                </exclusion>

            </exclusions>

        </dependency>

        <!--  el beg 使用spring data jpa 必须引入 -->

        <dependency>

            <groupId>javax.el</groupId>

            <artifactId>javax.el-api</artifactId>

            <version>3.0.0</version>

        </dependency>

        <dependency>

            <groupId>org.glassfish</groupId>

            <artifactId>javax.el</artifactId>

            <version>3.0.0</version>

        </dependency>

        <dependency>

            <groupId>org.slf4j</groupId>

            <artifactId>slf4j-api</artifactId>

            <version>${slf4j.version}</version>

        </dependency>

        <dependency>

            <groupId>org.apache.logging.log4j</groupId>

            <artifactId>log4j-api</artifactId>

            <version>${log4j.version}</version>

        </dependency>

        <dependency>

            <groupId>junit</groupId>

            <artifactId>junit</artifactId>

            <version>4.12</version>

        </dependency>

    </dependencies>

创建配置文件

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"

      xmlns:context="http://www.springframework.org/schema/context"

      xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:tx="http://www.springframework.org/schema/tx"

      xmlns:jpa="http://www.springframework.org/schema/data/jpa"

      xmlns:task="http://www.springframework.org/schema/task"

      xsi:schemaLocation="

        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd

        http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd

        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd

        http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc.xsd

        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd

        http://www.springframework.org/schema/data/jpa

        http://www.springframework.org/schema/data/jpa/spring-jpa.xsd">

    <!-- 配置实体类管理工厂 -->

    <bean id="entityManagerFactoryBean" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">

        <property name="dataSource" ref="dataSource"/>

        <!-- 配置实体类的扫描 -->

        <property name="packagesToScan" value="xyz.ytfs.entity"/>

        <!-- 配置jpa的提供方 -->

        <property name="persistenceProvider">

            <bean class="org.hibernate.jpa.HibernatePersistenceProvider"/>

        </property>

        <!-- 配置提供方的适配器 -->

        <property name="jpaVendorAdapter">

            <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">

                <!-- 数据库的类型 -->

                <property name="database" value="MYSQL"/>

                <!-- 不自动创表 -->

                <property name="generateDdl" value="false"/>

                <!-- 控制台打印sql语句 -->

                <property name="showSql" value="true"/>

                <property name="databasePlatform" value="org.hibernate.dialect.MySQL55Dialect"/>

            </bean>

        </property>

        <!--  jpa的方言:高级特性 -->

        <property name="jpaDialect">

            <bean class="org.springframework.orm.jpa.vendor.HibernateJpaDialect"/>

        </property>

        <property name="jpaProperties">

            <props>

                <prop key="hibernate.hbm2ddl.auto">update</prop>

            </props>

        </property>

    </bean>

    <!-- 配置JPA的事务管理器 -->

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">

        <property name="entityManagerFactory" ref="entityManagerFactoryBean"/>

    </bean>

    <!-- JPA -->

    <jpa:repositories base-package="xyz.ytfs.dao" transaction-manager-ref="transactionManager"

                      entity-manager-factory-ref="entityManagerFactoryBean"/>

    <!-- 配置数据源 -->

    <context:property-placeholder location="classpath:jdbcConfig.properties"/>

    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">

        <property name="driverClassName" value="${jdbc.driver}"/>

        <property name="url" value="${jdbc.url}"/>

        <property name="username" value="${jdbc.username}"/>

        <property name="password" value="${jdbc.password}"/>

    </bean>

</beans>

创建实体类

注解说明

@ManyToMany

    作用:用于映射多对多关系

    属性:

        cascade:配置级联操作。

        fetch:配置是否采用延迟加载。

        targetEntity:配置目标的实体类。映射多对多的时候不用写。

@JoinTable

    作用:针对中间表的配置

    属性:

        nam:配置中间表的名称

        joinColumns:中间表的外键字段关联当前实体类所对应表的主键字段                         

        inverseJoinColumn:中间表的外键字段关联对方表的主键字段


@JoinColumn

    作用:用于定义主键字段和外键字段的对应关系。

    属性:

        name:指定外键字段的名称

        referencedColumnName:指定引用主表的主键字段名称

        unique:是否唯一。默认值不唯一

        nullable:是否允许为空。默认值允许。

        insertable:是否允许插入。默认值允许。

        updatable:是否允许更新。默认值允许。

        columnDefinition:列的定义信息。


User

package xyz.ytfs.entity;

import javax.persistence.*;

import java.util.HashSet;

import java.util.Set;

/**

* @author by ytfs

* @Classname User

* @Description TODO(用户类)

* @Date 2020/5/9 23:47

*/

@Entity

@Table(name = "sys_user")

public class User {

    @Id

    @GeneratedValue(strategy = GenerationType.IDENTITY)

    @Column(name = "user_id")

    private Long userId;

    @Column(name = "user_name")

    private String userName;

    /**

    * 配置用户到角色的多对多关系

    * 配置多对多的映射关系

    * 1.声明表关系的配置

    *

    * @ManyToMany(targetEntity = Role.class)  //多对多

    * targetEntity:代表对方的实体类字节码

    * 2.配置中间表(包含两个外键)

    * @JoinTable name : 中间表的名称

    * joinColumns:配置当前对象在中间表的外键

    * @JoinColumn的数组 name:外键名

    * referencedColumnName:参照的主表的主键名

    * inverseJoinColumns:配置对方对象在中间表的外键

    */

    @ManyToMany(targetEntity = Role.class, cascade = CascadeType.ALL)

    @JoinTable(name = "sys_user_role",

            //joinColumns,当前对象在中间表中的外键

            joinColumns = {@JoinColumn(name = "sys_user_id", referencedColumnName = "user_id")},

            //inverseJoinColumns,对方对象在中间表的外键

            inverseJoinColumns = {@JoinColumn(name = "sys_role_id", referencedColumnName = "role_id")})

    private Set<Role> roles = new HashSet<>();

    public Long getUserId() {

        return userId;

    }

    public void setUserId(Long userId) {

        this.userId = userId;

    }

    public String getUserName() {

        return userName;

    }

    public void setUserName(String userName) {

        this.userName = userName;

    }

    public Set<Role> getRoles() {

        return roles;

    }

    public void setRoles(Set<Role> roles) {

        this.roles = roles;

    }

    @Override

    public String toString() {

        return "User{" +

                "userId=" + userId +

                ", userName='" + userName + '\'' +

                ", roles=" + roles +

                '}';

    }

}

Role

package xyz.ytfs.entity;

import javax.persistence.*;

import java.util.HashSet;

import java.util.Set;

/**

* @author by ytfs

* @Classname Role

* @Description TODO(用户对应的类型)

* @Date 2020/5/9 23:47

*/

@Entity

@Table(name = "sys_role")

public class Role {

    @Id

    @GeneratedValue(strategy = GenerationType.IDENTITY)

    @Column(name = "role_id")

    private Long roleId;

    @Column(name = "role_name")

    private String roleName;

    /**

    * 配置用户到角色的多对多关系

    *      配置多对多的映射关系

    *          1.声明表关系的配置

    *              @ManyToMany(targetEntity = User.class)  //多对多

    *                  targetEntity:代表对方的实体类字节码

    *          2.配置中间表(包含两个外键)

    *                @JoinTable

    *                  name : 中间表的名称

    *                  joinColumns:配置当前对象在中间表的外键

    *                      @JoinColumn的数组

    *                          name:外键名

    *                          referencedColumnName:参照的主表的主键名

    *                  inverseJoinColumns:配置对方对象在中间表的外键

    */

    /*  @ManyToMany(targetEntity = User.class)

        @JoinTable(name = "sys_user_role",

            //joinColumns,当前对象在中间表中的外键

            joinColumns = {@JoinColumn(name = "sys_role_id", referencedColumnName = "role_id")},

            //inverseJoinColumns,对方对象在中间表的外键

            inverseJoinColumns = {@JoinColumn(name = "sys_user_id", referencedColumnName = "user_id")})*/

    @ManyToMany(mappedBy = "roles")

    private Set<User> users = new HashSet<>();

    public Set<User> getUsers() {

        return users;

    }

    public void setUsers(Set<User> users) {

        this.users = users;

    }

    public Long getRoleId() {

        return roleId;

    }

    public void setRoleId(Long roleId) {

        this.roleId = roleId;

    }

    public String getRoleName() {

        return roleName;

    }

    public void setRoleName(String roleName) {

        this.roleName = roleName;

    }

    @Override

    public String toString() {

        return "Role{" +

                "roleId=" + roleId +

                ", roleName='" + roleName + '\'' +

                ", users=" + users +

                '}';

    }

}

创建数据访问层的接口

IUserDao

package xyz.ytfs.dao;

import org.springframework.data.jpa.repository.JpaRepository;

import org.springframework.data.jpa.repository.JpaSpecificationExecutor;

import xyz.ytfs.entity.User;

/**

* @author by ytfs

* @Classname UserDao

* @Description TODO(用户的数据访问层接口)

* @Date 2020/5/10 0:05

*/

public interface IUserDao extends JpaRepository<User, Long>, JpaSpecificationExecutor<User> {

}

IRoleDao

package xyz.ytfs.dao;

import org.springframework.data.jpa.repository.JpaRepository;

import org.springframework.data.jpa.repository.JpaSpecificationExecutor;

import xyz.ytfs.entity.Role;

/**

* @author by ytfs

* @Classname IRoleDao

* @Description TODO(用户对应的职位数据访问层)

* @Date 2020/5/10 0:07

*/

public interface IRoleDao extends JpaRepository<Role, Long>, JpaSpecificationExecutor<Role> {

}

测试类

package xyz.ytfs.test;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.test.annotation.Rollback;

import org.springframework.test.context.ContextConfiguration;

import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import xyz.ytfs.dao.IRoleDao;

import xyz.ytfs.dao.IUserDao;

import xyz.ytfs.entity.Role;

import xyz.ytfs.entity.User;

import javax.transaction.Transactional;

import java.util.HashSet;

/**

* @author by ytfs

* @Classname JpaManyToManyTest

* @Description TODO(多对多的测试类)

* @Date 2020/5/10 0:08

*/

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations = "classpath:applicationContext.xml")

public class JpaManyToManyTest {

    @Autowired

    private IRoleDao roleDao;

    @Autowired

    private IUserDao userDao;

    /**

    * 保存一个用户,保存一个角色

    * <p>

    * 多对多放弃维护权:被动的一方(这里就是被选择的一方Role)放弃

    */

    @Test

    @Transactional

    @Rollback(false)

    public void testSave() {

        User user = new User();

        user.setUserName("雨听风说");

        Role role = new Role();

        role.setRoleName("程序猿");

        /*

        表达多对多的关系,双方都表达的时候必须有一方放弃维护中间表,不然会抛出异常

          */

        role.getUsers().add(user);

        user.getRoles().add(role);

        this.userDao.save(user);

        this.roleDao.save(role);

    }

    //测试级联添加(保存一个用户的同时保存用户的关联角色)

    @Test

    @Transactional

    @Rollback(false)

    public void testCasacdeSave() {

        User user = new User();

        user.setUserName("雨听风说");

        Role role = new Role();

        role.setRoleName("程序猿");

        /*

        表达多对多的关系

          */

        role.getUsers().add(user);

        user.getRoles().add(role);

        //级联操作,在保存用户的同时保存用户的类型

        this.userDao.save(user);

    }

    /**

    * 级联删除,删除id为1的客户

    */

    @Test

    @Transactional

    @Rollback(false)

    public void testDeleteCasaced() {

        //查询id为1的用户

        User user = this.userDao.getOne(1L);

        //删除id为1的用户,级联删除用户对应的类型

        this.userDao.delete(user);

    }

}

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