轉(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/得到如下界面:
點(diǎn)擊注冊(cè)鏈接耕突,跳轉(zhuǎn)到注冊(cè)頁(yè)面:
輸入賬戶名及密碼笤成,點(diǎn)擊注冊(cè),若成功眷茁,則顯示如下圖
若該用戶名已被注冊(cè)炕泳,則提示對(duì)應(yīng)的錯(cuò)誤信息。
進(jìn)入springbootbase數(shù)據(jù)庫(kù)查看該注冊(cè)用戶信息是否存到user表里
select * from user;
說明我們的數(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