Spring Boot(六):Spring Boot 集成 hibernate & JPA

轉(zhuǎn)眼間,2018年的十二分之一都快過完了壹店,忙于各類事情,博客也都快一個(gè)月沒更新了芝加。今天我們繼續(xù)來學(xué)習(xí)Springboot對(duì)象持久化硅卢。

首先JPA是Java持久化API,定義了一系列對(duì)象持久化的標(biāo)準(zhǔn)藏杖,而hibernate是當(dāng)前非常流行的對(duì)象持久化開源框架将塑,Spring boot就默認(rèn)集成了這種框架,加速web應(yīng)用開發(fā)蝌麸。

1. 創(chuàng)建數(shù)據(jù)庫(kù)

Hibernate 可以自動(dòng)幫我們創(chuàng)建表点寥,但不能幫我們創(chuàng)建數(shù)據(jù)庫(kù),所以說創(chuàng)建數(shù)據(jù)庫(kù)及為數(shù)據(jù)庫(kù)指定用戶和權(quán)限来吩,這是我們必須要做的工作敢辩。本文用的是MySQL數(shù)據(jù)庫(kù)管理系統(tǒng)軟件蔽莱。

a) 首先用root進(jìn)入MySQL;

mysql -hlocalhost -uroot -proot;

其中:

-h 后面跟著MySQL主機(jī)名稱,我們這里將MySQL安裝在本地戚长,所以用localhost;

-u 后面跟著要訪問MySQL的用戶名盗冷,安裝成功后,默認(rèn)會(huì)有root用戶历葛,該用戶擁有訪問整個(gè)數(shù)據(jù)庫(kù)的所有權(quán)限;

-p 后面跟著該用戶的密碼嘀略,root用戶的密碼由安裝MySQL里指定恤溶,如若未指定,MySQL會(huì)隨機(jī)生成一個(gè)密碼帜羊,可在C:\Users\{username}\AppData\Roaming\si.data 里查看咒程。

b) 新創(chuàng)建一個(gè)數(shù)據(jù)庫(kù)

登錄MySQL成功后,我們可以用我們的超級(jí)用戶root為我們的項(xiàng)目創(chuàng)建一個(gè)新的database.

create database springbootbase;

執(zhí)行上面的sql腳本讼育,即可創(chuàng)建一個(gè)名為springbootbase新的空數(shù)據(jù)庫(kù)帐姻。

c) 為新數(shù)據(jù)庫(kù)springbootbase新增一張表

理論上,hibernate可以根據(jù)JPA POJO為我們自動(dòng)生成我們想要的表奶段,但是我個(gè)人覺得還是自己手動(dòng)創(chuàng)建會(huì)更好一點(diǎn)饥瓷,可以自己理解設(shè)計(jì)我們的關(guān)系型數(shù)據(jù),理清各張表的關(guān)聯(lián)情況痹籍。

接下來呢铆,我們來創(chuàng)建一張新表user

create table user (
  id int(11) auto_increment not null primary key, 
  username varchar(255) not null, 
  password varchar(255) not null, 
  role enum('ADMIN', 'USER') not null default 'USER');

該表有4個(gè)字段:id, username, password, role. 其中id類型為int(11), 自增,非空蹲缠,主鍵棺克;username, password類型為varchar(255), 非空;role類型為枚舉线定,非空娜谊,默認(rèn)值為USER。

d) 創(chuàng)建MySQL新用戶并授權(quán)

由于我們不能直接讓我們的web應(yīng)用直接使用MySQL root超級(jí)用戶斤讥,因?yàn)槿绻寃eb應(yīng)用直接使用root賬號(hào)纱皆,這樣會(huì)導(dǎo)致諸多風(fēng)險(xiǎn),比如我們?cè)驹O(shè)計(jì)web應(yīng)用只能訪問springbootbase數(shù)據(jù)庫(kù)芭商,但root卻也可以訪問其它database抹剩,這樣可能會(huì)導(dǎo)致其它信息暴露;再比如root可以刪除某個(gè)數(shù)據(jù)庫(kù)的權(quán)限蓉坎,如果不使用不當(dāng)澳眷,將某個(gè)數(shù)據(jù)庫(kù)刪除,那將悔之晚矣蛉艾。因?yàn)槲覀冇斜匾獮閣eb應(yīng)用單獨(dú)創(chuàng)建用戶钳踊,并授予相應(yīng)的權(quán)限來保證我們的數(shù)據(jù)庫(kù)安全衷敌。

create user 'onroad'@'localhost' identified by 'onroadtech';

該sql表示創(chuàng)建一個(gè)用戶名為onroad,密碼為onroadtech的新用戶拓瞪。

grant select, insert, update on springbootbase.user to 'onroad'@'localhost';

該sql表示授權(quán)onroad用戶對(duì)數(shù)據(jù)庫(kù)springbootbase具有查詢缴罗,插入及更新操作。一般普通用戶授權(quán)這三個(gè)權(quán)限即可滿足大部分web應(yīng)用開發(fā)需求祭埂。

到此面氓,我們的數(shù)據(jù)庫(kù)已搭建成功。

2. Spring boot 集成hibernate & JPA

我們還是以SpringBootBase這個(gè)項(xiàng)目為基礎(chǔ)來集成hibernate & JPA:

2.1. 添加Maven依賴

由于Spring boot默認(rèn)已經(jīng)集成了Hibernate, 所在我們只需在pom.xml引用jpa及mysql連接庫(kù).

<dependency>  
  <groupId>org.springframework.boot</groupId>  
  <artifactId>spring-boot-starter-data-jpa</artifactId>  
</dependency>  

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

2.2 在application.properties里配置數(shù)據(jù)庫(kù)地址及相應(yīng)的JPA

#----------------------database-------------------------
spring.datasource.url = jdbc:mysql://localhost:3306/springbootbase?useUnicode=true&characterEncoding=utf8
spring.datasource.username = onroad
spring.datasource.password = onroadtech
spring.datasource.driverClassName = com.mysql.jdbc.Driver

#----------------------JPA------------------------------
# Specify the DBMS
spring.jpa.database = MYSQL
# Show or not log for each sql query
spring.jpa.show-sql = true
# Hibernate ddl auto (create, create-drop, update)
spring.jpa.hibernate.ddl-auto = update
# Naming strategy
spring.jpa.hibernate.naming-strategy = org.hibernate.cfg.DefaultNamingStrategy

# stripped before adding them to the entity manager
spring.jpa.properties.hibernate.dialect = org.hibernate.dialect.MySQL5Dialect

2.3 創(chuàng)建一個(gè)實(shí)體類UserVO與之對(duì)應(yīng)

@Entity
@Table(name = "user") //引入@Table注解蛆橡,name賦值為表名
public class UserVO {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "id", nullable = false)
    private Long id;
    private String username;
    private String password;
    private String role;

    public Long getId() {
        return id;
    }

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

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public String getRole() {
        return role;
    }

    public void setRole(String role) {
        this.role = role;
    }

    @Override
    public String toString() {
        return "userame: " + username + ", password: " + password + "role: " + role;
    }

}

2.4 新建UserVORepository接口繼承CrudRepository舌界,實(shí)現(xiàn)對(duì)數(shù)據(jù)庫(kù)的增刪改查

public interface UserVORepository extends CrudRepository<UserVO, Long>{
  
    @Query("select u from UserVO u where u.username=:username")
    public UserVO findUserByName(@Param("username") String username);
  
    public UserVO findUserByUsername(String username);
    
}

其實(shí)上面這兩個(gè)接口是等價(jià)的,查詢的結(jié)果也是一樣的泰演,只是表達(dá)方式不一樣呻拌。

2.5 前端增加一個(gè)注冊(cè)頁(yè)面register.html

<!DOCTYPE html>
<html lang="zh-cn" xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="utf-8"/>
    
    <title>Register</title>
    
    <link rel="stylesheet" th:href="@{css/bootstrap.min.css}"/>
    <link rel="stylesheet" th:href="@{css/customer/login.css}"/>
</head>

<body>
    <h3 align="center">注冊(cè)</h3> 
    <div class="container">
        <form class="form-signin" th:action="@{/reg}" th:object="${user}" method="post">
            <input type="text" class="form-control" placeholder="賬號(hào)" th:field="*{username}"/>
            <input type="password" class="form-control" placeholder="密碼" th:field="*{password}"/>
            <p th:if="${param.success}" class="error-code">注冊(cè)成功</p>
            <p th:if="${param.error}" class="error-code">用戶名已被注冊(cè)</p>
            <button class="btn btn-lg btn-primary btn-block" type="submit">注冊(cè)</button>
        </form>
    </div>
</body>
</html>

2.6 首頁(yè)增加一個(gè)超鏈接到register頁(yè)面

<h2 class="form-signin-heading">請(qǐng) 登 錄     <a th:href="@{/register}">注冊(cè)</a></h2>

當(dāng)然這個(gè)鏈接請(qǐng)求也不能攔截, 所以需要在WebSecurityConfig類中的configure()方法加上:

http
    .authorizeRequests()
    .antMatchers("/", "/register", "/reg").permitAll()

2.7 再增加個(gè)RegisterController響應(yīng)register請(qǐng)求

@Controller
public class RegisterController {
    @Autowired UserVORepository userRepo;

    @RequestMapping(value ="/register", method = RequestMethod.GET)
    public String register(Model model, UserVO user) {
        model.addAttribute("user", user);
        return "register";
    }
    
    @RequestMapping(value ="/reg")
    public String addUser(Model model, UserVO user) {
        model.addAttribute("user", user);
        //UserVO isNewUser = userRepo.findUserByName(user.getUsername());
        UserVO isNewUser = userRepo.findUserByUsername(user.getUsername());
        //判斷該用戶名是否被注冊(cè)過
        if (null == isNewUser) {
            userRepo.save(user);
            return "redirect:register?success";
        } else {
            return "redirect:register?error";
        }
    }
}

點(diǎn)擊首頁(yè)注冊(cè)鏈接睦焕,跳轉(zhuǎn)到注冊(cè)頁(yè)面藐握,輸入注冊(cè)用戶名及密碼,點(diǎn)擊注冊(cè)垃喊,調(diào)用UserVORepository接口進(jìn)行查詢猾普,如果注冊(cè)的用戶不存在,則新增并且返回成功本谜,如果已存在抬闷,則返回錯(cuò)誤。

3. 驗(yàn)證

訪問https://localhost:8443/SpringBootBase/得到如下界面:

圖1

點(diǎn)擊注冊(cè)鏈接耕突,跳轉(zhuǎn)到注冊(cè)頁(yè)面:


圖2

輸入賬戶名及密碼笤成,點(diǎn)擊注冊(cè),若成功眷茁,則顯示如下圖


圖3

若該用戶名已被注冊(cè)炕泳,則提示對(duì)應(yīng)的錯(cuò)誤信息。


圖4

進(jìn)入springbootbase數(shù)據(jù)庫(kù)查看該注冊(cè)用戶信息是否存到user表里

select * from user;
圖5

說明我們的數(shù)據(jù)已經(jīng)正確寫到數(shù)據(jù)庫(kù)的user表里了上祈。


完整代碼可到我的github下載:
https://github.com/onroadtech/SpringbootBase/
branch: master
commit-id: 2af95a8a65ea3ed069502c58c026c4d8719eff99


本博文已同步發(fā)表于我的個(gè)人博客網(wǎng)站培遵,歡迎轉(zhuǎn)載指正并注明出處。
個(gè)人博客: www.onroad.tech
指正郵箱: onroad_tech@163.com

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末登刺,一起剝皮案震驚了整個(gè)濱河市籽腕,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌纸俭,老刑警劉巖皇耗,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異揍很,居然都是意外死亡郎楼,警方通過查閱死者的電腦和手機(jī)万伤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來呜袁,“玉大人敌买,你說我怎么就攤上這事〗捉纾” “怎么了虹钮?”我有些...
    開封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)膘融。 經(jīng)常有香客問我芙粱,道長(zhǎng),這世上最難降的妖魔是什么托启? 我笑而不...
    開封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任宅倒,我火速辦了婚禮攘宙,結(jié)果婚禮上屯耸,老公的妹妹穿的比我還像新娘。我一直安慰自己蹭劈,他們只是感情好疗绣,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著铺韧,像睡著了一般多矮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上哈打,一...
    開封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天塔逃,我揣著相機(jī)與錄音,去河邊找鬼料仗。 笑死湾盗,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的立轧。 我是一名探鬼主播格粪,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼氛改!你這毒婦竟也來了帐萎?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤胜卤,失蹤者是張志新(化名)和其女友劉穎疆导,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體葛躏,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡是鬼,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年肤舞,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片均蜜。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡李剖,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出囤耳,到底是詐尸還是另有隱情篙顺,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布充择,位于F島的核電站德玫,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏椎麦。R本人自食惡果不足惜宰僧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望观挎。 院中可真熱鬧琴儿,春花似錦、人聲如沸嘁捷。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)雄嚣。三九已至晒屎,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間缓升,已是汗流浹背鼓鲁。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留港谊,地道東北人骇吭。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像封锉,于是被迫代替她去往敵國(guó)和親绵跷。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容