JPA联合主键

​ 在实际工作中,我们会经常遇到联合主键的情况,所以我用简单例子列举JPA两种实现联合主键的方式。

1、如何通过@IdClass 实现联合主键

第一步:新建一个UserInfoID类,里面是联合主键

java 复制代码
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class UserInfoID implements Serializable {
    private String name, telephone;
}

第二步:新建一个UserInfo实体,采用@IdClass引用联合主键

java 复制代码
@Entity
@Data
@Builder
@IdClass(UserInfoID.class)
@NoArgsConstructor
@AllArgsConstructor
public class UserInfo {
    private Integer ages;

    @Id
    private String name;
    @Id
    private String telephone;
}

第三步:新建一个UserInfoRepsitory类完成CRUD

java 复制代码
@Repository
public interface UserInfoRepository  extends JpaRepository< UserInfo,UserInfoID> {
}

第四步:写一个测试

java 复制代码
@SpringBootTest
class SpringDataJpaTestApplicationTests {


    @Resource
    private UserInfoRepository userInfoRepository;

	@Test
	public void testIdClass() {
        userInfoRepository.save(UserInfo.builder().ages(29).name("sunfeng").telephone("18854885488").build());

        Optional<UserInfo> userInfoOptional = userInfoRepository.findById(UserInfoID.builder().name("sunfeng").telephone("18854885488").build());

        userInfoOptional.ifPresent(System.out::println);
    }

}

通过上面的例子可以发现表的主键是primary key(name, telephone),而Entity里面不是一个@Id字段了

2、@Embeddable与@EmbedDedId注解使用

第一步:我们在上面的例子中的UserInfoID里面添加@Embeddable注解

java 复制代码
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Embeddable
public class UserInfoID implements Serializable {
    private String name, telephone;
}

第二步:修改一下我们刚才的UserInfo对象,删除@IdClass添加 @EmbeddedId注解

java 复制代码
@Entity
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class UserInfo {
    private Integer ages;

    @EmbeddedId
    private UserInfoID userInfoID;
    
}

第三部:其他不变我们修改一下测试用例

java 复制代码
@SpringBootTest
class SpringDataJpaTestApplicationTests {


    @Resource
    private UserInfoRepository userInfoRepository;

    @Test
    public void testIdClass() {
        userInfoRepository.save(UserInfo.builder()
                .ages(29)
                .userInfoID(UserInfoID.builder()
                        .name("sunfeng")
                        .telephone("18854885488")
                        .build())
                .build());

        Optional<UserInfo> userInfoOptional = userInfoRepository.findById(UserInfoID.builder().name("sunfeng").telephone("18854885488").build());

        userInfoOptional.ifPresent(System.out::println);
    }

运行完之后,可以得到相同的结果。那么两种方式之间的区别是什么?

  1. 在使用的时候 @EmbeddedId使用的是对象,而IdClass用的是具体的字段
  2. 二者的JPQL也会不一样
    1. @IdClass的JPQL写法是:SELECT u.name FROM UserIndo u
    2. @EmbeddedId的JPQL写法是:select u.userInfo.name FROM UserInfo u
相关推荐
yunmi_14 分钟前
2024版 IDEA 用 Maven 创建 java 项目(+Maven 安装和配置)
java·maven
float_六七19 分钟前
Spring事务注解@Transactional核心机制详解
java·后端·spring
渣哥21 分钟前
Java 线程安全详解:定义、常见问题与解决方案
java
没有bug.的程序员21 分钟前
Redis 大 Key 与热 Key:生产环境的风险与解决方案
java·数据库·redis·缓存·热key·大key
We....28 分钟前
Java分布式编程:RMI机制
java·开发语言·分布式
玉衡子31 分钟前
七、InnoDB底层原理与日志机制
java·mysql
€81136 分钟前
Java入门级教程17——利用Java SPI机制制作验证码、利用Java RMI机制实现分布式登录验证系统
java·开发语言·java spi机制·远程传输数据
2301_8153577041 分钟前
parameterType和@Param注解的区别
java·开发语言·数据库
We....1 小时前
Java 分布式缓存实现:结合 RMI 与本地文件缓存
java·分布式·缓存
2401_845417451 小时前
set和map
java·开发语言