久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術(shù)文章
文章詳情頁

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

瀏覽:3日期:2022-08-11 16:26:10
目錄前言一、何為鑒權(quán)服務二、利用servlet+jdbc實現(xiàn)簡單的用戶登錄程序1.明確思路2.手把手教你實現(xiàn)一個簡單的web登錄程序三、回顧1.密碼未加密裸奔2.登錄信息未存儲3.對于其他資源并未進行權(quán)限管理四、優(yōu)化設計1.密碼加密存儲2.存儲登錄信息3.對資源進行管理五、關(guān)于鑒權(quán)問題1.Cookie/Session機制2.用Cookie/Session解決鑒權(quán)問題?3.使用token機制解決鑒權(quán)問題六、用SpringBoot+SSM實現(xiàn)一套簡單的鑒權(quán)服務(注冊,登錄,權(quán)限控制)1.注冊服務2.登錄服務3.權(quán)限控制(攔截器)七、效果展示1.注冊2.登錄3.訪問其他資源寫在最后前言

時遇JavaEE作業(yè),題目要求寫個簡單web登錄程序,按照老師的意思是用servlet、jsp和jdbc完成。本著要么不做,要做就要做好的原則,我開始著手完成此次作業(yè)(其實也是寫實訓作業(yè)的用戶鑒權(quán)部分),而之前寫項目的時候也有相關(guān)經(jīng)驗,這次正好能派上用場。

一、何為鑒權(quán)服務

引用百度百科的話說

鑒權(quán)(authentication)是指驗證用戶是否擁有訪問系統(tǒng)的權(quán)利。

鑒權(quán)包括兩個方面:

用戶鑒權(quán),網(wǎng)絡對用戶進行鑒權(quán),防止非法用戶占用網(wǎng)絡資源。網(wǎng)絡鑒權(quán),用戶對網(wǎng)絡進行鑒權(quán),防止用戶接入了非法的網(wǎng)絡,被騙取關(guān)鍵信息。

而我們這里的鑒權(quán)主要指用戶鑒權(quán),即如何確認“你是你”。最簡單的體現(xiàn)便是平常用的用戶登錄登出。

現(xiàn)今大部分系統(tǒng)都會有自己的鑒權(quán)服務,它是用戶與系統(tǒng)交互的第一步,系統(tǒng)需要一系列步驟明白你是誰,你可以做哪些事,明白了這些之后它才能更好的服務于你。

二、利用servlet+jdbc實現(xiàn)簡單的用戶登錄程序1.明確思路

首先,我們要仔細思考一下我們到底需要什么?

先讓我們回想一下一般的登錄是如何做的呢?

對于網(wǎng)頁,首先會出現(xiàn)一個登錄頁面,然后呢,輸入賬號密碼,點擊登錄,就會彈出成功/失敗的頁面。

那如何去判斷成功/失敗呢?

思考一下,最簡單的方法便是拿到前端傳來的數(shù)據(jù)之后便將其拿到數(shù)據(jù)中去查,看看密碼是不是一樣,然后給前端回復說——我找到了,他就是XXX或者我找不到他的記錄,讓他重新輸入賬號密碼。

然后前端對此回復做出相應的操作,比如登錄成功便跳轉(zhuǎn)到首頁,失敗讓用戶重新輸入。

2.手把手教你實現(xiàn)一個簡單的web登錄程序

出于某些原因,我這里手把手教你如何實現(xiàn)一個簡單的web登錄程序。

①創(chuàng)建web項目

打開idea,新建一個web項目

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

這里為了方便jar包的管理,選擇maven結(jié)構(gòu)的項目(至于什么是maven結(jié)構(gòu),不懂的可以百度,了解概念即可),然后選擇從原型創(chuàng)建,選擇webapp(這里只是方便,你也可以選擇空項目,不過會費點時間)。

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

點擊下一步,輸入項目名稱

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

這里選擇相應的maven,idea里有自帶的maven和jar包倉庫,不過我是自己去官網(wǎng)下了一個(不下也完全可以)。

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

選擇完成,這樣一個最簡單的項目結(jié)構(gòu)就出來了。

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

接下來需要配置一下pom.xml,因為要用到jdbc和tomcat的jar包(畢竟都是調(diào)用人家的接口(笑哭))

<dependencies> <dependency> <groupId>org.apache.tomcat.embed</groupId> <artifactId>tomcat-embed-core</artifactId> <version>9.0.37</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.20</version> </dependency> </dependencies>

(加在project標簽里就行),上面配置的意思就是導入兩個第三方工具包

②編寫簡單的登錄頁面

這里我既想要好看,又想偷懶,所以用了layui框架的模板

<!DOCTYPE html><html><head> <meta charset='UTF-8'> <title>后臺管理-登陸</title> <meta http-equiv='X-UA-Compatible' content='IE=edge,chrome=1'> <meta http-equiv='Access-Control-Allow-Origin' content='*'> <meta name='viewport' content='width=device-width, initial-scale=1, maximum-scale=1'> <meta name='apple-mobile-web-app-status-bar-style' content='black'> <meta name='apple-mobile-web-app-capable' content='yes'> <meta name='format-detection' content='telephone=no'> <link rel='stylesheet' href='http://www.gepszalag.com/lib/layui-v2.6.3/css/layui.css' media='all'> <!--[if lt IE 9]> <script src='https://cdn.staticfile.org/html5shiv/r29/html5.min.js'></script> <script src='https://cdn.staticfile.org/respond.js/1.4.2/respond.min.js'></script> <![endif]--> <style>.main-body {top:50%;left:50%;position:absolute;-webkit-transform:translate(-50%,-50%);-moz-transform:translate(-50%,-50%);-ms-transform:translate(-50%,-50%);-o-transform:translate(-50%,-50%);transform:translate(-50%,-50%);overflow:hidden;}.login-main .login-bottom .center .item input {display:inline-block;width:227px;height:22px;padding:0;position:absolute;border:0;outline:0;font-size:14px;letter-spacing:0;}.login-main .login-bottom .center .item .icon-1 {background:url(../images/icon-login.png) no-repeat 1px 0;}.login-main .login-bottom .center .item .icon-2 {background:url(../images/icon-login.png) no-repeat -54px 0;}.login-main .login-bottom .center .item .icon-3 {background:url(../images/icon-login.png) no-repeat -106px 0;}.login-main .login-bottom .center .item .icon-4 {background:url(../images/icon-login.png) no-repeat 0 -43px;position:absolute;right:-10px;cursor:pointer;}.login-main .login-bottom .center .item .icon-5 {background:url(../images/icon-login.png) no-repeat -55px -43px;}.login-main .login-bottom .center .item .icon-6 {background:url(../images/icon-login.png) no-repeat 0 -93px;position:absolute;right:-10px;margin-top:8px;cursor:pointer;}.login-main .login-bottom .tip .icon-nocheck {display:inline-block;width:10px;height:10px;border-radius:2px;border:solid 1px #9abcda;position:relative;top:2px;margin:1px 8px 1px 1px;cursor:pointer;}.login-main .login-bottom .tip .icon-check {margin:0 7px 0 0;width:14px;height:14px;border:none;background:url(../images/icon-login.png) no-repeat -111px -48px;}.login-main .login-bottom .center .item .icon {display:inline-block;width:33px;height:22px;}.login-main .login-bottom .center .item {width:288px;height:35px;border-bottom:1px solid #dae1e6;margin-bottom:35px;}.login-main {width:428px;position:relative;float:left;}.login-main .login-top {height:117px;background-color:#148be4;border-radius:12px 12px 0 0;font-family:SourceHanSansCN-Regular;font-size:30px;font-weight:400;font-stretch:normal;letter-spacing:0;color:#fff;line-height:117px;text-align:center;overflow:hidden;-webkit-transform:rotate(0);-moz-transform:rotate(0);-ms-transform:rotate(0);-o-transform:rotate(0);transform:rotate(0);}.login-main .login-top .bg1 {display:inline-block;width:74px;height:74px;background:#fff;opacity:.1;border-radius:0 74px 0 0;position:absolute;left:0;top:43px;}.login-main .login-top .bg2 {display:inline-block;width:94px;height:94px;background:#fff;opacity:.1;border-radius:50%;position:absolute;right:-16px;top:-16px;}.login-main .login-bottom {width:428px;background:#fff;border-radius:0 0 12px 12px;padding-bottom:53px;}.login-main .login-bottom .center {width:288px;margin:0 auto;padding-top:40px;padding-bottom:15px;position:relative;}.login-main .login-bottom .tip {clear:both;height:16px;line-height:16px;width:288px;margin:0 auto;}body {background:url(../images/loginbg.png) 0% 0% / cover no-repeat;position:static;font-size:12px;}input::-webkit-input-placeholder {color:#a6aebf;}input::-moz-placeholder {/* Mozilla Firefox 19+ */ color:#a6aebf;}input:-moz-placeholder {/* Mozilla Firefox 4 to 18 */ color:#a6aebf;}input:-ms-input-placeholder {/* Internet Explorer 10-11 */ color:#a6aebf;}input:-webkit-autofill {/* 取消Chrome記住密碼的背景顏色 */ -webkit-box-shadow:0 0 0 1000px white inset !important;}html {height:100%;}.login-main .login-bottom .tip {clear:both;height:16px;line-height:16px;width:288px;margin:0 auto;}.login-main .login-bottom .tip .login-tip {font-family:MicrosoftYaHei;font-size:12px;font-weight:400;font-stretch:normal;letter-spacing:0;color:#9abcda;cursor:pointer;}.login-main .login-bottom .tip .forget-password {font-stretch:normal;letter-spacing:0;color:#1391ff;text-decoration:none;position:absolute;right:62px;}.login-main .login-bottom .login-btn {width:288px;height:40px;background-color:#1E9FFF;border-radius:16px;margin:24px auto 0;text-align:center;line-height:40px;color:#fff;font-size:14px;letter-spacing:0;cursor:pointer;border:none;}.login-main .login-bottom .center .item .validateImg {position:absolute;right:1px;cursor:pointer;height:36px;border:1px solid #e6e6e6;}.footer {left:0;bottom:0;color:#fff;width:100%;position:absolute;text-align:center;line-height:30px;padding-bottom:10px;text-shadow:#000 0.1em 0.1em 0.1em;font-size:14px;}.padding-5 {padding:5px !important;}.footer a,.footer span {color:#fff;}@media screen and (max-width:428px) {.login-main {width:360px !important;} .login-main .login-top {width:360px !important;} .login-main .login-bottom {width:360px !important;}} </style></head><body><div class='main-body'> <div class='login-main'><div class='login-top'> <span>LayuiMini后臺登錄</span> <span class='bg1'></span> <span class='bg2'></span></div><form action='/login' method='post'> <div class='center'><div class='item'> <span class='icon icon-2'></span> <input type='text' name='uname' lay-verify='required' placeholder='請輸入登錄賬號' maxlength='24'/></div><div class='item'> <span class='icon icon-3'></span> <input type='password' name='pwd' lay-verify='required' placeholder='請輸入密碼' maxlength='20'> <span class='bind-password icon icon-4'></span></div> </div> <div class='tip'><span class='icon-nocheck'></span><span class='login-tip'>保持登錄</span><a href='javascript:' class='forget-password'>忘記密碼?</a> </div> <div style='text-align:center; width:100%;height:100%;margin:0px;'><button type='submit' lay-submit='' lay-filter='login'>立即登錄</button> </div></form> </div></div><div class='footer'> ©版權(quán)所有 2014-2018 叁貳柒工作室<span class='padding-5'>|</span><a target='_blank' >粵ICP備16006642號-2</a></div><script src='http://www.gepszalag.com/lib/layui-v2.6.3/layui.js' charset='utf-8'></script><script> //原本想用json的post發(fā)送,結(jié)果發(fā)現(xiàn)后端數(shù)據(jù)得自己解析,為了降低難度,直接用form表單的post提交,這樣后端直接拿數(shù)據(jù)即可(不然還得解析Json數(shù)據(jù)) // layui.use([’form’,’jquery’], function () { // var $ = layui.jquery, // form = layui.form, // layer = layui.layer; // // // 登錄過期的時候,跳出ifram框架 // if (top.location != self.location) top.location = self.location; // // $(’.bind-password’).on(’click’, function () { // if ($(this).hasClass(’icon-5’)) { // $(this).removeClass(’icon-5’); // $('input[name=’pwd’]').attr(’type’, ’password’); // } else { // $(this).addClass(’icon-5’); // $('input[name=’pwd’]').attr(’type’, ’text’); // } // }); // // $(’.icon-nocheck’).on(’click’, function () { // if ($(this).hasClass(’icon-check’)) { // $(this).removeClass(’icon-check’); // } else { // $(this).addClass(’icon-check’); // } // }); // // // 進行登錄操作 // form.on(’submit(login)’, function (data) { // data = data.field; // if (data.uname == ’’) { // layer.msg(’用戶名不能為空’); // return false; // } // if (data.pwd == ’’) { // layer.msg(’密碼不能為空’); // return false; // } // $.ajax({ // url:’/login’, // method:’post’, // data:data, // dataType:’JSON’, // success:function(res){ // if (res.msg===’登錄成功’){ // layer.msg(’登錄成功’, function () { // window.location = ’../index.html’; // }); // }else { // layer.msg('登錄失敗'); // } // }, // error:function (data) { // } // }) ; // // // return false; // }); // });</script></body></html>

當然以上代碼有一部分注釋掉了,原因是如果用JSON格式發(fā)送post請求,后端的servlet(準確的說是Tomcat的解析)并沒有幫我們解析封裝這部分數(shù)據(jù),所以我們無法直接get到,得自己另外解析數(shù)據(jù),當然也有一些第三方的工具包可以幫我們做這些事情(如阿里的fastjson等),這里為了使其更加簡單,所以采用表單提交post請求的方式,這樣解析的工作就不用我們做了。

效果是這樣的:

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

如果你沒學過layui或者對前端不太行,你也可以這樣

<!DOCTYPE html><htmllang='en'><head> <meta charset='UTF-8'> <title>用戶登錄</title></head><body><form action='/login' method='post'> 用戶名:<input type='text' name='uname'> 密碼:<input type='password' name='pwd'> <input type='submit' value='login'></form> </body></html>

一樣的功能,不過看上去的效果就不怎么好了。

③編寫servlet程序

當有了前端的頁面,看上去好了很多,但實質(zhì)校驗的程序我們還沒有寫。

想象一下我們就是后端程序,當前端的數(shù)據(jù)歷經(jīng)艱險,從錯綜復雜的網(wǎng)絡中到達我們的服務器,然后經(jīng)過系統(tǒng)分發(fā)到相應端口,這時恰在此端口的tomcat程序接受到了HTTP請求并對其封裝,經(jīng)過一系列騷操作后分發(fā)到了我們手中,而我們要做的就是拿著這個封裝好的請求進行校驗操作,然后對返回對象進行相應修改。

而這也是servlet類所需要做的(如果你想更好的理解servlet,可以看看bravo1988的回答),

package com.dreamchaser.loginTest;import com.dreamchaser.loginTest.mapper.UserMapper;import javax.servlet.ServletException;import javax.servlet.ServletOutputStream;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import java.io.IOException;public class LoginServlet extends HttpServlet { static UserMapper userMapper=UserMapper.getUserMapper(); @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request,response); } @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {String uname=req.getParameter('uname');String pwd=req.getParameter('pwd');ServletOutputStream outputStream = resp.getOutputStream();String result;if (pwd.equals(userMapper.getPwdByName(uname))){ //響應 result='登錄成功';}else { result='登錄失敗';}outputStream.write(result.getBytes()); }}

你可能會疑惑這個UserMapper是什么,別急,后面會介紹。

④封裝jdbc操作,編寫簡單的數(shù)據(jù)庫連接池

在操作數(shù)據(jù)庫之前,最好寫個簡單的數(shù)據(jù)庫連接池。一個是簡化我們的操作,一個是節(jié)省開銷,提高性能(Connection是個非常耗費資源的對象,頻繁的創(chuàng)建和回收將會是一筆巨大的開銷)

package com.dreamchaser.loginTest.utils;import java.sql.Connection;import java.sql.Driver;import java.sql.DriverManager;import java.sql.SQLException;import java.util.HashMap;import java.util.Map;/** * 一個簡單的數(shù)據(jù)庫連接池 */public class Pool { private static Driver driver; static {try { driver = new com.mysql.cj.jdbc.Driver(); DriverManager.registerDriver(driver);} catch (SQLException throwables) { throwables.printStackTrace();} } private static Map<Connection,Integer> pool=new HashMap<>(); private static String url='jdbc:mysql://localhost:3306/depository?serverTimezone=Asia/Shanghai'; private static String user='root'; private static String password='jinhaolin'; /** * 從連接池中獲取一個空閑連接,如果沒有則創(chuàng)建一個新的連接返回 * synchronized確保并發(fā)請求時,數(shù)據(jù)庫連接的正確性 * @return */ public synchronized static Connection getConnection(){for (Map.Entry entry:pool.entrySet()){ if (entry.getValue().equals(1)) {entry.setValue(0);return (Connection) entry.getKey(); }}Connection connection=null;try { connection=DriverManager.getConnection(url,user,password); pool.put(connection,0);} catch (SQLException throwables) { throwables.printStackTrace();}return connection; } /** * 釋放connection連接對象 * @param connection */ public synchronized static void releaseConnection(Connection connection){pool.put(connection,1); }}

當然上述實現(xiàn)非常簡陋,并發(fā)性能也不是很好,高并發(fā)時可能還會發(fā)生OOM,不過湊活著用吧(笑哭)。

⑤操作數(shù)據(jù)庫

package com.dreamchaser.loginTest.mapper;import com.dreamchaser.loginTest.utils.Pool;import java.sql.*;/** * 查詢用戶的Mapper */public class UserMapper { static UserMapper userMapper=new UserMapper(); //單例 public static UserMapper getUserMapper(){return userMapper; } private UserMapper(){ } //默認數(shù)據(jù)庫中用戶名唯一 public String getPwdByName(String name){Connection connection= Pool.getConnection();try { PreparedStatement statement=connection.prepareStatement('select pwd from `user` where uname=?'); statement.setString(1,name); ResultSet rs=statement.executeQuery(); //resultSet初始下標無法訪問,要調(diào)用next方法后移一位 rs.next(); return rs.getString(1);} catch (SQLException throwables) { throwables.printStackTrace();}return null; }}

這里采用單例的設計模式,保證UserMapper對象只有一個。(非常簡陋,實現(xiàn)也不優(yōu)雅,看著自己的代碼,突然感覺框架好方便啊(笑哭))

這里的作用就是根據(jù)用戶名查詢密碼。

⑥配置web.xml

雖然寫了servlet,但是tomcat并不知道你這個servlet的類在哪啊,所以必須讓tomcat知道,配置web.xml的目的就是通知tomcat在哪(更準確的說是servlet容器)的一種方式(當然也可以用注解)。配置如下:

<!DOCTYPE web-app PUBLIC '-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN' 'http://java.sun.com/dtd/web-app_2_3.dtd' ><web-app> <display-name>Archetype Created Web Application</display-name> <servlet> <servlet-name>LoginServlet</servlet-name> <servlet-class>com.dreamchaser.loginTest.LoginServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>LoginServlet</servlet-name> <url-pattern>/login</url-pattern> </servlet-mapping></web-app>

servlet-class里寫你這個Servlet類的路徑即可。

⑦idea運行配置

idea配置還是比較方便的。

點擊編輯配置,

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

點擊+(添加按鈕),選擇tomcat服務器(選哪個都可以,我選了tomcat本地)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

然后選擇相應的服務器程序,配置項目訪問的端口,就是tomcat在哪個端口運行(注意不要占用已有端口,默認8080,我這里是因為8080被占了,所以用了9090)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

這里還得配置一下工件,因為項目要運行一般有兩種方式:

一種是打成war包放在tomcat的webapps目錄下 一種是打成jar包直接運行(SpringBoot就是用這種方式,因為它內(nèi)置tomcat)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

這里工件的作用就是打成war包,至于每次運行部署?idea都會幫你搞定!是不是很方便?

這里那個應用程序上下文的作用就是給訪問路徑加個前綴

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

一般直接寫成'/'就行了,這樣我們要訪問login.html,只需訪問http://localhost:9090/login.html就行了,是不是很方便?

⑧運行程序

點擊運行

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

訪問localhost:9090/login.html(我因為是在login.html外面放了一個pages包,所以路徑是http://localhost:9090/pages/login.html)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

訪問成功,試試賬號密碼

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

我現(xiàn)在數(shù)據(jù)庫里只有root這一條數(shù)據(jù),試試效果

輸入錯誤的密碼

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

輸入正確的密碼

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

到這里,我們松了一口氣,終于完成了簡單的登錄功能。

三、回顧

別急,我們雖然實現(xiàn)了登錄這個功能,但是這個實現(xiàn)是在太簡陋了,各方各面都沒考慮,返回頁面也只登錄成功,登錄失敗的提示。我們回顧一下,仔細想想有哪些問題。

1.密碼未加密裸奔

我們在做上面的登錄時查詢時,密碼是查詢出來直接比對的,也就是說數(shù)據(jù)庫的密碼是明文存儲,而注冊登錄請求中密碼都是明文傳輸,這樣做的安全性極低,當黑客破解進入了你的數(shù)據(jù)庫時,你的數(shù)據(jù)庫的賬戶信息都在“裸奔”,比如前些年的csdn密碼泄露事件

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

如果我們存儲在數(shù)據(jù)庫的用戶密碼是加密過的,那么就算黑客進入了你的數(shù)據(jù)庫,損失也不會像明文存儲那樣大。

2.登錄信息未存儲

對于這個登錄操作,登錄成功后并未做其他處理,也就是說每次訪問都要登錄(如果對請求進行了攔截),或者這個登錄操作就是擺設,用戶訪問其他資源依舊暢通無阻。

3.對于其他資源并未進行權(quán)限管理

對于其他資源,如果不進行權(quán)限管理,那么登錄認證便失去了意義,不如做成一個靜態(tài)網(wǎng)頁來的省事。

四、優(yōu)化設計

針對上述缺點,我們可以進行以下改進:

1.密碼加密存儲

針對密碼未加密裸奔的問題,我們可以選擇在注冊的時候?qū)γ艽a進行加密,然后存儲;對于登錄功能,我們對前端傳過來的密碼進行加密,再根據(jù)這個密碼去數(shù)據(jù)庫中取數(shù)據(jù),這樣我們就實現(xiàn)了對密碼的加密存儲

2.存儲登錄信息

對于登錄操作,我們必須記錄下此次登錄狀態(tài),并在該用戶繼續(xù)訪問其他資源時予以放行,避免用戶多次進行登錄操作

3.對資源進行管理

對于系統(tǒng)資源我們必須進行管理,在用戶沒有相應權(quán)限時拒絕用戶的訪問請求,這個可以用過濾器或者SpringBoot的攔截器實現(xiàn)。

五、關(guān)于鑒權(quán)問題

在正式講思路之前,我還是想聊聊鑒權(quán)問題。

1.Cookie/Session機制

關(guān)于這個問題我不得不說說cookie/session機制(想了解的具體可以看這篇cookie和session的詳解與區(qū)別)。

總的來說,就是瀏覽器中有個叫做cookie的東西(其實就是個文件),它可以用來存儲一些信息,每次發(fā)送請求時,瀏覽器會自動把cookie字段信息加在請求頭里發(fā)送出去

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

這有什么用呢?

學過計算機網(wǎng)絡的人應該都清楚我們的http請求是無法保存狀態(tài)的,通俗點來講就是這次的請求無法知道上次的請求是什么,而這也對一些場景帶來的一些不便,就比如說登錄,我們就需要保存上次登錄的信息。

可http請求無法保存狀態(tài),所以我們必須把一些信息寫入到下次的請求里,保證服務器知道之前的關(guān)鍵信息,以便對之后的請求做出特定的操作。

而cookie便是解決這個問題而出現(xiàn)的,當我們需要存儲一些信息(狀態(tài)),就可以把信息存入cookie,瀏覽器每次發(fā)送請求時都會把cookie放在請求頭中(但這個要注意跨域問題,cookie在遇到跨域訪問時會失效,不過這個無關(guān)此次主題,就不細講了,感興趣的自行百度吧)。

總而言之,cookie就是存儲在瀏覽器(客戶端)的數(shù)據(jù)(文件),每次訪問時會帶上對應的cookie。

而session是什么呢?session和cookie類似,也是用來存放信息的,不過它是放在服務器上的。不過呢,session的本質(zhì)是存在于服務器內(nèi)存中的對象,閱讀源碼我們可以發(fā)現(xiàn)其對應的就是一個ConcurrentMap(線程安全的map容器)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

每一個客戶端會對應服務端一個session對象,而如何得到的關(guān)鍵就在于cookie中的JSESSIONID(tomcat默認是這個名字,名稱可以變,但用法是一樣的),其值便對應這map容器的鍵,而map的值便是session對象。這樣每次用戶發(fā)送請求來時,服務器就能準確的找到對應的session對象了。

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

2.用Cookie/Session解決鑒權(quán)問題?

明白了Cookie/Session的機制以后,我們不難設計出一套簡單的登錄方案——登錄成功后在對應的session對象中存放User信息并設置失效時間,每次訪問資源都看看session中有沒有對應user對象,如果有就說明之前登錄過了,直接通過即可,否則說明未登錄,此時可以跳轉(zhuǎn)至登錄頁面讓用戶進行登錄。

這一切看似都很完美,從某種角度上來說確實如此,但它沒有缺點嗎?

Cookie/Session機制的缺點

1.無法解決跨域問題

在跨域訪問時,cookie會失效,這是為了防止csrf攻擊(跨站請求偽造),但對于開發(fā)者來說造成了一定的困擾,因為現(xiàn)實中的服務器不可能只有一臺,大概率是集群分布,雖然可以用反向代理避免跨域訪問,但終究是有局限之處的。

2.session機制依賴于cookie

從cookie/session機制中我們不難看出,session的實現(xiàn)依賴于前端的cookie,因為其session的確定必須要前端請求中cookie,沒有了cookie,session是無法確定的。

而這會帶來什么問題呢?那就是對于多端訪問,如手機App端,其并沒有cookie的直接實現(xiàn)(可以實現(xiàn),其實也就是在請求頭中加入cookie字段,但使用此方式并不普遍,也挺麻煩的),如果cookie很難使用,那么session也無法使用。

3.可拓展性不強

如果將來搭建了多個服務器,雖然每個服務器都執(zhí)行的是同樣的業(yè)務邏輯,但是session數(shù)據(jù)是保存在內(nèi)存中的(不是共享的),用戶第一次訪問的是服務器1,當用戶再次請求時可能訪問的是另外一臺服務器2,服務器2獲取不到session信息,就判定用戶沒有登陸過。

與此同時,當你使用session的時候你會發(fā)現(xiàn)一個很尷尬的事情——你無法直接獲取到存放session的map(除非你用反射),這樣就導致你的操作受限,比如你想以某個身份強制下線某個用戶時,session將會變得力不從心。

4.服務器壓力增大

session存在于服務器內(nèi)存中,如果session很多,那么服務器壓力便會很大。會頻繁觸發(fā)gc操作,導致服務器響應變慢,吞吐量下降。

5.安全性問題

Cookie/Session機制并不是絕對安全,你必須小心應對,當然我接下來說的token方式同樣也有這樣那樣的問題,但是我們要明白一件事情——沒有絕對安全的系統(tǒng)!

當前的所謂安全措施不過是在增加黑客入侵系統(tǒng)的成本,但你要注意的是你在增加黑客入侵的難度和成本的同時,也同樣在增加自己系統(tǒng)的維護成本,它必然是以一定的性能作為代價的

所以如何權(quán)衡安全和性能,這是永遠是一件值得我們深思的事情

3.使用token機制解決鑒權(quán)問題

什么是token呢?

事實上它只是我們自己實現(xiàn)的一套類似cookie/Session的機制。

至于為啥叫token?

你也可以叫它cat,dog之類的,只要你喜歡,隨便你怎么取名字(笑哭)。

好了,開個玩笑,咱們回到正題,在我看來,token只是脫胎于cookie/session的一套機制,它的實現(xiàn)原理幾乎是和cookie/session一模一樣的(9成像,當然也有很多根據(jù)自己業(yè)務的變種)。

如果說cookie/session機制可以描述為下圖:

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

那么token機制可以描述為以下形式:

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

怎么樣?是不是很像?其實它們核心原理是一樣的。

那token機制相較于cookie/session機制有啥好處呢?

1.可以直接操作token令牌池 2.對于手機App端友好 3.跨域問題可以間接解決 4.對于服務器集群,token令牌池可以放在redis數(shù)據(jù)庫中(當然也可以是其他方案),這樣可以實現(xiàn)用戶登錄狀態(tài)多服務器共享

其實,總的來說,就只有一條(笑哭),那就是靈活!因為token機制是我們自己實現(xiàn)的(當然也可以借助框架),這樣操作這些東西的時候就不必拘泥于條條框框,可以根據(jù)自己的業(yè)務需求制定適合的鑒權(quán)方案。

悄悄告訴你一句:csdn也是用token的哦!(不過具體實現(xiàn)可能并不一樣)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

當然,相較于cookie/session機制而言,它也有個巨大的弊端——在網(wǎng)頁應用中,使用token機制會比使用cookie/session機制麻煩很多,所有都得“從頭再來”,不像cookie/session可以開箱即用。

六、用SpringBoot+SSM實現(xiàn)一套簡單的鑒權(quán)服務(注冊,登錄,權(quán)限控制)

這里我是用token來實現(xiàn)鑒權(quán)服務的。以下是我畫的大致流程圖(可能有點丑,有點亂)

手把手教你用Java實現(xiàn)一套簡單的鑒權(quán)服務

在展示代碼實現(xiàn)時,你可能會對某些類比較疑惑,以下是對這些類的說明:

RestResponse 這是我用來封裝響應格式的,Status用來封裝響應狀態(tài) CrudUtil 這是我用來封裝CRUD操作的工具類,該類主要為了簡化controller的響應操作

同時我會省略Service層和Dao層實現(xiàn)

1.注冊服務

①注冊頁面

<!DOCTYPE html><html lang='zh-CN' xmlns:th='http://www.thymeleaf.org'><head> <meta charset='utf-8'> <title>layui</title> <meta name='renderer' content='webkit'> <meta http-equiv='X-UA-Compatible' content='IE=edge,chrome=1'> <meta name='viewport' content='width=device-width, initial-scale=1, maximum-scale=1'> <link rel='stylesheet' href='http://www.gepszalag.com/bcjs/static/css/public.css'> <link rel='stylesheet' href='http://www.gepszalag.com/bcjs/static/lib/layui-v2.6.3/css/layui.css'> <style>body { background: url('static/images/loginbg.png') 0% 0% / cover no-repeat; position: static; font-size: 12px;} </style></head><body><div class='layui-container'> <div style='width: 500px;border-radius: 10px'><fieldset style='margin-top: 20%'> <legend style='font-size: 30px;padding-top: 20px;text-align: center'>用戶注冊</legend> <div class='layui-field-box'><div style='margin: 20px;margin-top: 30px'> <div class='layui-form-item'><label class='layui-form-label required'>用戶名</label><div class='layui-input-block'> <input type='text' name='uname' lay-verify='required' lay-reqtext='用戶名不能為空' placeholder='請輸入用戶名' value='' class='layui-input'> <tip>填寫自己真實姓名</tip></div> </div> <div class='layui-form-item'><label class='layui-form-label required'>性別</label><div class='layui-input-block'> <input type='radio' name='sex' value='男' checked=''> <input type='radio' name='sex' value='女'></div> </div> <div class='layui-form-item'><label class='layui-form-label required'>手機</label><div class='layui-input-block'> <input type='number' name='phone' lay-verify='phone' placeholder='請輸入手機號' value='' class='layui-input'></div> </div> <div class='layui-form-item'><label class='layui-form-label required'>郵箱</label><div class='layui-input-block'> <input type='email' name='email' lay-verify='email' placeholder='請輸入郵箱' value='' class='layui-input'></div> </div> <div class='layui-form-item'><label class='layui-form-label required'>密碼</label><div class='layui-input-block'> <input type='text' name='pwd' lay-verify='required' placeholder='請輸入密碼' value='' class='layui-input'></div> </div> <div class='layui-form-item'><label class='layui-form-label required'>入職時間</label><div class='layui-input-block'> <input type='text' name='entryDate' lay-verify='date' placeholder='請選擇入職時間' autocomplete='off' class='layui-input'></div> </div> <div class='layui-form-item'><label style='display: inline'>郵箱驗證碼</label><input type='text' name='code' placeholder='請輸入驗證碼' lay-verify='required' maxlength='5' style='width:160px;display: inline'><button lay-filter='saveBtn' style='display: inline;margin-left: 10px'>發(fā)送驗證碼</button> </div> <div style='margin-top: 20px'><div class='layui-input-block'> <button lay-submit lay-filter='registerBtn'>注冊 </button></div> </div></div> </div></fieldset> </div></div><script src='http://www.gepszalag.com/bcjs/static/lib/layui-v2.6.3/layui.js' charset='utf-8'></script><script> layui.use([’form’, ’layer’, ’laydate’,’element’], function () {var form = layui.form, layer = layui.layer, laydate = layui.laydate, element=layui.element, $ = layui.$;//日期laydate.render({ elem: ’#date’});//監(jiān)聽提交$(’#saveBtn’).bind(’click’, function () { var email = $(’#email’).val(); if (email===’’||email==null){layer.msg('請輸入正確的郵箱!'); }else {$.ajax({ url: '/sendCode', data:’{'email':’+JSON.stringify(email)+’}’, type: 'post', dataType: ’JSON’, contentType: 'application/json;charset=utf-8', success: function (data) {if (data.status !== 200) { layer.msg(data.statusInfo.message);//失敗的表情 return;} else { layer.msg('驗證碼發(fā)送成功,請前往郵箱查看', {icon: 6,//成功的表情time: 1000 //1秒關(guān)閉(如果不配置,默認是3秒) }, function () { });} }}); }});//監(jiān)聽提交form.on(’submit(registerBtn)’, function (data) { $.ajax({url: '/register',data: JSON.stringify(data.field),type: 'post',dataType: ’JSON’,contentType: 'application/json;charset=utf-8',success: function (data) { if (data.status !== 200) {layer.msg(data.statusInfo.message);//失敗的表情return; } else {layer.msg('注冊成功', { icon: 6,//成功的表情 time: 1000 //1秒關(guān)閉(如果不配置,默認是3秒)}, function () { window.location = ’/login’;}); }} }); return false;}); });</script></body></html>

②發(fā)送驗證碼

sendcode接口

/** * 驗證是否有此賬號,然后發(fā)送驗證碼 * @param map 主要認證主體,如賬號,郵箱,qq的openID,wechat的code等 * @return restResponse,附帶憑證token */ @PostMapping('/sendCode') public RestResponse sendCode(@RequestBody Map<String,Object> map){if (userService.findUserByCondition(map)==null){ String principal; if (map.get('phone')!=null){principal=String.valueOf(map.get('phone')); }else if (map.get('email')!=null){principal=String.valueOf(map.get('email')); }else {return CrudUtil.ID_MISS_RESPONSE; } //創(chuàng)建一個驗證碼 VerificationCode v=new VerificationCode(); //將驗證碼存入驗證碼等待池 VerificationCodePool.addCode(principal,v); //發(fā)送郵箱驗證碼 sendEmail(principal,v.getCode()); return new RestResponse();}return new RestResponse('',304,new StatusInfo('發(fā)送驗證碼失敗,該賬戶已存在!','發(fā)送驗證碼失敗,該賬戶已存在!')); }

郵件發(fā)送方法(調(diào)用SpringBoot提供的mail服務(需要導包))

/** * 發(fā)送帶有驗證碼的郵件信息 */ private void sendEmail(String email,String code){//發(fā)送驗證郵件try { SimpleMailMessage mailMessage = new SimpleMailMessage(); //主題 mailMessage.setSubject('倉庫管理系統(tǒng)的驗證碼郵件'); //內(nèi)容 mailMessage.setText('歡迎使用倉庫管理系統(tǒng),您正在注冊此賬戶。' + 'n您收到的驗證碼是: '+code+' ,請不要將此驗證碼透露給別人。'); //發(fā)送的郵箱地址 mailMessage.setTo(email); //默認發(fā)送郵箱郵箱 mailMessage.setFrom(fromEmail); //發(fā)送 mailSender.send(mailMessage);}catch (Exception e){ throw new MyException(e.toString());} }

驗證碼對象

package com.dreamchaser.depository_manage.security.bean;import lombok.Data;import java.time.Instant;import java.util.Random;/** * 驗證碼,默認有效期為五分鐘 * @author 金昊霖 */@Datapublic class VerificationCode { /** * 默認持續(xù)時間 */ private final long DEFAULT_TERM=60*5; /** * 驗證碼 */ private String code; /** * 創(chuàng)建時刻 */ private Instant instant; /** * 有效期 */ private long term; /** * 根據(jù)時間判斷是否有效 * @return boolean值 */ public boolean isValid(){return Instant.now().getEpochSecond()-instant.getEpochSecond()<=term; } public VerificationCode(Instant instant, long term) {//生成隨機驗證碼codegenerateCode();this.instant = instant;this.term = term; } public VerificationCode(Instant instant) {//生成隨機驗證碼codegenerateCode();this.instant = instant;this.term=DEFAULT_TERM; } public VerificationCode() {//生成隨機驗證碼codegenerateCode();this.instant=Instant.now();this.term=DEFAULT_TERM; } private void generateCode(){StringBuilder codeNum = new StringBuilder();int [] numbers = {0,1,2,3,4,5,6,7,8,9};Random random = new Random();for (int i = 0; i < 5; i++) { //目的是產(chǎn)生足夠隨機的數(shù),避免產(chǎn)生的數(shù)字重復率高的問題 int next = random.nextInt(10000); codeNum.append(numbers[next % 10]);}this.code= codeNum.toString(); }}

驗證碼池

package com.dreamchaser.depository_manage.security.pool;import com.dreamchaser.depository_manage.security.bean.VerificationCode;import java.util.Map;import java.util.concurrent.ConcurrentHashMap;/** * 驗證碼等待池 * @author 金昊霖 */public class VerificationCodePool { private static Map<String, VerificationCode> pool=new ConcurrentHashMap<>(10); /** * 增加一條驗證碼 * @param principal 主要內(nèi)容,如郵箱,電話號碼等 * @param verificationCode 驗證碼 */ public static void addCode(String principal,VerificationCode verificationCode){pool.put(principal, verificationCode); } /** * 根據(jù)principal主要信息獲取未過期的驗證碼,如果沒有未過期的令牌則返回null * @param principal 主要內(nèi)容,如郵箱,電話號碼等 * @return verificationCode 未過期的驗證碼或者null */ public static VerificationCode getCode(String principal){VerificationCode verificationCode=pool.get(principal);//如果沒有相應驗證碼則直接返回nullif (verificationCode==null){ return null;}//判斷令牌是否過期if (verificationCode.isValid()){ //將驗證碼取出 pool.remove(principal); return verificationCode;}else{ //清除過期驗證碼 pool.remove(principal); return null;} } /** * 根據(jù)主要信息principal刪除對應的驗證碼 * @param principal 主要信息 */ public static void removeCode(String principal){pool.remove(principal); }}

③注冊用戶

MD5加密類

/* * Copyright (c) JForum Team All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1) Redistributions of * source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2) Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the * following disclaimer in the documentation and/or other materials provided with the distribution. 3) Neither the name of 'Rafael Steil' nor the names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE This file creation date: Mar 29, 2003 / * 1:15:50 AM The JForum Project http://www.jforum.net */package com.dreamchaser.depository_manage.utils;import java.security.MessageDigest;import java.security.NoSuchAlgorithmException;/** * MD5加密 */public class Md5 {/** * Encodes a string * @param str String to encode * @return Encoded String */public static String crypt(String str) {if (str == null || str.length() == 0) {throw new IllegalArgumentException('String to encript cannot be null or zero length');}StringBuilder hexString = new StringBuilder();try {MessageDigest md = MessageDigest.getInstance('MD5');md.update(str.getBytes());byte[] hash = md.digest();for (byte b : hash) {if ((0xff & b) < 0x10) {hexString.append('0').append(Integer.toHexString((0xFF & b)));} else {hexString.append(Integer.toHexString(0xFF & b));}}} catch (NoSuchAlgorithmException e) {e.printStackTrace();}return hexString.toString();}}

注冊用戶接口

/** * 注冊用戶(通常為手機或者郵箱注冊) * @param map 參數(shù)列表,包括賬號(手機注冊就是phone,郵箱就是email)、密碼 * @return 成功則返回憑證,否則返回驗證失敗 */ @PostMapping('/register') public RestResponse register(@RequestBody Map<String,Object>map){String principal;Object password=map.get('pwd');Object code=map.get('code');UserToken userToken;//判斷必要參數(shù)是否滿足if (password==null||code==null){ return CrudUtil.ID_MISS_RESPONSE;}//從map中獲取對應參數(shù)if (map.get('email')!=null){ principal=String.valueOf(map.get('email')); userToken=new UserToken(LoginType.EMAIl_PASSWORD,principal,String.valueOf(password));}else { return CrudUtil.ID_MISS_RESPONSE;}//驗證碼正確且成功插入數(shù)據(jù)if (checkCode(principal,String.valueOf(code))){ //對密碼進行加密然后存儲用戶信息 map.put('pwd',Md5.crypt(String.valueOf(map.get('pwd')))); //如果用戶記錄插入成功 if (userService.insertUser(map)==1){String token= Md5.crypt(userToken.getPrincipal()+userToken.getInstant());//返回憑證return new RestResponse().setData(token); }}else { //驗證碼錯誤 return CrudUtil.CODE_ERROR;}return

這里的LoginType是登錄方式,這個之后會提到

檢驗驗證碼方法

/** * 用于注冊用戶的方法,主要為號碼驗證和郵箱驗證提供驗證碼核對的服務 * @param principal 認證主體 * @param code 驗證碼 * @return 是否驗證通過 */ private boolean checkCode(String principal,String code){if (code!=null){ VerificationCode verificationCode=VerificationCodePool.getCode(principal); if (verificationCode!=null){return code.equals(verificationCode.getCode()); }}return false; }2.登錄服務

登錄界面

這里為了方便起見,我把token存儲在cookie中

<!DOCTYPE html><html lang='zh-CN' xmlns:th='http://www.thymeleaf.org'><head> <meta charset='UTF-8'> <title>后臺管理-登陸</title> <meta http-equiv='X-UA-Compatible' content='IE=edge,chrome=1'> <meta http-equiv='Access-Control-Allow-Origin' content='*'> <meta name='viewport' content='width=device-width, initial-scale=1, maximum-scale=1'> <meta name='apple-mobile-web-app-status-bar-style' content='black'> <meta name='apple-mobile-web-app-capable' content='yes'> <meta name='format-detection' content='telephone=no'> <link rel='stylesheet' href='http://www.gepszalag.com/bcjs/static/lib/layui-v2.6.3/css/layui.css' media='all'> <!--[if lt IE 9]> <script src='https://cdn.staticfile.org/html5shiv/r29/html5.min.js'></script> <script src='https://cdn.staticfile.org/respond.js/1.4.2/respond.min.js'></script> <![endif]--> <style>html, body {width: 100%;height: 100%;overflow: hidden}body {background: #1E9FFF;}body:after {content:’’;background-repeat:no-repeat;background-size:cover;-webkit-filter:blur(3px);-moz-filter:blur(3px);-o-filter:blur(3px);-ms-filter:blur(3px);filter:blur(3px);position:absolute;top:0;left:0;right:0;bottom:0;z-index:-1;}.layui-container {width: 100%;height: 100%;overflow: hidden}.admin-login-background {width:360px;height:300px;position:absolute;left:50%;top:40%;margin-left:-180px;margin-top:-100px;}.logo-title {text-align:center;letter-spacing:2px;padding:14px 0;}.logo-title h1 {color:#1E9FFF;font-size:25px;font-weight:bold;}.login-form {background-color:#fff;border:1px solid #fff;border-radius:3px;padding:14px 20px;box-shadow:0 0 8px #eeeeee;}.login-form .layui-form-item {position:relative;}.login-form .layui-form-item label {position:absolute;left:1px;top:1px;width:38px;line-height:36px;text-align:center;color:#d2d2d2;}.login-form .layui-form-item input {padding-left:36px;}.captcha {width:60%;display:inline-block;}.captcha-img {display:inline-block;width:34%;float:right;}.captcha-img img {height:34px;border:1px solid #e6e6e6;height:36px;width:100%;} </style></head><body><div class='layui-container'> <div class='admin-login-background'><div class='layui-form login-form'> <form action=''><div class='layui-form-item logo-title'> <h1>倉庫信息管理系統(tǒng)登錄</h1></div><div class='layui-form-item'> <label ></label> <input type='text' name='principal' lay-verify='required|account' placeholder='請輸入郵箱' autocomplete='off' ></div><div class='layui-form-item'> <label ></label> <input type='password' name='credentials' lay-verify='required|password' placeholder='密碼' autocomplete='off' class='layui-input'></div><!-- 徒有其表的驗證碼,主要是不想另外弄了 --><div class='layui-form-item'> <label ></label> <input type='text' name='captcha' lay-verify='required|captcha' placeholder='圖形驗證碼' autocomplete='off' class='layui-input verification captcha'> <div class='captcha-img'><img src='http://www.gepszalag.com/bcjs/static/images/captcha.jpg'> </div></div><div class='layui-form-item'> <input type='checkbox' name='rememberMe' value='true' lay-skin='primary'></div><div class='layui-form-item'> <button lay-submit='' lay-filter='login'>登 入</button></div> </form></div> </div></div><script src='http://www.gepszalag.com/bcjs/static/lib/jquery-3.4.1/jquery-3.4.1.min.js' charset='utf-8'></script><script src='http://www.gepszalag.com/bcjs/static/lib/layui-v2.6.3/layui.js' charset='utf-8'></script><script src='http://www.gepszalag.com/bcjs/static/lib/jq-module/jquery.particleground.min.js' charset='utf-8'></script><script src='http://www.gepszalag.com/bcjs/static/js/cookie.js' charset='utf-8'></script><script> layui.use([’layer’,’form’], function () {var form = layui.form, layer = layui.layer;// 登錄過期的時候,跳出ifram框架if (top.location != self.location) top.location = self.location;// 粒子線條背景$(document).ready(function(){ $(’.layui-container’).particleground({dotColor:’#7ec7fd’,lineColor:’#7ec7fd’ });});// 進行登錄操作form.on(’submit(login)’, function (data) { data = data.field; if (data.principal === ’’) {layer.msg(’用戶名不能為空’);return false; } if (data.credentials === ’’) {layer.msg(’密碼不能為空’);return false; } if (data.captcha === ’’) {layer.msg(’驗證碼不能為空’);return false; } data.loginType='email'; $.ajax({url:'/login',type:’post’,dataType:’json’,contentType: 'application/json;charset=utf-8',data:JSON.stringify(data),beforeSend:function () { this.layerIndex = layer.load(0, { shade: [0.5, ’#393D49’] });},success:function(data){ if(data.status !== 200){layer.msg(data.statusInfo.message);//失敗的表情return; }else{layer.msg('登錄成功', { icon: 6,//成功的表情 time: 1000 //1秒關(guān)閉(如果不配置,默認是3秒)}, function(){ cookieUtil.createCookie('token',data.data) window.location = ’/index’;}); }},complete: function () { layer.close(this.layerIndex);} }) return false;}); });</script></body></html>

當然這里封裝了cookie的操作

var cookieUtil={ createCookie:function (name,value,days){var expires='';if (days){ var date=new Date(); date.setTime(date.getTime()+(days*14*24*3600*1000)); expires=';expires='+date.toGMTString();}document.cookie=name+'='+value+expires+';path=/'; }, /*設置cookie*/ set:function(name,value,expires,path,domain,secure){var cookie=encodeURIComponent(name)+'='+encodeURIComponent(value);if(expires instanceof Date){ cookie+='; expires='+expires.toGMTString();}else{ var date=new Date(); date.setTime(date.getTime()+expires*24*3600*1000); cookie+='; expires='+date.toGMTString();}if(path){ cookie+='; path='+path;}if(domain){ cookie+='; domain='+domain;}if (secure) { cookie+='; '+secure;}document.cookie=cookie; }, /*獲取cookie*/ get:function(name){var cookieName=encodeURIComponent(name);/*正則表達式獲取cookie*/var restr='(^| )'+cookieName+'=([^;]*)(;|$)';var reg=new RegExp(restr);var cookieValue=document.cookie.match(reg)[2];/*字符串截取cookie*//*var cookieStart=document.cookie.indexOf(cookieName+“=”);var cookieValue=null;if(cookieStart>-1){ var cookieEnd=document.cookie.indexOf(';',cookieStart); if(cookieEnd==-1){cookieEnd=document.cookie.length; } cookieValue=decodeURIComponent(document.cookie.substring(cookieStart +cookieName.length,cookieEnd));}*/return cookieValue; }}

登錄接口

這里的token憑證是根據(jù)用戶密碼+當前時刻(鹽)加密得到的

/** * 登錄接口 * @param map 登錄信息 * loginType 登錄方式,目前支持的有email,qq,wechat * principal 主要認證主體,如賬號,郵箱,qq的openID,wechat的code等 * credentials 類似于密碼,如果是qq,wechat則不需要傳改參數(shù) * restResponse,附帶憑證token */ @PostMapping('/login') public RestResponse login(@RequestBody Map<String,String> map) {UserToken userToken=new UserToken(LoginType.getType(map.get('loginType')),map.get('principal'),map.get('credentials'));return login(userToken); }

認證方法

/** * 將生成的令牌拿去認證,如果認證成功則返回帶有token憑證響應,否則返回用戶密碼錯誤的響應 * @param userToken 未認證的令牌 * @return restResponse 如果認證成功則返回帶有token憑證響應,否則返回用戶密碼錯誤的響應 */ private RestResponse login(UserToken userToken) {String token=loginRealms.authenticate(userToken);if (token!=null){ return new RestResponse(token);}else { return CrudUtil.NOT_EXIST_USER_OR_ERROR_PWD_RESPONSE;} }

登錄方式enum類

這里可以看到我里面有多種方式登錄,不過我的代碼里只實現(xiàn)了郵箱登錄,其余方式可以自己去實現(xiàn)拓展

package com.dreamchaser.depository_manage.security.bean;/** * 登錄方式枚舉類 * @author 金昊霖 */public enum LoginType { /** * 通用 */ COMMON('common_realm'), /** * 用戶密碼登錄 */ EMAIl_PASSWORD('user_password_realm'), /** * 手機驗證碼登錄 */ USER_PHONE('user_phone_realm'), /** * 第三方登錄(微信登錄) */ WECHAT_LOGIN('wechat_login_realm'), /** * 第三方登錄(qq登錄) */ QQ_LOGIN('qq_login_realm'); private String type; LoginType(String type) {this.type = type; } public String getType() {return type; } /** * 根據(jù)簡單的字符串返回對應的LoginType * @param s 簡單的字符串 * @return 對應的LoginType */ public static LoginType getType(String s){switch (s) { case 'email':return EMAIl_PASSWORD; case 'qq':return QQ_LOGIN; case 'wechat':return WECHAT_LOGIN; case 'phone':return USER_PHONE; default:return null;} } @Override public String toString() {return this.type; }}

登錄方式類

這里面可以根據(jù)自己的業(yè)務拓展,我只實現(xiàn)了郵箱登錄

package com.dreamchaser.depository_manage.security.bean;import com.dreamchaser.depository_manage.entity.User;import com.dreamchaser.depository_manage.exception.MyExcept

標簽: Java
相關(guān)文章:
主站蜘蛛池模板: 中国黄色在线视频 | 亚洲三区在线观看 | 久草在线免费福利资源 | 色综合久久网 | 久久精品国产亚洲一区二区三区 | 91免费版在线观看 | 久久精品欧美 | 欧美第一色 | 7878www免费看片 | 成人深夜免费视频 | 久久蜜桃av一区二区天堂 | 亚洲精品久久久久久一区二区 | 99精品久久久久久久免费看蜜月 | 成人免费毛片嘿嘿连载视频 | 亚洲毛片| 成人高清 | 亚洲视频综合 | 亚洲电影中文字幕 | 亚洲欧美日韩国产综合精品二区 | 国产精品免费久久 | 免费看特级毛片 | 欧美在线视频一区二区 | 99re国产| 国产精品久久久久久久午夜片 | 在线观看国精产品二区1819 | 小泽玛丽娅 | 国产传媒日韩欧美 | 国产91成人在在线播放 | 久久国产精品99久久久久久老狼 | 国产激情 | 国产精品一区一区 | 欧美精品一区二区三区蜜桃视频 | 国产99在线播放 | 欧美日韩在线观看中文字幕 | 色免费视频 | 欧美不卡视频一区发布 | 999精品在线| 午夜精品久久久久久久白皮肤 | 久久久国产视频 | 欧洲一区 | 国产精品欧美久久久久一区二区 | 超级乱淫片国语对白免费视频 | 欧美黑人一级爽快片淫片高清 | 久久2 | 玖玖国产精品视频 | 国产精品久久久久久久9999 | 中国免费看的片 | 91精品国产91久久久久游泳池 | 亚洲啊v| 亚洲激情网站 | 国产伦精品一区二区三区在线 | 成人在线免费 | 黄篇网址| 亚洲精品国产电影 | 嫩草视频在线播放 | 欧美综合区 | 亚洲视频在线观看 | 97成人精品视频在线观看 | 午夜午夜精品一区二区三区文 | 女人毛片 | 国产性一级片 | 久久99精品久久久久久青青日本 | 国产免费观看一区二区三区 | 精品一区二区三区在线视频 | 亚洲免费在线视频 | 一级毛片在线播放 | 亚洲精品一 | 日韩欧美~中文字幕 | 国产精品视频久久久 | 久久最新网址 | 在线视频一区二区 | 中文天堂在线观看视频 | 国产精品永久免费自在线观看 | 91亚洲国产| 日韩中出| 性欧美大战久久久久久久免费观看 | 欧美手机在线 | 色一色视频 | 国产欧美日韩在线观看 | 国产视频精品在线 | 欧美日本亚洲 | 亚洲一区二区三区视频 | 久久精精品 | 一级黄色影片在线观看 | 色黄视频在线观看 | 黄色免费观看 | 成人午夜在线观看 | av超碰在线 | 久久国产精品一区二区 | 狠狠狠干 | 久久99久 | 四虎免费在线播放 | 久久久久久毛片免费播放 | 精品久久久久久亚洲综合网站 | 能免费看av的网站 | 日韩免费视频一区二区 | 国产成人免费视频网站高清观看视频 | 男人的天堂一级片 | 久久国产精彩视频 | 99精品电影| 国产黄色免费网站 | 久热精品视频在线播放 | 精品99视频 | 日韩精品免费 | 成人欧美一区二区三区黑人孕妇 | 一区二区在线视频免费观看 | 中文字幕在线观看网站 | 久久av一区二区 | 电影午夜精品一区二区三区 | 99精品久久久 | 久草中文在线 | 99久久日韩精品视频免费在线观看 | 久久久久久久久久一区二区 | 国产精品第一国产精品 | 精品入口麻豆88视频 | 精品国产一区二区三区久久久蜜月 | 中文字幕亚洲一区 | 亚洲视频自拍 | www.久久久 | 日日日日干干干干 | 成人精品视频在线 | 久久99精品国产.久久久久 | 午夜影院免费 | 色综合一区二区三区 | 欧美一区二区在线视频 | 98精品国产高清在线xxxx天堂 | 91亚洲国产成人精品性色 | 成人一级电影在线观看 | 国产婷婷| 免费中文字幕日韩欧美 | 欧美激情在线精品一区二区三区 | 看一级黄色大片 | 久久久久国产精品一区二区三区 | 日韩欧美~中文字幕 | 亚洲精品久久久久999中文字幕 | 国产精品二区一区 | 中文字幕在线观看 | 欧洲精品视频在线观看 | 久久青青| 国产成人在线网站 | 国产精品久久久久久久久久99 | 另类五月天 | 亚洲美女网站 | 伊人网站 | 久久中文字幕一区 | 日韩不卡一区二区三区 | 青青草精品 | 久久兔费看a级 | 成人影院网站ww555久久精品 | 成人网av | 高清国产视频 | 中文字幕一区二区三区乱码在线 | 超碰97免费在线 | 久久国产精品视频一区 | 国产在线一区二区 | 91av官网| 欧美黄视频在线观看 | 国产成人免费在线 | 97伦理电影院 | 成人在线免费视频观看 | 日韩网站免费观看 | 日韩精品在线一区 | 黄色一级毛片在线观看 | 一区二区精品在线 | 三级免费毛片 | 午夜精品视频 | 欧美午夜在线观看 | 中文字幕 亚洲一区 | 国产日韩欧美一区 | 全免一级毛片 | 亚洲一区二区三区四区五区中文 | 日韩av在线一区二区三区 | 99久久免费视频在线观看 | 久久精品久久久久 | 国产视频久久久久久久 | 亚洲电影在线观看 | 国产一区在线观看视频 | www.日韩在线观看 | 互换娇妻呻吟hd中文字幕 | 三区在线 | 91婷婷射 | 91综合网 | 一区国产精品 | 国产日韩欧美 | 久久99操 | 亚洲视频在线观看免费 | 亚洲大尺度视频 | 亚洲 欧美 日韩 精品 | 日韩激情综合网 | 国产视频中文字幕 | 欧美一级欧美三级在线观看 | 精久久 | 国产精品久久一区 | 欧美视频在线免费 | 日本三级做a全过程在线观看 | 一区二区精品在线 | 91久久夜色精品国产网站 | 成人欧美一区二区三区黑人孕妇 | 日本免费xxxx | 欧美色v| 中文字幕日韩欧美一区二区三区 | 成人不卡 | 久久久久久一区 | 黄色片免费看. | 精品一区二区三区久久 | 一区二区三区四区精品 | 日日综合| 亚洲第一黄色 | 亚洲一区二区三区在线播放 | 欧美日韩亚洲国产 | 欧美日韩一| 一区二区在线电影 | 在线观看a视频 | 国内精品一区二区 | 另类亚洲专区 | 人人干在线视频 | 一区二区三区视频在线观看 | av黄色在线观看 | 欧美一级在线 | 国产小视频在线播放 | 日韩成人三级 | 国产精品久久久久一区二区三区共 | 国产精品久久久久久久久污网站 | 国产高清在线观看 | 在线中文av | 亚洲欧美一区二区三区在线 | 国产三级精品三级 | 国产乱码精品一区二区三区忘忧草 | 亚洲精品一区二区另类图片 | 青青草精品 | 国产 日韩 欧美 中文 在线播放 | 久久久大 | 一区二区视频 | 亚州成人| 玖玖精品 | 一区二区三区亚洲 | 超级乱淫片国语对白免费视频 | 国产精品久久久久久中文字 | 久久99国产精品久久99大师 | 亚洲欧美日韩国产综合精品二区 | 国产一区二区三区久久久 | 欧洲尺码日本国产精品 | 影音先锋男人网 | 亚洲国产精品久久久 | 精品久久久久久 | 日日干日日操 | 欧美激情一区二区三区 | 久久精品网 | 7777奇米影视 | 天天躁人人躁人人躁狂躁 | 羞羞的视频在线观看 | av大片网| 黄色直接看 | 日韩精品在线免费观看视频 | 日韩婷婷 | 国产在线不卡 | 美女视频一区二区三区 | 午夜精品久久久久久久白皮肤 | 久久麻豆| 欧美一区2区三区4区公司二百 | 国产精品久久久久久久久久久新郎 | 国产在视频一区二区三区吞精 | 欧美理论片在线 | 麻豆成人在线 | jizz在亚洲 | 99在线精品视频 | 开操网| 黄色拍拍视频 | 毛片视频播放 | 99re视频在线 | 黄a免费 | 日韩手机在线观看 | 精品99在线 | 国产精品免费视频观看 | 夜夜爽网址 | 麻豆国产一区二区三区四区 | 亚洲国产精品久久 | av网站免费在线观看 | 天堂在线一区二区 | 久久久成人精品视频 | 欧洲一级黄 | 欧美日韩国产综合网 | 欧美日韩中文在线观看 | 久久线视频 | 精品国产乱码久久久久久88av | 国产精品久久久久久久久久东京 | 成人亚洲电影 | 9久9久 | 久久99精品久久久久久久青青日本 | 极品白嫩少妇无套内谢 | 99久久视频| 日日干夜夜操 | 天天操天天插天天干 | 日韩午夜在线 | 人人干视频| 国产精品永久免费 | 日韩在线观看 | 中文日韩在线 | 一区亚洲| 亚洲一区中文 | 国产一区二区三区久久 | 成人片网址| 国产在线精品一区二区三区 | 能直接看的av网站 | 中文字幕国产一区 | 日本在线观看视频一区 | 精品无码久久久久久国产 | 亚洲国产精品久久久久久 | 亚洲黄色成人 | 黄色资源网站 | 精品国产乱码一区二区三区 | 久久国产成人午夜av影院宅 | 中国妞xxx| 亚洲高清一区二区三区 | 一区二区精品在线 | 亚洲美乳中文字幕 | 天堂精品一区二区三区 | 国产欧美日本 | 国产精品美女久久久久久免费 | 久久综合九九 | 毛片大全| 日本精品一区二区三区在线观看视频 | 国产精品久久久久久久久久东京 | av在线免费观看网站 | 一区二区三区免费 | 日韩在线播放网址 | 97成人在线 | 精品视频在线观看 | 成人黄色在线视频 | 日本一二三视频 | 欧美日韩国产综合视频 | 亚洲一区久久 | 欧美极品一区二区三区 | 国产高清精品一区二区三区 | 国产精品久久久久久亚洲影视 | 杏导航aⅴ福利网站 | www.中文字幕.com | 一级片av | 国产欧美一区二区视频 | 亚洲视频在线观看 | 日韩三级 | 欧美第一网站 | 91亚洲免费 | 久草视频在线看 | 国产一区二区三区在线视频 | 日韩久久久久 | 国产精品一区二区av | 国产精品国产精品国产专区不片 | 在线国产视频 | 美日韩一区二区三区 | 日韩欧美一区在线 | 亚洲免费片| 亚洲精品一区二区三区蜜桃久 | 四虎影院网站 | 天堂福利影院 | 男人的天堂在线视频 | av免费在线播放 | 国产一区二区精品丝袜 | 国产成人一区二区三区 | 国产亚洲精品久久久久久久久 | 国产高清在线精品一区 | 爱免费视频 | 色花av | porn在线视频| 久久99精品视频 | 日本在线视频一区二区 | 免费午夜电影 | 国产亚洲欧美一区二区 | 小情侣高清国产在线播放 | 国产真实乱全部视频 | 国产成人视屏 | 欧美黄视频 | 国产精品永久免费自在线观看 | 日本黄色一级片视频 | 一区二区三区精品 | 日韩在线免费 | 国产一级做a爰片在线看免费 | 日韩中文不卡 | 亚洲欧美日韩国产综合 | 亚洲另类视频 | 午夜视频黄 | www.一区| 久久久久中文字幕 | 日本黄色影片在线观看 | 国产精品久久国产精品 | 一色视频 | 日韩视频一区 | 欧美日韩国产在线观看 | 成人在线免费观看 | 国产精品永久免费自在线观看 | 日本久久免费 | 亚洲 自拍 另类 欧美 丝袜 | 一区视频 | 狠狠ri| 久久国| 欧美日韩视频在线观看一区 | 欧美成人h版在线观看 | 99热免费精品 | 亚洲网色 | 青青久久 | 成人精品一区二区三区中文字幕 | www.成人在线视频 | 这里只有精品在线 | 欧美一级在线免费观看 | 国产精品人人做人人爽 | 成人一级视频在线观看 | 91亚洲国产| 中文字幕亚洲一区二区三区 | 性欧美大战久久久久久久免费观看 | 日韩综合在线 | 日韩欧美一区二区三区视频 | 成人午夜精品一区二区三区 | 国产高清一区二区三区 | 久久国产99 | 综合激情久久 | 欧美黄色一区 | 日韩中文字幕在线免费 | 欧美性v | 日韩一区二区在线观看 | 亚洲国产精品精华液网站 | 欧美激情一区二区三区在线视频 | 97国产超碰 | 欧美激情综合色综合啪啪五月 | 三级黄色片在线观看 | 伊人色综合久久天天五月婷 | 日本美女影院 | 久久男人的天堂 | av大片网| 亚洲精品视频免费 | 爱爱免费视频网站 | 亚洲精品视频大全 | 日韩欧美国产一区二区三区 | 色小妹三区 | 四虎免费在线播放 | 国产综合av | 国产精品久久久久久久久久免费 | 国产成人午夜高潮毛片 | 久草电影网 | 一区二区三区在线播放 | 成人欧美一区二区三区白人 | 超碰人人99 | 精品在线看 | 91天堂| 99色影院 | 成人免费看黄 | 女人久久久久 | 色婷婷av一区二区三区软件 | 96自拍视频| 91综合网 | 91社影院在线观看 | 天天操操| 国产精久久久久久久妇剪断 | 国产视频一区在线 | 久久国产精品免费一区二区三区 | 久久国产精品一区二区 | 中文字幕视频在线 | 国产精品99久久免费观看 | 国产精品美女久久久久久久久久久 | 激情网站免费 | 日韩精品一区二区三区 | 曰批免费视频播放免费 | 国产一区二区三区四区在线观看 | 涩涩天堂| 成人激情视频在线观看 | 色十八 | 国产精品一区久久 | 国产人妖一区二区 | 伊人网综合在线 | 国产精品视频入口 | av国产精品 | 蜜臀网| 亚洲精品电影在线观看 | 国产精品日韩一区二区 | 夸克满天星在线观看 | 在线观看免费的网站www | 亚洲蜜臀av乱码久久精品蜜桃 | 国产乱码精品一区二区三区忘忧草 | 中文字幕7777 | 成人免费视频一区二区 | 一区二区三区在线视频播放 | 五月天狠狠爱 | 欧美 日韩 中文字幕 | 毛片av在线 | 亚洲网站久久 | 亚洲一区视频 | 国产精品美女久久久久aⅴ国产馆 | 久久在线视频 | 日本最新免费二区 | 91免费在线视频 | 91在线视频播放 | 夫妻午夜影院 | 日本高清h色视频在线观看 日日干日日操 | 欧美成人免费在线视频 | 日韩一区二区免费视频 | 国产h视频在线观看 | 国产精品极品美女在线观看免费 | 嫩草成人影院 | 亚洲成年片 | 中国一级大黄大黄大色毛片 | 亚洲色图88 | 在线日韩 | 成人免费在线视频 | 亚洲男人的天堂网站 | 午夜久久 | 亚洲成av人片在线观看 | 欧美亚洲视频 | 国产精品久久久久aaaa九色 | 亚洲三级网站 | 亚洲欧美韩国 | 91亚色 | 国产激情一区二区三区成人免费 | 日韩精品视频免费专区在线播放 | 国产精品资源在线 | 99国产精品视频免费观看一公开 | 超碰导航 | 色婷婷国产精品综合在线观看 | 国产ts余喵喵和直男多体位 | 午夜久久久 | 天天天干天天射天天天操 | 日韩精品在线一区 | 成人免费高清 | 91免费影片 | 久久手机在线视频 | 国产 亚洲 网红 主播 | 久久久精品综合 | 欧美伦理一区二区三区 | 欧美精品一区二区在线观看 | 欧美三级视频在线播放 | 国产成人av在线播放 | 成人福利视频网 | 免费一区二区三区 | 激情欧美一区二区三区中文字幕 | 亚洲精品一区在线观看 | 久久99成人 | 精品免费久久久久 | 色一情一乱一伦一区二区三区 | 日韩午夜场| 爱色区综合网 | 国内自拍偷拍视频 | 久久精品91 | 免费一级在线观看 | baoyu133. con永久免费视频 | 岛国av在线 | 韩国一区二区视频 | 国产精品网站在线观看 | 91视频在线| 国产一在线 | 五月婷婷激情 | 久久99精品久久久久久久青青日本 | 欧美成人一区二区三区片免费 | 欧美一级艳片视频免费观看 | 日韩免费| 欧美视频在线观看不卡 | 精品在线播放 | 日韩精品一二三区 | 久久综合精品视频 | 国产精品网站在线观看 | 日本中文字幕电影 | 久久h| 欧美99| av一区二区三区四区 | 中文字幕在线免费 | 国产男女视频在线观看 | 黄色片免费观看 | 麻豆成人在线 | 欧美美乳| 中文字幕av网 | 亚洲精品成人 | 精品久久久久久久久久久久久久 | 超级黄色一级片 | 亚洲人成人一区二区在线观看 | 国产aⅴ一区二区 | 日本黄色免费大片 | 黄色一级片免费播放 | 欧美日韩精品一区 | 亚洲免费视频在线观看 | 亚洲国产成人精品女人 | 国产精品美女久久久久久久久久久 | 久久久久久网站 | 国产精品国产精品国产专区不片 | 午夜精品网站 | 国产欧美精品一区二区三区 | 日本另类αv欧美另类aⅴ | 日本久久久久 | www.日韩大片 | 91精品久久久久久久久久入口 | 日韩激情视频一区 | 亚洲不卡视频 | 欧美三级电影在线播放 | 色婷婷久久一区二区三区麻豆 | 91久草视频 | 欧美一区二区三区在线观看视频 | 久草久草 | 亚洲综合大片69999 | 国产女人和拘做受在线视频 | 久久久久亚洲一区二区三区 | 国产九九精品 | 黄色av网站在线免费观看 | 精品无码久久久久国产 | av一区在线 | 欧洲一区在线 | 成人高清视频在线观看 | 九九免费在线观看 | 欧美福利视频 | 国产精品久久久久久久久久妞妞 | 天堂中文av在线 | 欧美日韩一区二区视频在线观看 | 亚洲aⅴ天堂av在线电影软件 | 在线观看免费视频91 | 免费国产视频 | 在线成人www免费观看视频 | 免费高清av | 中文字幕高清av | 伊人午夜| 蜜桃视频麻豆女神沈芯语免费观看 |