相信大部分java小伙伴在日常開發過程中,在存儲操作層都會選擇更容易上手的jpa,各種封裝,通過注解等方式,簡化了大量的代碼內容,同時提升了開發效率,但是(敲黑板,永遠都躲不開的但是...)相應也將一些問題隱藏,需要更多的時間去理解、分析才能找到病根。
JPA簡介
JPA全稱(Java Persistence API),通常叫持久層API。它提供面向對象的編程模型,讓使用者更容易理解與上手;底層也提供了通用接口,可以實現與其它框架快速集成。下面整理一些基礎概念,不了解的可以參考下:
JPA實戰-級聯操作
目前小編所開發項目的數據持久層大部分采用的是Spring Data JPA,經過長時間的使用驗證,發現出了級聯保存問題,即調用repository的save方法后,發現數據結果卻是“丟三落四”,這就很讓人腦袋疼,下面結合代碼一起分析下
- 代碼場景:用戶與住址是多對多的關系,JPA中描述這種關系通常使用@ManyToMany注解,首先是創建對象
//維護端
public class User{
//屬性代碼省略
@ManyToMany(cascade = CascadeType.ALL,fetch = FetchType.LAZY)
@JoinTable(name = "user_addresses",
joinColumns = {@JoinColumn(name = "user_id",referencedColumnName = "user_id")},
inverseJoinColumns = {@JoinColumn(name = "address_id",referencedColumnName = "address_id")})
private Set< Address > addresses = new HashSet< >();
}
//被維護方
public class Address{
//屬性代碼省略
@ManyToMany(mappedBy = "addresses")
@JsonIgnore
private Set< User > users = new HashSet< >();
}
- 最初我只在維護端設置的級聯屬性(Cascade.ALL),然后通過維護端進行保存,很順利,User表、Address表、user_addresses關聯表都能保存,代碼如下,沒毛病。
// 維護端操作
public User save(){
Address address = new Address();
address.setAddressName("北京海淀");
User user1 = new User();
user1.setName("阿韋");
user1.getAddresses().add(address);
return UserService.save(User1);
}
- 然后我換了個思路,從被維護端能不能進行保存,然后就立即進行嘗試。代碼如下
// 被維護端操作
public Address save(){
Address address = new Address();
address.setAddressName("北京海淀");
User user1 = new User();
user1.setName("阿韋");
address.getUsers().add(user1);
return addressService.save(address);
}
- 結果,我以為能夠成功的時候,發現只保存了Address一張表,還沒報錯,就是User表和關聯表死活沒有數據。此時,問題已經復現,查閱各種文檔說明,大概是需要添加級聯屬性,改造內容如下
@ManyToMany(cascade = CascadeType.ALL,fetch = FetchType.EAGER,mappedBy = "addresses")
private Set< User > users = new HashSet< >();
- 修改之后再一次進行保存,兩個實體類的表確實都生成了數據,但是關聯表還是沒有,這時已經快要抓狂,但問題還是沒解決,繼續耐心分析,很快就在控制臺打印sql處發現了一些貓膩:
- 果然是沒有對關聯表的數據操作,接下來又是一頓探索,終于找到根源了,由被維護端發起操作的時候,需要為關聯類屬性添加彼此,感覺真是關聯到“恩愛”的地步了,羨煞旁人啊!!!
// 被維護端操作
public Address save(){
Address address = new Address();
address.setAddressName("中央花園");
User user1 = new User();
user1.setName("阿韋");
user1.setHeight("170");
user1.setSex("1");
//在彼此定義對方類集合中添加
user1.getAddresses().add(address);
address.getUsers().add(user1);
return addressService.save(address);
}
- 最終徹底解決掉了問題,數據持久化到相應的表中。自己也確實松了一口氣,過程雖然充滿曲折,但結果還是非常鼓舞自己的。
心得
希望本文能夠幫你避開jpa的坑,這就是小編最驕傲的事情。每一條道路上都是坑洼不平的,有些時候可能會掉入坑里,希望每個人都能重新站起來,跳過阻擋你的“坑”,最終會收獲到很多,讓自己變得更強。
-
數據
+關注
關注
8文章
7145瀏覽量
89587 -
存儲
+關注
關注
13文章
4355瀏覽量
86182 -
API
+關注
關注
2文章
1511瀏覽量
62401 -
編程
+關注
關注
88文章
3637瀏覽量
93989
發布評論請先 登錄
相關推薦
評論