欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

shiro多驗(yàn)證登錄代碼怎么編寫(xiě)

今天就跟大家聊聊有關(guān)shiro多驗(yàn)證登錄代碼怎么編寫(xiě),可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

讓客戶滿意是我們工作的目標(biāo),不斷超越客戶的期望值來(lái)自于我們對(duì)這個(gè)行業(yè)的熱愛(ài)。我們立志把好的技術(shù)通過(guò)有效、簡(jiǎn)單的方式提供給客戶,將通過(guò)不懈努力成為客戶在信息化領(lǐng)域值得信任、有價(jià)值的長(zhǎng)期合作伙伴,公司提供的服務(wù)項(xiàng)目有:域名與空間、網(wǎng)頁(yè)空間、營(yíng)銷(xiāo)軟件、網(wǎng)站建設(shè)、神農(nóng)架林區(qū)網(wǎng)站維護(hù)、網(wǎng)站推廣。

1. 首先新建一個(gè)shiroConfig shiro的配置類(lèi),代碼如下:

@Configuration是標(biāo)識(shí)這個(gè)類(lèi)是一個(gè)配置文件,在啟動(dòng)時(shí)會(huì)加載這個(gè)類(lèi)里面的內(nèi)容,這個(gè)配置文件的位置的一定一定一定不能防止啟動(dòng)類(lèi)外面的文件夾中,否則還會(huì)在啟動(dòng)類(lèi)上加注解

@Bean是將這個(gè)類(lèi)交給spring管理

@Configurationpublic class SpringShiroConfig {  /**   * @param realms 這兒使用接口集合是為了實(shí)現(xiàn)多驗(yàn)證登錄時(shí)使用的   * @return   */  @Bean  public SecurityManager securityManager(Collection<Realm> realms) {    DefaultWebSecurityManager sManager = new DefaultWebSecurityManager();    sManager.setRealms(realms);    return sManager;  }  @Bean  public ShiroFilterFactoryBean shiroFilterFactory(SecurityManager securityManager) {    ShiroFilterFactoryBean sfBean = new ShiroFilterFactoryBean();    sfBean.setSecurityManager(securityManager);    //如果是匿名訪問(wèn)時(shí),訪問(wèn)了不能訪問(wèn)的資源跳轉(zhuǎn)的位置    sfBean.setLoginUrl("/index");    //定義map指定請(qǐng)求過(guò)濾規(guī)則(哪些資源允許匿名訪問(wèn),哪些必須認(rèn)證訪問(wèn))    LinkedHashMap<String, String> map = new LinkedHashMap<>();    //靜態(tài)資源允許匿名訪問(wèn):"anon" 靜態(tài)資源授權(quán)時(shí)不能寫(xiě)static下面所有的開(kāi)放,要將static下面的所有文件夾一個(gè)一個(gè)的開(kāi)放,templates同理    //map的key可以為文件的位置,也可以為請(qǐng)求的路徑    map.put("/bower_components/**", "anon");    map.put("/json/**", "anon");    map.put("/pages", "anon");    map.put("/user/userPasswordLogin", "anon");    map.put("/user/login", "anon");    map.put("/user/reg", "anon");    //訪問(wèn)這個(gè)路徑時(shí)不會(huì)進(jìn)入controller,會(huì)在這兒直接攔截退出,問(wèn)為什么的,自己想請(qǐng)求流程去    map.put("/user/userLogout", "logout");    //攔截除上面之外的所有請(qǐng)求路徑    map.put("/**", "user");    sfBean.setFilterChainDefinitionMap(map);    return sfBean;  }  @Bean  public LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {    return new LifecycleBeanPostProcessor();  }

2. 寫(xiě)Realms的實(shí)現(xiàn)類(lèi),一般繼承自AuthorizingRealm(這個(gè)是實(shí)現(xiàn)用戶名,密碼登錄),代碼如下:

@Servicepublic class ShioUserRealm extends AuthorizingRealm {  //注入userdao  @Autowired  private UserDao userDao;  /**   * 設(shè)置憑證匹配器   *   * @param credentialsMatcher   */  @Override  public void setCredentialsMatcher(CredentialsMatcher credentialsMatcher) {    /*這里設(shè)置了MD5鹽值加密,這兒就必須使用HashedCredentialsMatcher才能有下面兩個(gè)方法*/    HashedCredentialsMatcher matcher = new HashedCredentialsMatcher();    //這里是設(shè)置加密方式    matcher.setHashAlgorithmName("MD5");    //這里是設(shè)置加密的次數(shù)    matcher.setHashIterations(2);    super.setCredentialsMatcher(matcher);  }  /**   * 這兒是設(shè)置授權(quán)的   * @param principalCollection   * @return   */  @Override  protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principalCollection) {    return null;  }  /**   * 通過(guò)此方法完成認(rèn)證數(shù)據(jù)的獲取及封裝,系統(tǒng)底層會(huì)將認(rèn)證數(shù)據(jù)傳遞認(rèn)證管理器,有認(rèn)證管理器完成認(rèn)證操作   * @param authenticationToken   * @return   * @throws AuthenticationException   */  @Override  protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {    //先判斷這個(gè)是否是來(lái)及這個(gè)令牌的數(shù)據(jù):我們這兒分為了UsernamePasswordToken(shiro給我們提供的。)、UserPhoneToken    if (!(authenticationToken instanceof UsernamePasswordToken)) {      return null;    }    //獲取controller傳過(guò)來(lái)的數(shù)據(jù)    UsernamePasswordToken upToken = (UsernamePasswordToken) authenticationToken;    //upToken.setRememberMe(true);shiro默認(rèn)為false,是是否記住我的功能    //這兒為用戶提交的username    String username = upToken.getUsername();    //去數(shù)據(jù)更加name取到用戶的信息    User user = userDao.findUserByUserName(username);    //判斷數(shù)據(jù)庫(kù)是否有這用戶    if (user == null) {      throw new UnknownAccountException();    }    //判斷用戶的狀態(tài)是否被禁用(數(shù)據(jù)庫(kù)的字段)    if (user.getState() == 0) {      throw new LockedAccountException();    }    //這兒是取到用戶信息中的鹽值,鹽值要轉(zhuǎn)換為ByteSource這個(gè)類(lèi)型才能使用    ByteSource credentialsSalt = ByteSource.Util.bytes(user.getSalt());    //這兒是將這個(gè)用戶的信息交給shiro(user為用戶對(duì)象,user.getPassword()是要加密的對(duì)象,credentialsSalt為鹽值,getName()當(dāng)前對(duì)象)    SimpleAuthenticationInfo info = new SimpleAuthenticationInfo(user, user.getPassword(), credentialsSalt, getName());    return info;  }}

3. 此時(shí)用戶的賬號(hào)密碼登錄已經(jīng)可以使用了controller代碼如下:

@RequestMapping("userPasswordLogin")  @ResponseBody  public JsonResult userPasswordLogin(String username, String password) {    Subject subject = SecurityUtils.getSubject();    UsernamePasswordToken token = new UsernamePasswordToken(username, password);    subject.login(token);    return new JsonResult("login Ok");  }

4. 我們現(xiàn)在來(lái)實(shí)現(xiàn)短信驗(yàn)證碼登錄實(shí)現(xiàn):

4.1 先寫(xiě)UserPhoneToken,我放在l和springShiroConfig同一目錄下:

@Componentpublic class UserPhoneToken extends UsernamePasswordToken implements Serializable {  private static final long serialVersionUID = 6293390033867929958L;  // 手機(jī)號(hào)碼  private String phoneNum;  //無(wú)參構(gòu)造  public UserPhoneToken(){}    //獲取存入的值  @Override  public Object getPrincipal() {    if (phoneNum == null) {      return getUsername();    } else {      return getPhoneNum();    }  }  @Override  public Object getCredentials() {    if (phoneNum == null) {      return getPassword();    }else {      return "ok";    }  }  public UserPhoneToken(String phoneNum) {    this.phoneNum = phoneNum;  }  public UserPhoneToken(final String userName, final String password) {    super(userName, password);  }  public String getPhoneNum() {    return phoneNum;  }  public void setPhoneNum(String phoneNum) {    this.phoneNum = phoneNum;  }  @Override  public String toString() {    return "PhoneToken [PhoneNum=" + phoneNum + "]";  }}

4.2 在寫(xiě)shiroUserPhoneRealm,代碼如下:

@Servicepublic class ShioUserPhoneRealm extends AuthorizingRealm {  @Autowired  private UserDao userDao;  @Override  public void setCredentialsMatcher(CredentialsMatcher credentialsMatcher) {    //這兒的CredentialsMatcher的new的對(duì)象必須是AllowAllCredentialsMatcher    CredentialsMatcher matcher = new AllowAllCredentialsMatcher();    super.setCredentialsMatcher(matcher);  }  @Override  protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principalCollection) {    return null;  }  /**   * 通過(guò)此方法完成認(rèn)證數(shù)據(jù)的獲取及封裝,系統(tǒng)底層會(huì)將認(rèn)證數(shù)據(jù)傳遞認(rèn)證管理器,有認(rèn)證管理器完成認(rèn)證操作   * @param authenticationToken   * @return   * @throws AuthenticationException   */  @Override  protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {    UserPhoneToken token = null;    if (authenticationToken instanceof UserPhoneToken) {      token = (UserPhoneToken) authenticationToken;    }else {      return null;    }    //獲取我發(fā)送驗(yàn)證碼是存入session中的驗(yàn)證碼和手機(jī)號(hào)    String verificationCode = (String) SecurityUtils.getSubject().getSession().getAttribute("verificationCode");    String phone = (String) SecurityUtils.getSubject().getSession().getAttribute("phone");    //獲取controller傳過(guò)來(lái)的數(shù)據(jù)    String verificationCode1 = (String) token.getPrincipal();    //去數(shù)據(jù)庫(kù)根據(jù)手機(jī)號(hào)查詢(xún)用戶信息    User user = userDao.findUserByUserPhone(phone);    if (StringUtils.isEmpty(verificationCode)) {      throw new ServiceException("網(wǎng)絡(luò)錯(cuò)誤");    }    //比對(duì)手機(jī)號(hào)    if (!verificationCode.equals(verificationCode1)) {      throw new ServiceException("驗(yàn)證碼不正確");    }    if (user == null) {      throw new UnknownAccountException();    }    if (user.getState() == 0) {      throw new LockedAccountException();    }    return new SimpleAuthenticationInfo(user,phone,getName());  }}

4.3 手機(jī)號(hào)碼登錄驗(yàn)證已經(jīng)基本完成:controller代碼如下:

password為接收的驗(yàn)證碼

@PostMapping("verificationCodeLogin")  @ResponseBody  public JsonResult verificationCodeLogin(String password) {    Subject subject = SecurityUtils.getSubject();    UserPhoneToken token = new UserPhoneToken(password);    subject.login(token);    return new JsonResult("login OK");  }

使用過(guò)程中遇到的bug

1.

org.apache.shiro.authc.UnknownAccountException: Realm [cn.tedu.wxacs.service.impl.ShioUserPhoneRealm@768d8431] was unable to find account data for the submitted AuthenticationToken [org.apache.shiro.authc.UsernamePasswordToken - 張三, rememberMe=false].

出現(xiàn)這個(gè)問(wèn)題是我的是因?yàn)镽ealm中的某個(gè)實(shí)現(xiàn)類(lèi)沒(méi)有加注解,我這兒演示時(shí)是應(yīng)為ShiroUserRealm為加@Service注解

2.

org.apache.shiro.authc.AuthenticationException: Authentication token of type [class org.apache.shiro.authc.UsernamePasswordToken] could not be authenticated by any configured realms. Please ensure that at least one realm can authenticate these tokens.

這兒出現(xiàn)的問(wèn)題是應(yīng)為我的ShioUserRealm的AuthenticationInfo方法的User user = userDao.findUserByUserName(username);這行代碼出現(xiàn)的問(wèn)題,debug的時(shí)候就發(fā)現(xiàn)這一句執(zhí)行后就保錯(cuò)

原因:是因?yàn)槲业腶pplication.yml文件中沒(méi)有寫(xiě)dao對(duì)應(yīng)的mapper文件的路徑

3. 在ShioUserPhoneRealm的doGetAuthenticationInfo方法的new SimpleAuthenticationInfo(user,phone,getName())這個(gè)位置后就報(bào)錯(cuò)是應(yīng)為ShioUserPhoneRealm的這個(gè)方法中你沒(méi)有將new的對(duì)象設(shè)置為AllowAllCredentialsMatcher();

@Override  public void setCredentialsMatcher(CredentialsMatcher credentialsMatcher) {    //這兒的CredentialsMatcher的new的對(duì)象必須是AllowAllCredentialsMatcher    CredentialsMatcher matcher = new AllowAllCredentialsMatcher();    super.setCredentialsMatcher(matcher);  }

注解中有一些需要注意的地方,建議看看,注解不對(duì)的地方還希望在下放評(píng)論指出或者聯(lián)系我

應(yīng)為我的知識(shí)有限,此方法本人實(shí)現(xiàn)目前沒(méi)有問(wèn)題,其中有什么不對(duì)的地方還希望各位指出,謝謝!

使用的是jdk8,spring boot 的2.2.1版本,shiro的1,.4.1版本

看完上述內(nèi)容,你們對(duì)shiro多驗(yàn)證登錄代碼怎么編寫(xiě)有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝大家的支持。

分享文章:shiro多驗(yàn)證登錄代碼怎么編寫(xiě)
URL地址:http://aaarwkj.com/article12/gjjdgc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供服務(wù)器托管網(wǎng)站導(dǎo)航、微信公眾號(hào)、網(wǎng)頁(yè)設(shè)計(jì)公司定制網(wǎng)站、網(wǎng)站策劃

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

成都做網(wǎng)站
漂亮人妻中文字幕av| 永久免费观看黄色录像| 色偷偷亚洲精品一区二区| 操老熟女一区二区三区| 国产午夜三级视频在线观看| 97乱碰视频在线观看| 在线观看国产自拍精品| 日韩黄色大片免费在线观看| 日本不卡一区二区在线视频| 国产精品一区二区三区欧美| 中文字幕乱码高清欧美日韩| 日本韩国精品视频在线| 亚洲综合成人av在线| 91香蕉国产精品日韩| 91免费福利激情视频| 亚洲欧美一区二区三区日本| 人妻少妇久久中文字幕韩| 人妻少妇被猛烈进入久久精品| 91亚洲蜜桃内射后入在线观看 | 麻豆视频91免费观看| 日韩欧美高清一区二区| 国产在线观看一区二区三区精品| 亚洲成av人片青草影院| 熟女高潮av一区二区| 国产美女自拍视频一区| 日韩欧美一二区久久麻豆| 国产精品毛片在线看不卡| 午夜欧美日韩精品久久久| 男女爱爱视频网站久久精选| 国语对白精品视频在线| 欧美精品一区二区三区在线| 亚洲欧美国产日韩天堂区| 韩国av网址在线观看| 国产三级视频在线观看视频 | 精品一区中文字幕少妇人妻| 手机不卡高清播放一区二区| 精品一区二区三区毛卡片| 日韩精品视频播放一区 | 日本美女阴部毛茸茸视频| 免费在线观看av大全| 麻豆视频国产一区二区|