日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 >

springboot+springsecurity+mybatis+jwt实现单点登录(详细到爆了)

發(fā)布時間:2024/10/5 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 springboot+springsecurity+mybatis+jwt实现单点登录(详细到爆了) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

2021年奉上我最喜歡的一句話:愿你孤獨(dú)的努力都有回報,愿你前行的路上有人陪伴。加油😁!

項目下載:https://download.csdn.net/download/Kevinnsm/16751962

文章目錄

  • 一、springsecuriyt基礎(chǔ)
  • 二、jwt
    • Ⅰ、jwt是什么
    • Ⅱ、為什么要使用jwt
      • 2.1、什么是有狀態(tài)
      • 2.2、什么是無狀態(tài)
    • Ⅲ、jwt的組成
  • 三、springboot+springsecurity+mybatis+jwt實現(xiàn)單點(diǎn)登錄
    • Ⅰ、核心流程分析
    • Ⅱ、具體代碼實現(xiàn)及其分析
      • 1.JwtTokenUtil
      • 2.統(tǒng)一結(jié)果返回封裝類
      • 3、handler處理類
        • 3.1、AjaxAccessDeniedHandler權(quán)限不足處理類
        • 3.2、AjaxAuthenticationEntryPoint匿名無權(quán)限處理類
        • 3.3、AjaxAuthenticationFailureHandler認(rèn)證失敗處理類
        • 3.4、AjaxAuthenticationSuccessHandler認(rèn)證成功處理類
        • 3.5、AjaxLogoutSuccessHandler退出成功處理類
      • 4、AuthenticationFilter(token的驗證類)
      • 5、SecurityConfig配置類
      • 6、自定義異常處理類
        • 6.1、RegisterFailureException注冊失敗異常處理類
        • 6.2、RegisterUsernameHasBeenExists注冊時用戶名已經(jīng)存在異常處理類
        • 6.3、Controller層全局異常處理類
      • 7、controller,service,mapper,entity,mapper.xml編寫
        • 7.1、controller
        • 7.2、service
        • 7.4、mapper
        • 7.5、mapper.xml
        • 7.6、entity
      • 8、application.yml
      • 9、html測試


一、springsecuriyt基礎(chǔ)

1、spring security安全框架入門篇:blog.csdn.net/Kevinnsm/article/details/11

2、springsecurity之用戶認(rèn)證:blog.csdn.net/Kevinnsm/article/details/12

3、springsecurity之用戶授權(quán):blog.csdn.net/Kevinnsm/article/details/13

4、spring security實現(xiàn)登錄和注銷:blog.csdn.net/Kevinnsm/article/details/14

5、springsecurity出現(xiàn)重定向次數(shù)過多:blog.csdn.net/Kevinnsm/article/details/15

6、手把手帶你擼一把springsecurity框架源碼中的認(rèn)證流程:blog.csdn.net/Kevinnsm/article/details/16


二、jwt

Ⅰ、jwt是什么

JWT,全稱是Json Web Token, 是JSON風(fēng)格輕量級的授權(quán)和身份認(rèn)證規(guī)范,可實現(xiàn)無狀態(tài)、分布式的Web應(yīng)用授權(quán)。

Ⅱ、為什么要使用jwt

互聯(lián)網(wǎng)認(rèn)證方式由兩種:無狀態(tài)和有狀態(tài)

2.1、什么是有狀態(tài)

即服務(wù)端需要記錄每次會話的客戶端信息,典型的如session

有狀態(tài)流程

1.當(dāng)用戶登錄后,客戶端會保存該用戶的信息到服務(wù)端的session中,然后返回一個sessionId,將其保存到客戶端的Cookie中

2.當(dāng)下次用戶訪問時攜帶者Cokkie值,這樣服務(wù)端就能識別對應(yīng)的session

缺點(diǎn)

1.服務(wù)端保存大量用戶狀態(tài)信息,增大了服務(wù)端的壓力
2.服務(wù)端保存用戶狀態(tài)信息,無法進(jìn)行水平擴(kuò)展(用戶狀態(tài)信息只保存在這一個服務(wù)器中,訪問其他服務(wù)器還需要重新登錄認(rèn)證)

簡單了說有狀態(tài)就是服務(wù)端需要保存用戶狀態(tài)信息

2.2、什么是無狀態(tài)

服務(wù)端不需要保存任何客戶端的用戶狀態(tài)信息
我是劉松林的而

無狀態(tài)流程

1.用戶第一次訪問時,服務(wù)端要求用戶進(jìn)行身份認(rèn)證(登錄)
2.用戶認(rèn)證成功后,服務(wù)端返回一個token,返回給客戶端作為令牌
3.當(dāng)用戶再次訪問時需要攜帶該token(令牌),服務(wù)端對令牌進(jìn)行解密判斷

優(yōu)點(diǎn)

1.減少了服務(wù)端的壓力
2.服務(wù)端可以進(jìn)行任意的伸縮
3.用戶登錄后可以向多個服務(wù)器進(jìn)行訪問而不用進(jìn)行二次登錄

當(dāng)然無狀態(tài)最重要的就是保證token的安全性

Ⅲ、jwt的組成

Jwt由三部分組成

1、header

header由兩部分組成

1、聲明類型:這里是jwt
2、簽名算法:比如SHA256、SHA512等

header經(jīng)過base64編碼之后就形成了jwt第一部分

2、Payload(載荷)

payload是JWT的主體內(nèi)容部分,也是一個JSON對象,包含需要傳遞的數(shù)據(jù)

iss:發(fā)行人
exp:到期時間
sub:主題

經(jīng)過base64編碼之后形成了jwt的第二部分

3、Signature(簽名)

一般根據(jù)前兩步的數(shù)據(jù),再加上服務(wù)的的密鑰(secret),通過header里面聲明的加密算法生成jwt的第三部分。

secret是服務(wù)端用來進(jìn)行jwt的簽發(fā)和校驗,所以說secret非常重要,任何時候都不應(yīng)該泄露jwt,一旦jwt泄露,意味著惡意用戶可以自我簽發(fā)jwt了。


三、springboot+springsecurity+mybatis+jwt實現(xiàn)單點(diǎn)登錄

Ⅰ、核心流程分析

如果使用springsecurity進(jìn)行過認(rèn)證操作,那么這下面將會對你來說很簡單

1、根據(jù)前述的無狀態(tài)概念,當(dāng)用戶登錄成功后,我們需要在服務(wù)端進(jìn)行token的生成(當(dāng)然這里是使用jwt生成token);然后當(dāng)用戶進(jìn)行再次訪問時,我們需要對token進(jìn)行解析、判斷等;所以我們可以將token的創(chuàng)建、解析、判斷等封裝到一個工具類JwtTokenUtil中。

2、我們還需要自定義一個AuthenticationFilter過濾器用來攔截請求;判斷請求中是否攜帶token,如果攜帶了token,那就進(jìn)行解析判斷;如果沒有token,則可能是首次登錄,所以可以放行,交給springsecurity進(jìn)行認(rèn)證(也就是交給UsernamePasswordAuthenticationFilter過濾器)

3、第二步中有一個關(guān)鍵點(diǎn),我們自定義的AuthenticationFilter過濾器應(yīng)該加到哪個地方,我們都知道springsecurity本質(zhì)上是一串過濾器鏈,將請求進(jìn)行層層攔截、判斷、放行或者不放行。根據(jù)第二步的分析我們可以將AuthenticationFilter過濾器加入到UsernamePasswordAuthenticationFilter過濾器前面,這樣一來第二步就行很清晰了。

其他就類似于springsecurity進(jìn)行認(rèn)證和授權(quán)的步驟了

Ⅱ、具體代碼實現(xiàn)及其分析

1.JwtTokenUtil

package com.jwt.utils;import io.jsonwebtoken.Claims; import io.jsonwebtoken.Jws; import io.jsonwebtoken.Jwts; import io.jsonwebtoken.SignatureAlgorithm;import java.util.Date; import java.util.HashMap; import java.util.Map;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/13* @description:*/ public class JwtTokenUtils {/*** 請求頭*/private static final String TOKEN_HEADER = "Authorization";/*** TOKEN 前綴*/public static final String TOKEN_PREFIX = "Bearer";/*** 私鑰*/private static final String TOKEN_SECRET = "secret";/*** 令牌過期時間 : one day*/private static final long TOKEN_EXPIRATION = 1000 * 60 * 60 * 24;/*** 角色權(quán)限定義*/private static final String ROLE_CLAIMS = "role";/*** 創(chuàng)建令牌** @param username* @param role*/public static String createToken(String username, String role) {Map<String, Object> map = new HashMap<>();map.put(ROLE_CLAIMS, role);return Jwts.builder().setClaims(map).setSubject(username).claim("username", username).setIssuedAt(new Date()).setExpiration(new Date(System.currentTimeMillis() + TOKEN_EXPIRATION)).signWith(SignatureAlgorithm.HS256, TOKEN_SECRET).compact();}/*** 獲取解析后的token信息* @param token* @return*/public static Claims getTokenBody(String token) {return parseToken(token).getBody();}/*** 檢查token是否存在* @param token* @return*/public static Claims checkToken(String token) {try {Claims claims = getTokenBody(token);return claims;} catch (Exception e) {return null;}}/*** 判斷令牌是否過期* @param token 令牌* @return boolean* @describe getExpiration()獲取令牌過期時間*/public static boolean isExpiration(String token) {return getTokenBody(token).getExpiration().before(new Date());}/*** 解析令牌* @param token* @return*/public static Jws<Claims> parseToken(String token) {return Jwts.parser().setSigningKey(TOKEN_SECRET).parseClaimsJws(token);}/*** 解析token,獲取用戶名*/public static String parseTokenToUsername(String token) {String username = getTokenBody(token).getSubject();return username;} }

2.統(tǒng)一結(jié)果返回封裝類

使用泛型,因為在實際的場景中可能由多種不同的數(shù)據(jù)要進(jìn)行返回

package com.jwt.utils;import lombok.Data;import java.io.Serializable;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/12* @description:*/ @Data public class ResponseBody<T> implements Serializable {private Integer code;private String msg;private T data;public ResponseBody(Integer code, String msg) {this.code = code;this.msg = msg;}public ResponseBody(Integer code, String msg, T data) {this.code = code;this.msg = msg;this.data = data;} }

3、handler處理類

這些處理類將在SecurityConfig配置類中進(jìn)行配置,所以需要先將每個handler處理類放到spring容器中====》@Component注解

3.1、AjaxAccessDeniedHandler權(quán)限不足處理類

package com.jwt.handler;import com.alibaba.fastjson.JSON; import org.springframework.security.access.AccessDeniedException; import org.springframework.security.web.access.AccessDeniedHandler; import org.springframework.stereotype.Component;import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/12* @description: 權(quán)限不足處理類*/ @Component public class AjaxAccessDeniedHandler implements AccessDeniedHandler {@Overridepublic void handle(HttpServletRequest request, HttpServletResponse response, AccessDeniedException accessDeniedException) throws IOException, ServletException {response.setStatus(HttpServletResponse.SC_FORBIDDEN);response.getWriter().write(JSON.toJSONString("權(quán)限不足"));} }

3.2、AjaxAuthenticationEntryPoint匿名無權(quán)限處理類

當(dāng)用戶匿名訪問資源時(即不登陸去訪問資源),會跳轉(zhuǎn)到登錄頁面

package com.jwt.handler;import org.springframework.security.core.AuthenticationException; import org.springframework.security.web.AuthenticationEntryPoint; import org.springframework.stereotype.Component;import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description: 匿名未認(rèn)證登錄處理類*/ @Component public class AjaxAuthenticationEntryPoint implements AuthenticationEntryPoint {@Overridepublic void commence(HttpServletRequest request, HttpServletResponse response, AuthenticationException authException) throws IOException, ServletException {response.setContentType("application/json;charset=utf-8");request.getRequestDispatcher("/login.html").forward(request,response);} }

3.3、AjaxAuthenticationFailureHandler認(rèn)證失敗處理類

當(dāng)用戶登錄失敗時,會拋出相應(yīng)的異常,然后交給該類進(jìn)行判斷是哪一種異常。

package com.jwt.handler;import com.alibaba.fastjson.JSON; import lombok.extern.slf4j.Slf4j; import org.springframework.security.authentication.*; import org.springframework.security.core.AuthenticationException; import org.springframework.security.core.userdetails.UsernameNotFoundException; import org.springframework.security.web.authentication.AuthenticationFailureHandler; import org.springframework.stereotype.Component;import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/11* @description: 認(rèn)證失敗Handler處理類*/ @Slf4j //日志 @Component public class AjaxAuthenticationFailureHandler implements AuthenticationFailureHandler {private String url;public AjaxAuthenticationFailureHandler(){}public AjaxAuthenticationFailureHandler(String url) {this.url = url;}@Overridepublic void onAuthenticationFailure(HttpServletRequest request, HttpServletResponse response, AuthenticationException exception) throws IOException, ServletException {String failData = "";if (exception instanceof AccountExpiredException) {failData = "賬號過期";} else if (exception instanceof UsernameNotFoundException) {failData = "賬號不存在";} else if (exception instanceof CredentialsExpiredException) {failData = "密碼過期";} else if (exception instanceof DisabledException) {failData = "賬號不可用";} else if (exception instanceof LockedException) {failData = "賬號鎖定";} else if (exception instanceof BadCredentialsException) {failData = "密碼錯誤";} else {failData = "未知異常";}log.info("認(rèn)證失敗,"+failData);//設(shè)置編碼格式,否則中文會亂碼response.setCharacterEncoding("utf-8);response.setContentType("application/json;charset=utf-8");//返回統(tǒng)一數(shù)據(jù)response.getWriter().write(JSON.toJSONString(failData));} }

3.4、AjaxAuthenticationSuccessHandler認(rèn)證成功處理類

當(dāng)用戶認(rèn)證成功后,將交與該處理類處理;返回token(前綴帶bearer,所以服務(wù)端在進(jìn)行解析token時,應(yīng)該將bearer去掉。

package com.jwt.handler;import com.alibaba.fastjson.JSON; import com.jwt.entity.User; import com.jwt.utils.JwtTokenUtils; import com.jwt.utils.ResponseBody; import org.springframework.security.core.Authentication; import org.springframework.security.web.authentication.AuthenticationSuccessHandler; import org.springframework.stereotype.Component;import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/11* @description: 認(rèn)證成功Handler處理類*/ @Component public class AjaxAuthenticationSuccessHandler implements AuthenticationSuccessHandler {private String url;public AjaxAuthenticationSuccessHandler() {}public AjaxAuthenticationSuccessHandler(String url) {this.url = url;}@Overridepublic void onAuthenticationSuccess(HttpServletRequest request, HttpServletResponse response, Authentication authentication) throws IOException, ServletException {User user = (User) authentication.getPrincipal();String token = JwtTokenUtils.createToken(user.getUsername(), String.valueOf(user.getAuthorities()));//設(shè)置編碼,如果不設(shè)置會亂碼response.setCharacterEncoding("utf-8");response.setContentType("application/json;charset=utf-8");//設(shè)置返回的token 帶有Bearer的前綴字符串response.setHeader("Authorization", JwtTokenUtils.TOKEN_PREFIX+token);response.getWriter().write(new ResponseBody<User>(200,"登錄成功",user).toString());} }

3.5、AjaxLogoutSuccessHandler退出成功處理類

當(dāng)用戶退出登錄成功時,跳轉(zhuǎn)到登錄頁面

package com.jwt.handler;import lombok.extern.slf4j.Slf4j; import org.springframework.security.core.Authentication; import org.springframework.security.web.authentication.logout.LogoutSuccessHandler; import org.springframework.stereotype.Component;import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/16* @description:*/ @Component @Slf4j public class AjaxLogoutSuccessHandler implements LogoutSuccessHandler {@Overridepublic void onLogoutSuccess(HttpServletRequest request, HttpServletResponse response, Authentication authentication) throws IOException, ServletException {request.getRequestDispatcher("/login.html").forward(request,response);} }

4、AuthenticationFilter(token的驗證類)

該類對請求進(jìn)行攔截,配置到UsernamePasswordAuthenticationFilter過濾器類的前面,至于功能前面已經(jīng)分析過。

Spring Security使用一個Authentication對象來描述當(dāng)前用戶的相關(guān)信息。SecurityContextHolder中持有的是當(dāng)前用戶的SecurityContext,而SecurityContext持有的是代表當(dāng)前用戶相關(guān)信息的Authentication的引用。這個Authentication對象不需要我們自己去創(chuàng)建,在與系統(tǒng)交互的過程中,Spring Security會自動為我們創(chuàng)建相應(yīng)的Authentication對象,然后賦值給當(dāng)前的SecurityContext,我們經(jīng)常會使用SecurityContextHolder獲取SecurityContext實例,然后獲取Authentication實例。下面會是使用到

package com.jwt.filter; import com.jwt.entity.User; import com.jwt.service.LoginUserDetailService; import com.jwt.utils.JwtTokenUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.security.core.userdetails.UserDetails; import org.springframework.security.web.authentication.WebAuthenticationDetailsSource; import org.springframework.stereotype.Component; import org.springframework.web.filter.OncePerRequestFilter;import javax.servlet.FilterChain; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;@Component public class AuthenticationFilter extends OncePerRequestFilter {@AutowiredLoginUserDetailService userDetailsService;@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {//獲取請求頭中的token加密串(名字是Authorization,JwtTokenUtils工具類里面定義的String authHeader = request.getHeader("Authorization"); //判斷是否是以Bearer開頭的,這是我自定義的前綴名)if (authHeader != null && authHeader.startsWith("Bearer ")) {//將Bearer去掉,因為返回token時,我在token前綴加入了Bearer字符串String authToken = authHeader.substring("Bearer ".length()); //解析加密串,獲取用戶名String username = JwtTokenUtils.parseTokenToUsername(authToken);//看當(dāng)前SecurityContext中是否有Authentication實例(前面已經(jīng)介紹過)if (username != null && SecurityContextHolder.getContext().getAuthentication() == null) { //如果Authentication沒有當(dāng)前用戶的信息,然后從數(shù)據(jù)庫中查詢出相應(yīng)的信息User user = (User) userDetailsService.loadUserByUsername(username);if (user != null) {//封裝到UsernamePasswordAuthenticationToken令牌類中UsernamePasswordAuthenticationToken authentication =new UsernamePasswordAuthenticationToken(user, user.getPassword(), user.getAuthorities());authentication.setDetails(new WebAuthenticationDetailsSource().buildDetails(request)); //將用戶信息放到Authenticatin實例中進(jìn)行存儲SecurityContextHolder.getContext().setAuthentication(authentication);}}}filterChain.doFilter(request, response);} }

這個過濾器類里面是jwt最核心的點(diǎn)
首先請求進(jìn)入到該過濾器類時,獲取token串,進(jìn)行解析;然后查看Authentication是否為null,如果為null,根據(jù)解析到的用戶名去數(shù)據(jù)庫查詢相應(yīng)的信息,一并將其放到Authentication實例中。
當(dāng)你下一次再進(jìn)行token判斷時,直接從Authentication拿取相應(yīng)信息即可,不需要再查詢數(shù)據(jù)庫。

5、SecurityConfig配置類

package com.jwt.config; import com.jwt.filter.AuthenticationFilter; import com.jwt.handler.*; import com.jwt.service.LoginUserDetailService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.security.authentication.AuthenticationManager; import org.springframework.security.authentication.AuthenticationProvider; import org.springframework.security.authentication.dao.DaoAuthenticationProvider; import org.springframework.security.config.annotation.authentication.builders.AuthenticationManagerBuilder; import org.springframework.security.config.annotation.web.builders.HttpSecurity; import org.springframework.security.config.annotation.web.builders.WebSecurity; import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter; import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter; import org.springframework.security.web.authentication.rememberme.JdbcTokenRepositoryImpl; import org.springframework.security.web.authentication.rememberme.PersistentTokenRepository;import javax.annotation.Resource; import javax.sql.DataSource;@Configuration public class SecurityConfig extends WebSecurityConfigurerAdapter {@Resourceprivate LoginUserDetailService userDetailsService;@Autowiredprivate AjaxAccessDeniedHandler deniedHandler;@Autowiredprivate AjaxAuthenticationEntryPoint ajaxAuthenticationEntryPoint;@Autowiredprivate AjaxAuthenticationSuccessHandler successHandler;@Autowiredprivate AjaxAuthenticationFailureHandler failureHandler;@Autowiredprivate DataSource dataSource;@Autowiredprivate AuthenticationFilter authenticationFilter;@Autowiredprivate AjaxLogoutSuccessHandler logoutSuccessHandler;/*** 攔截策略* @param http* @throws Exception*/@Overrideprotected void configure(HttpSecurity http) throws Exception {http.authorizeRequests().and().formLogin().loginPage("/login.html").loginProcessingUrl("/login").successHandler(successHandler).failureHandler(failureHandler).and().authorizeRequests().antMatchers("/login.html","/register.html","/register","/js/**").permitAll().and().exceptionHandling().accessDeniedHandler(deniedHandler).authenticationEntryPoint(ajaxAuthenticationEntryPoint).and().authorizeRequests().anyRequest().authenticated();http.addFilterBefore(authenticationFilter, UsernamePasswordAuthenticationFilter.class);//設(shè)置記住我http.rememberMe().tokenRepository(persistentTokenRepository()).tokenValiditySeconds(600).userDetailsService(userDetailsService);//配置退出登錄操作http.logout().logoutUrl("/logout").logoutSuccessHandler(logoutSuccessHandler);//關(guān)閉csrf防護(hù)http.csrf().disable();}@Beanpublic BCryptPasswordEncoder bCryptPasswordEncoder(){return new BCryptPasswordEncoder();}/*** 配置忽略的URL* @param web* @throws Exception*/@Overridepublic void configure(WebSecurity web) throws Exception {web.ignoring().antMatchers("/");}/*** 攔截后需要使用自定義的類和加密解密方式* @param auth* @throws Exception*/@Overrideprotected void configure(AuthenticationManagerBuilder auth) throws Exception {auth.authenticationProvider(daoAuthenticationProvider());auth.userDetailsService(userDetailsService).passwordEncoder(bCryptPasswordEncoder());}@Override@Beanprotected AuthenticationManager authenticationManager() throws Exception {return super.authenticationManager();}/*** * @return HideUserNotFoundExceptions(false),否則UsernameNotFoundException異常會被BadCredentialsException異常覆蓋*/@Beanpublic AuthenticationProvider daoAuthenticationProvider() {DaoAuthenticationProvider daoAuthenticationProvider = new DaoAuthenticationProvider();daoAuthenticationProvider.setUserDetailsService(userDetailsService);daoAuthenticationProvider.setPasswordEncoder(new BCryptPasswordEncoder());daoAuthenticationProvider.setHideUserNotFoundExceptions(false);return daoAuthenticationProvider;}//這是spirngsecurity記住我功能需要加入代碼 //JdbcTokenRepositoryImpl是將其保存到數(shù)據(jù)庫中@Beanpublic PersistentTokenRepository persistentTokenRepository() {//這個是記住我保存到數(shù)據(jù)庫的類,當(dāng)然還有保存到內(nèi)存的類JdbcTokenRepositoryImpl tokenRepository = new JdbcTokenRepositoryImpl();//設(shè)置數(shù)據(jù)源tokenRepository.setDataSource(dataSource);//這個會在第一次使用記住我時在數(shù)據(jù)庫中創(chuàng)建一張表(用戶、過期時間等)//在第二次一定要將其注釋掉,否者會報錯。// tokenRepository.setCreateTableOnStartup(true);return tokenRepository;} }

6、自定義異常處理類

6.1、RegisterFailureException注冊失敗異常處理類

package com.jwt.exception;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description:*/ public class RegisterFailureException extends Exception{private String message;public RegisterFailureException() {super();}public RegisterFailureException(String message) {this.message = message;} }

6.2、RegisterUsernameHasBeenExists注冊時用戶名已經(jīng)存在異常處理類

package com.jwt.exception;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/15* @description:*/ public class RegisterUsernameHasBeenExists extends Exception{private String message;public RegisterUsernameHasBeenExists(String message) {this.message = message;} }

6.3、Controller層全局異常處理類

package com.jwt.exception.controllerException;import com.jwt.exception.RegisterUsernameHasBeenExists; import lombok.extern.slf4j.Slf4j; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.http.HttpStatus; import org.springframework.validation.BindException; import org.springframework.validation.BindingResult; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.ResponseStatus; import org.springframework.web.bind.annotation.RestControllerAdvice;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.sql.SQLException; import java.util.Objects;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/15* @description:*/ @Slf4j @RestControllerAdvice public class ControllerHandlerExceptionAdvice {private static final Logger logger = LoggerFactory.getLogger(ControllerHandlerExceptionAdvice.class);/*** 攔截表單參數(shù)異常處理* @param exception* @param request* @return*/@ResponseStatus(HttpStatus.OK)@ExceptionHandler({BindException.class})public String bindException(BindException exception, HttpServletRequest request) {logger.info("表單攔截校驗處理=====>");logger.info("表單數(shù)據(jù)======>"+request.getContentType());BindingResult bindingResult = exception.getBindingResult();return Objects.requireNonNull(bindingResult.getFieldError().getDefaultMessage());}@ExceptionHandlerpublic String handler(HttpServletRequest request, HttpServletResponse response, Exception e) {logger.info("RestFul 請求發(fā)生異常........");if (response.getStatus() == HttpStatus.BAD_REQUEST.value()) {logger.info("狀態(tài)值不是200,正準(zhǔn)備修改為200");response.setStatus(HttpStatus.OK.value());}if (e instanceof NullPointerException) {logger.error("發(fā)生了空指針異常======》",e.getMessage());return "空指針異常";} else if (e instanceof IllegalArgumentException) {logger.error("請求參數(shù)不匹配異常======>",e.getMessage());return "請求參數(shù)不匹配";} else if (e instanceof SQLException) {logger.error("數(shù)據(jù)庫訪問異常======>",e.getMessage());return "數(shù)據(jù)庫訪問異常";} else if (e instanceof BindException) {BindingResult bindingResult = ((BindException) e).getBindingResult();logger.error("表單校驗異常",bindingResult.getFieldError().getDefaultMessage());return Objects.requireNonNull(bindingResult.getFieldError().getDefaultMessage());} else if (e instanceof RegisterUsernameHasBeenExists) {logger.info("用戶名已經(jīng)存在");return "用戶名已經(jīng)存在";} else {logger.error("未知異常",e.getMessage());return "服務(wù)器端未知異常,請去檢查!";}} }

7、controller,service,mapper,entity,mapper.xml編寫

這幾個類就是簡單的crud了,沒什么可說的

7.1、controller

package com.jwt.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/7* @description:*/ @Controller public class LoginController {@RequestMapping("/toMain")public String main() {return "redirect:success.html";}@RequestMapping("/")public String demo() {return "main";}@RequestMapping("/logout")public String logout() {return "redirect:login.html";} } package com.jwt.controller;import com.jwt.entity.User; import com.jwt.entity.UserDto; import com.jwt.exception.RegisterFailureException; import com.jwt.exception.RegisterUsernameHasBeenExists; import com.jwt.service.RegisterService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.RequestMapping;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description:*/ @Controller @Slf4j public class RegisterController {@Autowiredprivate RegisterService registerService;@RequestMapping("/register")public String register(@Validated UserDto userDto) throws RegisterUsernameHasBeenExists {log.info("注冊的數(shù)據(jù)為:"+userDto.toString() );User user = registerService.selectDup(userDto.getUsername());if (user == null) {throw new RegisterUsernameHasBeenExists("用戶已存在");}try {registerService.register(userDto);} catch (RegisterFailureException e) {e.printStackTrace();}return "/";} } package com.jwt.controller;import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity; import org.springframework.web.bind.annotation.*;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/15* @description:*/ @RestController @RequestMapping("/api") @EnableWebSecurity public class TestController {@RequestMapping("/add")public String add() {return "add";}@RequestMapping("select")public String select() {return "select";}@RequestMapping("/delete")public String delete() {return "delete";}@RequestMapping("/update")public String update() {return "update";} }

7.2、service

package com.jwt.service;import com.jwt.entity.User; import com.jwt.entity.UserDto; import com.jwt.exception.RegisterFailureException; import com.jwt.exception.RegisterUsernameHasBeenExists;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description:*/ public interface RegisterService {boolean register(UserDto userDto) throws RegisterFailureException;User selectDup(String username) throws RegisterUsernameHasBeenExists; }

這個LoginUserDetailService是要實現(xiàn)UserDetailService,根據(jù)用戶名查詢數(shù)據(jù)數(shù)據(jù)庫用戶信息,將其交給SpringSecurity進(jìn)行認(rèn)證

至于為什么要這樣:https://blog.csdn.net/Kevinnsm/article/details/115189976

package com.jwt.service;import com.jwt.entity.User; import com.jwt.mapper.LoginMapper; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.security.core.GrantedAuthority; import org.springframework.security.core.authority.SimpleGrantedAuthority; import org.springframework.security.core.userdetails.UserDetails; import org.springframework.security.core.userdetails.UserDetailsService; import org.springframework.security.core.userdetails.UsernameNotFoundException; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.stereotype.Service;import java.util.ArrayList; import java.util.List;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/7* @description: 根據(jù)傳來的username去數(shù)據(jù)庫中查詢用戶信息,然后交給springsecurity去認(rèn)證*/ @Service @Slf4j public class LoginUserDetailService implements UserDetailsService {@Autowiredprivate LoginMapper loginMapper;@Autowiredprivate PasswordEncoder passwordEncoder;/**** 自定義認(rèn)證* @param username* @return* @throws UsernameNotFoundException*/@Overridepublic UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {if (username == null || username.equals("")) {throw new RuntimeException("用戶名不能為空!");}User user = loginMapper.findByUsername(username);if (user == null) {throw new UsernameNotFoundException("用戶不能為空!");}log.info("根據(jù)username查詢的用戶信息為===>"+user.toString() + "==》等待認(rèn)證!");List<GrantedAuthority> authorities = new ArrayList<>(); // loginMapper.findAuthorityByUsername(username).forEach(role-> // authorities.add(new SimpleGrantedAuthority((String) role)) // );authorities.add(new SimpleGrantedAuthority(loginMapper.findAuthorityByUsername(username)));user.setAuthorities(authorities);log.info(user.getUsername()+"用戶的所有權(quán)限為權(quán)限===>"+authorities.toString());return new User(user.getUsername(), passwordEncoder.encode(user.getPassword()),authorities);}} package com.jwt.service.impl;import com.jwt.entity.User; import com.jwt.entity.UserDto; import com.jwt.exception.RegisterFailureException; import com.jwt.exception.RegisterUsernameHasBeenExists; import com.jwt.mapper.RegisterMapper; import com.jwt.service.RegisterService; import com.sun.deploy.association.RegisterFailedException; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description:*/ @Service @Slf4j public class RegisterServiceImpl implements RegisterService {@Autowiredprivate RegisterMapper registerMapper;@Overridepublic boolean register(UserDto userDto) throws RegisterFailureException {boolean b = registerMapper.register(userDto);System.out.println(b);if (!b) {log.info("注冊失敗");throw new RegisterFailureException("注冊失敗");}return b;}@Overridepublic User selectDup(String username) throws RegisterUsernameHasBeenExists {User user = registerMapper.selectDup(username);if (user != null) {throw new RegisterUsernameHasBeenExists("用戶名已經(jīng)存在");}return null;} }

7.4、mapper

@Repository public interface LoginMapper {User findByUsername(String username);String findAuthorityByUsername(String username); } @Repository public interface RegisterMapper {boolean register(UserDto userDto);User selectDup(String username); }

7.5、mapper.xml

<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.jwt.mapper.LoginMapper"><select id="findByUsername" resultType="User" parameterType="string">select * from user where username = #{username} limit 1</select><select id="findAuthorityByUsername" resultType="string">select authority from role where username = #{username} limit 1</select> </mapper> <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.jwt.mapper.RegisterMapper"><insert id="register" parameterType="userDto">insert into user (username,password) values(#{username},#{password})</insert><select id="selectDup" resultType="user">select * from user where username=#{username} limit 1</select> </mapper>

7.6、entity

@Data @NoArgsConstructor public class User implements UserDetails {private String username;private String password;private List<GrantedAuthority> authorities;@Overridepublic Collection<? extends GrantedAuthority> getAuthorities() {return this.authorities;}public User(String username, String password, List<GrantedAuthority> authorities) {this.username = username;this.password = password;this.authorities = authorities;}@Overridepublic boolean isAccountNonExpired() {return true;}@Overridepublic boolean isAccountNonLocked() {return true;}@Overridepublic boolean isCredentialsNonExpired() {return true;}@Overridepublic boolean isEnabled() {return true;} } @Data public class Authority implements Serializable {private String roles; } package com.jwt.entity;import com.sun.istack.internal.NotNull; import lombok.Data;import javax.validation.constraints.Size; import java.io.Serializable;/*** @author:抱著魚睡覺的喵喵* @date:2021/4/14* @description:*/ @Data public class UserDto implements Serializable {@NotNull@Size(min = 3, max = 15, message = "用戶名必須在3~15之間")private String username;@NotNull@Size(min = 5, max = 15, message = "密碼必須在5~15之間")private String password; }

8、application.yml

spring:datasource:url: jdbc:mysql://localhost:3306/security?serverTimezone=UTCusername: rootpassword: hao20001010mybatis:mapper-locations: mapper/**type-aliases-package: com.jwt.entityserver:port: 8081

9、html測試

templates模板下的main.html

<!DOCTYPE html> <html lang="en" xmlns="http://www.w3.org/1999/xhtml" > <head><meta charset="UTF-8"><title>Title</title> </head> <body><h1>這是主頁面</h1> <br/><a href="login.html">登錄</a><a href="register.html">注冊</a><br><hr><h2>操作如下</h2><a href="/api/add">添加商品</a><a href="/api/select">查詢商品</a><a href="/api/update">修改商品</a><a href="/api/delete">刪除商品</a><a href="/logout">退出登錄</a></body> </html>

static文件下

<!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Title</title></head> <body> <h2>=======登錄頁面=========</h2><form action="/login" method="post">username:<input type="text" name="username"/><br/>password:<input type="password" name="password" /><br/>remember me:<input type="checkbox" name="remember-me" value="true"/><br/><input type="submit" value="提交" /></form> </body></html> <!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Title</title> </head> <body> <h2>=========注冊頁面=========</h2> <form action="/register" method="post"><table>username:<input type="text" name="username"/><br/>password:<input type="password" name="password"/><br/><input type="submit" value="提交"/></table> </form> </body> </html> <!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Title</title> </head> <body> <h1>分頁面</h1><br> <a href="/toMain">主頁面</a><br> <a href="/logout">退出</a></body> </html>

到這里也就是結(jié)束了,測試截圖就不再發(fā)了,Controller層應(yīng)該使用Rest風(fēng)格的。

回顧這個小demo,其實簡單了分析就是:springsecurity是一串過濾器鏈,然后自定義jwt的校驗token過濾器,最后將其加入到過濾器鏈中,至于加到什么位置,想必都明白了。

核心就前端那一段,其他的基本數(shù)就是springsecurity的基本操作和一些crud操作及其某些細(xì)節(jié)。把握住核心即可。

總結(jié)

以上是生活随笔為你收集整理的springboot+springsecurity+mybatis+jwt实现单点登录(详细到爆了)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

a视频在线看 | 国产精品视频一二三 | 久热免费 | 西西www4444大胆视频 | 青青草国产在线 | 免费视频久久久 | 香蕉视频在线视频 | 黄色在线网站噜噜噜 | 中文字幕在线视频第一页 | 精品国产免费观看 | 人人狠狠 | 国产又粗又猛又色又黄网站 | 久久国产精品一区二区三区四区 | 中文字幕在线精品 | 美女在线免费观看视频 | 操操操com | 精品久久久久久久久久久久久 | 国产精品久久久av久久久 | 国产资源免费在线观看 | 欧美少妇18p | 欧美激情奇米色 | 亚洲国产影院av久久久久 | 国产一区二区三精品久久久无广告 | 97在线观看免费观看高清 | 在线国产视频一区 | 婷婷激情五月 | 国产精品综合久久久久 | 在线看一区 | 国产中文字幕在线看 | 97超碰人人看 | 亚洲国产经典视频 | 日韩在线免费视频观看 | 亚洲手机天堂 | 国产区第一页 | 色婷婷国产精品一区在线观看 | 日韩精品在线播放 | 亚洲一级在线观看 | 国产不卡在线观看视频 | 日韩欧美一区二区三区免费观看 | 99在线视频网站 | 99热.com| 高清不卡毛片 | 欧美色图30p| 国产精品视频app | 青青草国产精品视频 | 精品欧美日韩 | 天天射成人 | 亚洲无吗av| 国产黄色精品视频 | 国产精品永久免费 | 在线播放日韩av | 久久国产一二区 | 久久久午夜剧场 | 97超碰人人澡人人爱 | 亚洲乱亚洲乱亚洲 | 国产又粗又猛又爽又黄的视频先 | 91丨九色丨丝袜 | 97视频一区| 丰满少妇在线观看网站 | 色综合欧洲 | 狠狠干夜夜操天天爽 | 中文字幕一区2区3区 | 成年人免费看的视频 | 999在线观看视频 | 最近最新最好看中文视频 | 九九在线国产视频 | 国产精品成人久久 | 欧美性春潮 | 免费av网址在线观看 | 亚洲免费专区 | 青青草国产免费 | 亚洲草视频 | 国产黄色片一级三级 | 亚洲第二色 | 日韩免费观看一区二区 | 色噜噜日韩精品一区二区三区视频 | 尤物一区二区三区 | 欧美日韩国产一区二区在线观看 | 97色在线观看免费视频 | 91资源在线观看 | 国偷自产中文字幕亚洲手机在线 | 精品国产亚洲在线 | 五月婷婷丁香 | 亚洲区另类春色综合小说 | 国际av在线 | 精品美女在线视频 | 综合精品久久 | 精品久久久久久久久久久久久久久久久久 | 中文字幕黄色av | 日日夜夜婷婷 | 久热爱 | 国产一区精品在线观看 | 免费在线观看成人 | 啪啪激情网 | 亚洲欧美视频一区二区三区 | 国产一级电影在线 | 在线观看mv的中文字幕网站 | 国产精品一区二区在线免费观看 | 九九热1 | 亚洲国内精品在线 | 九色精品免费永久在线 | 欧美性色黄 | 日韩啪啪小视频 | 亚洲精品久久视频 | 欧美日韩国产一区二区三区 | 亚洲国产网址 | 狠狠狠色丁香综合久久天下网 | 久久免费资源 | 91视频亚洲 | 黄色精品视频 | 日韩一区视频在线 | 色干干| 国产色在线视频 | 国产视频精品在线 | 国产精品高潮在线观看 | 精品国产成人在线影院 | 久久久高清 | 久久高清国产 | 在线观看视频中文字幕 | 久久九九国产视频 | 国产黄色视 | 福利网址在线观看 | 欧美在线视频二区 | 亚洲精品国产综合99久久夜夜嗨 | 黄p在线播放| 欧美国产日韩在线视频 | 婷婷精品国产欧美精品亚洲人人爽 | 日韩午夜在线播放 | 国内精品久久久久久久97牛牛 | 一级黄视频 | 黄色软件在线观看免费 | 欧美性色综合网 | 在线观看日韩一区 | 欧美黄色成人 | 国产精品人人做人人爽人人添 | 久久精品免费观看 | 人人澡人摸人人添学生av | 成年人视频在线免费观看 | 久久久久久久久久久国产精品 | 国产精品成人国产乱 | 99久久精品免费视频 | 激情五月网站 | 最近更新中文字幕 | 久久精品精品电影网 | 激情五月在线视频 | 99久久99视频只有精品 | 黄色91在线观看 | 国产二区免费视频 | 男女激情网址 | 国产在线高清精品 | 免费日韩视频 | 久草在线中文视频 | 超碰97免费观看 | 一区二区三区不卡在线 | 99在线高清视频在线播放 | 亚洲成人午夜av | 99精品国产一区二区三区麻豆 | 午夜丁香视频在线观看 | 亚洲色图色 | 国产精品自产拍在线观看蜜 | 国产黄视频在线观看 | 国产在线视频一区 | 久久久午夜电影 | 成人黄在线 | 九九热精品视频在线观看 | 久久电影色 | 欧美性黑人 | 99精品免费| 成人av在线播放网站 | 免费午夜在线视频 | 国产精品mv在线观看 | 欧美a在线看 | 久久九精品| 男女视频国产 | 久久99久久精品 | 久久久久www| 美女精品 | 久久久久97国产 | 天天爽人人爽 | 国产黄a三级三级 | 久久久久成| 色狠狠狠 | 亚洲va欧美 | 国产黄色在线网站 | 中文字幕在线观看免费高清电影 | 国产一区免费看 | 欧美性超爽 | 国产精品一区欧美 | 欧美a级片网站 | 99久久久久 | 黄色av网站在线免费观看 | 亚洲精品视频网址 | 九热在线 | a在线免费观看视频 | 亚洲三级在线播放 | 日韩免费观看高清 | 国产高清免费在线观看 | 国产美女精品在线 | 免费av片在线 | 欧美综合在线视频 | 激情视频在线高清看 | 午夜精品久久久久久99热明星 | www.五月婷 | 国内一区二区视频 | 成年人视频在线免费 | 视频在线观看一区 | 久久精品三 | 麻豆免费在线视频 | 成人黄色在线看 | 久久国产剧场电影 | 日韩精品久久久久久中文字幕8 | 国产视频2区 | www色网站 | 黄网站www| 色香蕉在线视频 | 精品国产观看 | 免费av网址在线观看 | 91热视频 | av免费观看高清 | 三级大片网站 | 亚洲四虎 | 久久综合婷婷 | 国产色网| 97热在线观看 | 中文字幕av一区二区三区四区 | 美女黄网站视频免费 | 欧美日本一区 | 国产一区二区三区免费在线 | 国产老太婆免费交性大片 | 国产在线a不卡 | 亚洲精品乱码久久久久久9色 | 亚洲人天堂 | 国产在线精品福利 | 久久观看最新视频 | 日韩电影黄色 | 久久久99国产精品免费 | 国产精品毛片一区视频播不卡 | 成人一区二区在线 | 午夜av在线电影 | 婷婷在线免费 | 玖玖999 | 成人免费视频播放 | 一区二区精品视频 | 亚洲免费在线视频 | www.色com | 欧美精品一区二区免费 | 欧美日韩免费在线观看视频 | 天堂av色婷婷一区二区三区 | 亚洲2019精品 | 久久精品视频免费观看 | 日日夜夜天天综合 | 久久国产美女视频 | 国产精品久久久久久久久久三级 | a国产精品| 日韩精品一区二区免费 | 欧美性脚交| 国产精品 日韩 欧美 | 亚洲精品福利在线 | 中文字幕精 | 日本性xxxxx 亚洲精品午夜久久久 | 米奇四色影视 | 欧美日韩中文字幕在线视频 | 日韩欧美精品在线观看视频 | 久久99视频免费 | 91麻豆精品国产91久久久无限制版 | 国产一级视频在线观看 | 成人av资源网站 | 五月婷婷一区 | 午夜国产一区 | 97精产国品一二三产区在线 | 亚洲精品国产麻豆 | 国产精品18久久久久久久网站 | 91最新地址永久入口 | 天天天插 | 最新91在线视频 | 亚洲精品视频在线免费 | 中文字幕久久精品 | 91豆花在线 | 国产精品久久久久久电影 | 91免费视频黄 | 亚洲国产精品女人久久久 | avove黑丝 | 亚洲最大免费成人网 | 九九视频免费观看视频精品 | 日韩av午夜在线观看 | 香蕉视频久久 | 中文字幕高清免费日韩视频在线 | 99久久精品免费看国产一区二区三区 | 天堂视频中文在线 | 国产黄色播放 | 亚洲精品成人av在线 | 欧美视频在线观看免费网址 | 久久在线观看视频 | 亚洲免费一级电影 | 久久久久久久久久电影 | 欧美极品xxxxx | 国产成人黄色在线 | 亚洲欧美日韩精品久久奇米一区 | 91香蕉嫩草 | 国产福利一区二区三区在线观看 | 免费一级日韩欧美性大片 | 国产999在线观看 | 天天射天 | 超碰成人网 | 九九免费观看全部免费视频 | 五月婷婷丁香激情 | 四虎成人精品永久免费av | 精品久久91| 国产成人一区二 | 伊在线视频 | 中文字幕在线一区观看 | 国产精品久久久999 国产91九色视频 | 欧美日韩久久不卡 | 国产午夜精品一区二区三区在线观看 | 国产精品不卡一区 | 国产精品乱码一区二三区 | 麻豆av一区二区三区在线观看 | 91丨九色丨国产在线 | 在线观看久 | 久久久久激情电影 | 中文字幕乱码视频 | 激情欧美一区二区三区 | 国产精品一区二区三区在线看 | 精品国产1区 | av在线网站免费观看 | 国产精品成人自产拍在线观看 | 精品亚洲欧美无人区乱码 | 精品女同一区二区三区在线观看 | 九九免费观看全部免费视频 | 黄色免费电影网站 | 9在线观看免费高清完整 | 精品亚洲男同gayvideo网站 | 免费在线成人av | 在线观看免费一级片 | 成年人免费看的视频 | 婷婷在线色 | 日韩精品观看 | 91成人免费看 | 日韩在线视频观看 | 欧美大香线蕉线伊人久久 | 久久久久国产成人精品亚洲午夜 | 国产午夜一级毛片 | 国产高清久久久久 | 伊人伊成久久人综合网站 | 久久与婷婷 | 国产精品第72页 | 蜜臀av性久久久久av蜜臀妖精 | 黄色精品一区 | 最新国产精品亚洲 | 一级精品视频在线观看宜春院 | 高清一区二区三区 | 国产特级毛片aaaaaaa高清 | 亚洲乱码国产乱码精品天美传媒 | 五月婷婷丁香色 | 天堂av观看 | 国产一区二区在线免费播放 | 天操夜夜操 | 精品久久中文 | 久久久久亚洲国产精品 | av色图天堂网 | 九九免费在线看完整版 | 夜夜躁日日躁狠狠躁 | 成人午夜网址 | 久久精品79国产精品 | 久久精品视频播放 | 日韩欧美一区二区不卡 | 久久精品亚洲 | 91成人免费在线 | 久久久麻豆精品一区二区 | 91精品综合在线观看 | 美女在线免费观看视频 | 夜夜操天天 | 综合色婷婷 | 超碰97.com| 一区二区三区久久精品 | 91桃色在线免费观看 | 国产精品久久久久久久久久久久冷 | 午夜精品久久久久久久久久 | 国产精品视频地址 | 天天操天天舔天天爽 | 国产探花在线看 | 中文字幕av专区 | 国产高清在线精品 | 美国av片在线观看 | 婷婷六月丁香激情 | 亚洲男男gaygay无套同网址 | 国产高清视频在线播放一区 | avwww在线| 国产黄色大片免费看 | 欧美激情综合五月色丁香 | 欧美巨乳波霸 | avwww在线| 日韩精品一卡 | 99热国内精品 | 色婷婷综合久色 | 久久精品久久久久久久 | 国产午夜在线 | 美女黄网站视频免费 | 视频一区二区免费 | 天天在线操 | 中文字幕a∨在线乱码免费看 | 国产精品久久久久久a | 中文字幕乱码一区二区 | 久久久午夜视频 | 精品在线观看一区二区三区 | 日韩免费播放 | 亚洲精品欧美专区 | 日本一区二区三区免费观看 | 免费a网址 | 激情欧美丁香 | 亚洲春色成人 | 亚洲伦理一区二区 | 色婷婷狠狠五月综合天色拍 | 最新日本中文字幕 | 在线观看国产福利片 | 国产成人一级电影 | 97在线视频免费观看 | 麻豆国产视频下载 | 日韩有码网站 | 免费亚洲视频在线观看 | 一级欧美黄 | 欧美精品久久久久久久久久白贞 | 射久久久 | 日韩系列在线观看 | 黄色电影在线免费观看 | 国产视频观看 | 欧美va天堂va视频va在线 | 中文国产在线观看 | 久久曰视频 | 久久精品一区八戒影视 | 国产精品美女视频网站 | 久久精品之 | 超碰97.com| 亚洲精品看片 | 国产精品福利无圣光在线一区 | 久久久久亚洲天堂 | 色诱亚洲精品久久久久久 | 在线成人观看 | 91九色蝌蚪国产 | 97超碰福利久久精品 | 日韩精品中文字幕在线不卡尤物 | 韩日精品在线 | 久在线观看视频 | 91中文字幕网 | 欧美精品久久久久久久久久 | 免费看精品久久片 | 日韩中文字幕免费电影 | 国产99久久99热这里精品5 | 91精品婷婷国产综合久久蝌蚪 | 精品国产a | 国产高清一级 | 少妇激情久久 | 欧美一级免费在线 | 亚洲欧美日韩国产一区二区 | av一级黄| 97国产在线视频 | 日韩aⅴ视频 | 亚洲国产中文字幕 | 91九色综合| 啪啪动态视频 | 色美女在线 | 狠狠操天天射 | 国产成人三级一区二区在线观看一 | 久久久一本精品99久久精品 | 国产精品1区2区 | 日韩系列在线 | 久久av网| 中文字幕丝袜制服 | 玖玖视频精品 | 亚洲欧美日韩精品久久久 | 色久天| 五月天激情视频在线观看 | 久久艹影院 | 久久久久久久久久久免费视频 | 亚洲精品国产麻豆 | 欧美激情精品久久久久久变态 | 欧美色图88 | 国产精品涩涩屋www在线观看 | 日本性生活一级片 | 久久精品日本啪啪涩涩 | 久久久精品二区 | 久草在线视频首页 | 色www免费视频 | 九九久久国产精品 | 福利视频 | 美女视频黄是免费的 | 337p西西人体大胆瓣开下部 | 亚洲 中文 欧美 日韩vr 在线 | 欧美一区二区在线 | 国内99视频 | 精品亚洲二区 | 在线观看福利网站 | 奇米影视四色8888 | 91香蕉视频好色先生 | 亚洲国产合集 | av中文字幕在线播放 | 欧美大jb| 91一区二区三区在线观看 | 国产美女视频免费观看的网站 | 欧美精品一区二区在线观看 | 成人黄色免费在线观看 | 天天干天天做天天操 | 国内精品久久久久影院日本资源 | 久久草在线视频国产 | 久久国产精品小视频 | 国产福利小视频在线 | 免费观看成人 | 午夜视频免费在线观看 | 国产中文字幕视频在线观看 | 一区二区三区四区精品视频 | 91成年视频 | 久久极品 | 免费观看av| 亚洲永久精品一区 | 亚洲一区精品二人人爽久久 | 在线观看完整版免费 | 欧美成人日韩 | 日韩高清www | 久久99热国产 | 国产自在线观看 | 亚洲国内精品 | 日韩精品视频免费在线观看 | 亚洲一二三在线 | 999国产| av视屏在线 | 99在线视频免费观看 | 日韩成人免费在线 | 天天插天天干 | 一区二区视频在线播放 | 99久久日韩精品免费热麻豆美女 | 四虎成人精品在永久免费 | 久久免费视频这里只有精品 | 天天操天天操天天操天天 | 免费av小说| 五月婷在线播放 | 韩国精品视频在线观看 | 女女av在线 | 成人一级免费视频 | 国产精品一区二区免费 | 一区二区在线不卡 | 亚洲五月花 | 日日摸日日 | 69精品在线 | 日韩三级中文字幕 | 国产亚洲精品久久久久久大师 | 久久兔费看a级 | 91成人黄色 | 午夜三级大片 | 夜夜操网| 99精品在线免费在线观看 | 在线免费观看黄色大片 | 国产理论一区二区三区 | 97国产精品一区二区 | 亚洲成色777777在线观看影院 | 黄色综合 | av看片在线| 欧美九九九| 热久久电影 | 最新日韩在线观看视频 | 欧美黑人性猛交 | 超碰人人超| 久久久久综合精品福利啪啪 | 久久激情婷婷 | 在线精品亚洲一区二区 | av在线免费在线观看 | 日韩电影在线看 | 欧美精品免费视频 | 色综合天天爱 | 91精品国产入口 | 免费色网 | 成年人国产在线观看 | 一级黄色片在线免费看 | 日韩午夜网站 | 婷婷在线播放 | 视频在线观看亚洲 | 欧美最猛性xxxx | 亚洲欧美日本一区二区三区 | 国产色网站 | 久久久九九 | 四虎国产精 | 狠狠色丁香久久婷婷综合丁香 | 午夜黄色一级片 | 国产精品一区二区麻豆 | 国产黄色一级片 | 在线欧美最极品的av | 午夜美女wwww | 日韩在线观看不卡 | 99国产视频在线 | 久久久亚洲成人 | 天天干天天操天天干 | 日韩中文在线电影 | 天天射天天搞 | 精品国产乱码一区二区三区在线 | 日本高清中文字幕有码在线 | 成人免费网站视频 | 欧美极品xxx | 亚洲欧美视频在线 | 国产手机av在线 | 爱爱一区 | 免费在线观看av网站 | 超碰人人99 | 久久综合欧美精品亚洲一区 | 成人av网址大全 | 在线免费观看av网站 | 国产一区在线播放 | 五月视频| 超碰成人网 | 蜜桃视频在线视频 | 亚洲免费在线观看视频 | 免费亚洲视频在线观看 | 欧美在线99 | 国产精品国产三级国产专区53 | 欧美激情视频一区二区三区免费 | 9i看片成人免费看片 | 久久久久久久久久网 | 中文字幕乱在线伦视频中文字幕乱码在线 | 涩涩资源网 | 亚洲综合狠狠干 | 国产精品视频免费观看 | 久久精品黄 | 中文字幕视频观看 | 久久久精品 一区二区三区 国产99视频在线观看 | 欧美日韩大片在线观看 | 日韩av网站在线播放 | 久久在线 | 又黄又爽的免费高潮视频 | 欧美激情视频一二区 | av片免费播放 | 狂野欧美激情性xxxx | 国产精品99久久久久久宅男 | 久久亚洲综合国产精品99麻豆的功能介绍 | 天天爽天天摸 | 国产精品免费久久久久 | 久久99久久99精品 | 91在线视频免费播放 | 久久久久久高潮国产精品视 | 亚洲mv大片欧洲mv大片免费 | 成年人免费在线播放 | 伊人官网| 国产黄色一级片在线 | 美女免费视频一区 | 亚洲男模gay裸体gay | 国产精品毛片久久 | 最近日本字幕mv免费观看在线 | 日韩中文在线电影 | 黄网av在线 | 91国内在线 | 亚洲影视资源 | 天天射成人 | 国产精品一区二区av麻豆 | 美女视频黄网站 | 欧美特一级 | 日韩视频在线播放 | 国产精品 国内视频 | 欧美激情视频一区二区三区免费 | 国产一区免费在线 | 在线观看蜜桃视频 | 久草在线免费播放 | 又黄又爽又湿又无遮挡的在线视频 | h动漫中文字幕 | 久草剧场 | 视频一区二区三区视频 | 97国产大学生情侣白嫩酒店 | 日韩一级黄色大片 | 97精品免费视频 | 天天操夜夜爱 | 天天透天天插 | 九九免费在线观看视频 | 天天射天天射 | 91成人精品一区在线播放69 | 婷婷丁香花五月天 | 欧美一区二区三区在线 | 久草观看 | 日韩理论电影在线观看 | 日韩在线视频播放 | 在线播放国产一区二区三区 | 国产va饥渴难耐女保洁员在线观看 | 国产69精品久久久久久 | 久久久久婷 | 在线观看国产日韩 | 国产不卡av在线播放 | 最新婷婷色 | 青青河边草免费视频 | 国产精品久久久久久久久久久久久 | 免费看的黄网站软件 | 99久久精品国产欧美主题曲 | 国产一线二线三线性视频 | 日韩免费在线视频观看 | 欧美成人影音 | 国产成人精品一区二 | 91人人干 | 91大片成人网 | 国产精品福利在线观看 | 99热精品国产一区二区在线观看 | 91免费在线看片 | mm1313亚洲精品国产 | 国产精品自产拍在线观看桃花 | 亚洲精品理论 | 99这里只有精品视频 | 99视频这里只有 | 亚洲区精品 | 99热官网 | 一区二区三区免费在线观看视频 | 国产福利不卡视频 | 久久久久久久久久久免费视频 | 五月天丁香综合 | 久久久网页| av在线激情 | 亚洲激色 | 色播五月激情五月 | 国产精品久久久久久婷婷天堂 | 欧美色图狠狠干 | 成人免费看视频 | 日韩一级片观看 | www.神马久久 | 日本久久久久久科技有限公司 | 免费网址你懂的 | 精品99在线观看 | 精品国产一区二区三区在线观看 | 美女久久久 | 色综合久久99 | 午夜久久影视 | 视频直播国产精品 | 在线亚洲欧美视频 | 国产大片黄色 | 超碰日韩 | 人人讲下载 | 性色在线视频 | 久久精品1区2区 | 日韩在线不卡av | 91麻豆产精品久久久久久 | 国产在线91在线电影 | 91久久人澡人人添人人爽欧美 | 亚洲撸撸 | 日本高清中文字幕有码在线 | 天堂中文在线视频 | 中文字幕视频网站 | 高清av在线免费观看 | 日韩高清无线码2023 | 精品一区二区免费 | 精品99在线视频 | 九草在线视频 | 国产亚洲精品xxoo | 国产精品白丝jk白祙 | 97国产情侣爱久久免费观看 | 亚洲精品小区久久久久久 | 麻豆系列在线观看 | 亚洲黄色一级视频 | 91免费在线看片 | av7777777 | 在线免费av网站 | 精品久久久久一区二区国产 | 黄色成年片 | wwwww.国产 | 中文字幕av日韩 | 久久免费视频在线观看6 | 久草视频免费看 | 亚洲精品xxxx | 国产专区欧美专区 | 国产一区av在线 | 天天射天天爱天天干 | 国产韩国精品一区二区三区 | 欧美一区二区三区四区夜夜大片 | 亚洲国产视频网站 | 成年人电影免费在线观看 | 成人网在线免费视频 | 国产精品乱码一区二三区 | 92国产精品久久久久首页 | 成人在线免费看视频 | 久久99精品久久久久久 | 丁香激情综合国产 | 国产精品免费久久 | 免费进去里的视频 | 成年免费在线视频 | 丁香九月婷婷综合 | 深爱婷婷久久综合 | 亚洲天堂色婷婷 | 国产中文字幕视频在线观看 | 一区视频在线 | 成年人国产在线观看 | 国产精品永久免费视频 | 999久久久久久久久久久 | 国产又粗又猛又黄又爽视频 | 香蕉视频国产在线 | 人人舔人人爽 | 91精品国产麻豆 | 国产爽妇网| 久久久久国产精品免费 | 国内久久精品视频 | 色欲综合视频天天天 | 岛国一区在线 | 亚洲国产免费看 | 婷婷av网| 亚洲伊人第一页 | 蜜桃视频日韩 | 97精品在线观看 | 久久亚洲福利视频 | 999国内精品永久免费视频 | 国产亚洲精品久久久久秋 | www视频在线观看 | 在线中文日韩 | 亚洲老妇xxxxxx | 中文字幕一区二区在线播放 | www.五月婷 | 国内成人精品2018免费看 | 在线播放一区二区三区 | a一片一级 | 日韩欧美精品在线视频 | 在线观看一二三区 | 国产免费高清视频 | 五月天精品视频 | 国产午夜小视频 | 日b视频国产| 91av原创| 亚在线播放中文视频 | 中文字幕色在线视频 | 少妇bbbb搡bbbb桶 | 最新av在线网站 | 国精产品永久999 | 久久久久久国产一区二区三区 | aa级黄色大片 | 国产免费嫩草影院 | 欧美91精品久久久久国产性生爱 | 天天插日日射 | 婷婷综合成人 | 永久免费精品视频网站 | 在线观看你懂的网址 | 九九视频网站 | 久久欧美精品 | 日本久久久久久久久久 | 婷婷激情av | 日韩高清免费在线 | 在线91网| 国产无遮挡又黄又爽在线观看 | 99热这里精品 | 狠狠干五月天 | 天天射综合网站 | 狠狠躁夜夜a产精品视频 | 开心激情五月婷婷 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 亚洲日本黄色 | 99这里只有| 久久久麻豆| 激情视频91 | 国产乱视频 | 91黄色在线看 | 九九视频精品在线 | 亚洲精品乱码久久久久久蜜桃不爽 | 91视频传媒| 国产成人在线看 | 天天激情站 | 人人爽人人做 | 91尤物在线播放 | 在线视频18在线视频4k | 午夜电影av | 五月婷婷综合网 | 四虎影视精品 | 日韩经典一区二区三区 | 99久久er热在这里只有精品66 | 黄色av播放| 麻豆91在线播放 | 91爱爱网址 | 免费在线一区二区 | 玖玖在线播放 | 青草视频在线免费 | 欧美日韩视频在线 | 国产午夜在线 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 欧美一级久久久久 | av电影不卡在线 | av成人在线看 | 国产精品福利久久久 | 国产高清免费av | 国产91影院 | 日韩毛片在线一区二区毛片 | 色婷婷免费 | 天天摸天天舔 | 国产清纯在线 | 日韩高清免费在线观看 | 日韩午夜网站 | av电影免费在线 | 99久久精品国产毛片 | 日韩大片在线播放 | 久久影视网| 国产成人精品午夜在线播放 | 日一日干一干 | 国产成人在线免费观看 | 中文字幕在线网址 | 久久久免费播放 | 久久久精品视频网站 | 国产成人在线看 | 国产成人精品久久久久蜜臀 | 日本精品久久久久久 | 色婷婷五| 国产二区视频在线 | 一区二区视频电影在线观看 | 最新日韩在线 | 国产精品成人久久久久 | 手机看国产毛片 | 色小说av | 国产 中文 日韩 欧美 | 在线 国产一区 | 久久好看免费视频 | 久久999久久| 在线av资源 | 午夜在线观看一区 | 欧美 日韩 视频 | 久久国产精品久久久久 | 日韩在线视频免费播放 | 永久免费精品视频网站 | 午夜精品一区二区三区在线视频 | 久久久久日本精品一区二区三区 | 久久精品精品电影网 | 成人午夜影院 | 亚洲综合一区二区精品导航 | 99精品在线直播 | 九九九在线观看视频 | 久久论理 | 亚洲成人一区 | 国产精品原创av片国产免费 | 亚洲国产美女精品久久久久∴ | 91日韩国产 | 欧美美女一级片 | 免费www视频 | 国产99久久久欧美黑人 | 国产在线观看午夜 | 国产精品你懂的在线观看 | av一区二区在线观看中文字幕 | 欧美久久久影院 | 国产玖玖在线 | 97超碰色偷偷 | 亚洲 在线 | 91视频在线观看下载 | 伊人久久婷婷 | 中文字幕精品久久 | 91桃色在线免费观看 | 欧美日韩一级久久久久久免费看 | 免费看黄电影 | 亚洲日本韩国一区二区 | 午夜精品麻豆 | adn—256中文在线观看 | 日日色综合 | 69国产盗摄一区二区三区五区 | 精品99免费 | 在线最新av| 成人在线电影观看 | 又湿又紧又大又爽a视频国产 | 久草视频手机在线 | 国产在线国偷精品产拍免费yy | 99久久影院 | 久草在线免费看视频 | 久久久久久久久久久久久9999 | 欧美久久久久久久久久 | 六月色| 午夜视频在线观看一区二区 | 干av在线 | 亚洲欧美激情精品一区二区 | 麻豆视频在线免费看 | 午夜色性片 | 国产精品免费不卡 | 深爱激情五月综合 | 99久久夜色精品国产亚洲96 | 午夜久久久精品 | 美女性爽视频国产免费app | 成人免费视频播放 | 黄色视屏在线免费观看 | 成人国产精品久久久久久亚洲 | 日日麻批40分钟视频免费观看 | 视频在线观看一区 | 亚洲最大在线视频 | 国产欧美在线一区 | 国产视频一区在线免费观看 | 久久伊人精品天天 | 91人人射 | 国产精品久久人 | 亚洲成av人影院 | 人人要人人澡人人爽人人dvd | 国产精品久久久网站 | 狠狠的日| 成人国产亚洲 | 日韩中文字幕免费电影 | 亚洲精品九九 | 99中文字幕在线观看 | 国产精品美女 | 亚洲精品国产欧美在线观看 | 国产永久免费高清在线观看视频 | 国产a国产a国产a | 在线亚洲人成电影网站色www | 日韩免费高清在线观看 | 亚洲国产成人精品电影在线观看 | 亚洲色影爱久久精品 | 精品高清美女精品国产区 | 国产精品电影在线 | 久久激情五月丁香伊人 | 99精品国产高清在线观看 | 国产精品成人久久久久久久 | 天天操天天干天天爱 | 黄色一级网 |