Java實現(xiàn)郵件找回密碼功能
本文實例為大家分享了Java實現(xiàn)郵件找回密碼功能的具體代碼,供大家參考,具體內(nèi)容如下
1、有個需求就是,忘記密碼后通過郵箱找回。現(xiàn)在的系統(tǒng)在注冊的時候都會強(qiáng)制輸入郵箱,其一目的就是 通過郵件綁定找回,可以進(jìn)行密碼找回。通過java發(fā)送郵件的功能我就不說了,重點講找回密碼。
2、參考別人的思路:發(fā)送郵件→請求郵件里的URL→驗證url→{驗證成功修改密碼,不成功跳轉(zhuǎn)到失敗頁面}
重點就是如何生成這個url和如何解析這個url.
需要注意的是一個url只能修改一次密碼,當(dāng)同一帳號發(fā)送多封郵件,只有最后一封郵件的url
3、加密能防止偽造攻擊,一次url只能驗證一次,并且綁定了用戶。生成url: 可以用UUID生成隨機(jī)密鑰。
數(shù)字簽名 = MD5(用戶名+'$'+過期時間+‘$'+密鑰key)
數(shù)據(jù)庫字段(用戶名(主鍵),密鑰key,過期時間)
url參數(shù)(用戶名,數(shù)字簽名) ,密鑰key的生成:在每一個用戶找回密碼時候為這個用戶生成一個密鑰key ,url example: http://localhost:8080/user/reset_password?sid=D622D6A23FBF86FFE696B593D55351A54AEAEA77&userName=test4
生成過期時間,生成數(shù)字簽名,生成url,發(fā)送郵件.saveOrUpdate(用戶名,密鑰key,過期時間)
以下為springMvc代碼
@RequestMapping(value = "/user/i_forget_password")
@ResponseBody
public
Map forgetPass(HttpServletRequest request,String userName){
Users users = userService.findUserByName(userName);
Map map = new
HashMap<String ,String >();
String msg = "";
if(users == null){ //用戶名不存在
msg = "用戶名不存在,你不會忘記用戶名了吧?";
map.put("msg",msg);
return
map;
}
try{
String secretKey= UUID.randomUUID().toString(); //密鑰
Timestamp outDate = new
Timestamp(System.currentTimeMillis()+30*60*1000);//30分鐘后過期
long
date = outDate.getTime()/1000*1000; //忽略毫秒數(shù)
users.setValidataCode(secretKey);
users.setRegisterDate(outDate);
userService.update(users); //保存到數(shù)據(jù)庫
String key = users.getUserName()+"$"+date+"$"+secretKey;
String digitalSignature = MD5.MD5Encode(key); //數(shù)字簽名
String emailTitle = "有方云密碼找回";
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
String resetPassHref = basePath+"user/reset_password?sid="+digitalSignature+"&userName="+users.getUserName();
String emailContent = "請勿回復(fù)本郵件.點擊下面的鏈接,重設(shè)密碼<br/><a href="+resetPassHref +" rel="external nofollow" target='_BLANK'>點擊我重新設(shè)置密碼</a>" +
"<br/>tips:本郵件超過30分鐘,鏈接將會失效,需要重新申請'找回密碼'"+key+"\t"+digitalSignature;
System.out.print(resetPassHref);
SendMail.getInstatnce().sendHtmlMail(emailTitle,emailContent,users.getEmail());
msg = "操作成功,已經(jīng)發(fā)送找回密碼鏈接到您郵箱。請在30分鐘內(nèi)重置密碼";
logInfo(request,userName,"申請找回密碼");
}catch
(Exception e){
e.printStackTrace();
msg="郵箱不存在?未知錯誤,聯(lián)系管理員吧。";
}
map.put("msg",msg);
return
map;
}
找回鏈接已經(jīng)發(fā)到郵箱了。進(jìn)入郵箱點開鏈接
以下為鏈接檢驗代碼,驗證通過 跳轉(zhuǎn)到修改密碼界面,否則跳轉(zhuǎn)到失敗界面
@RequestMapping(value = "/user/reset_password",method = RequestMethod.GET)
public
ModelAndView checkResetLink(String sid,String userName){
ModelAndView model = new
ModelAndView("error");
String msg = "";
if(sid.equals("") || userName.equals("")){
msg="鏈接不完整,請重新生成";
model.addObject("msg",msg) ;
logInfo(userName,"找回密碼鏈接失效");
return
model;
}
Users users = userService.findUserByName(userName);
if(users == null){
msg = "鏈接錯誤,無法找到匹配用戶,請重新申請找回密碼.";
model.addObject("msg",msg) ;
logInfo(userName,"找回密碼鏈接失效");
return
model;
}
Timestamp outDate = users.getRegisterDate();
if(outDate.getTime() <= System.currentTimeMillis()){ //表示已經(jīng)過期
msg = "鏈接已經(jīng)過期,請重新申請找回密碼.";
model.addObject("msg",msg) ;
logInfo(userName,"找回密碼鏈接失效");
return
model;
}
String key = users.getUserName()+"$"+outDate.getTime()/1000*1000+"$"+users.getValidataCode(); //數(shù)字簽名
String digitalSignature = MD5.MD5Encode(key);
System.out.println(key+"\t"+digitalSignature);
if(!digitalSignature.equals(sid)) {
msg = "鏈接不正確,是否已經(jīng)過期了?重新申請吧";
model.addObject("msg",msg) ;
logInfo(userName,"找回密碼鏈接失效");
return
model;
}
model.setViewName("user/reset_password"); //返回到修改密碼的界面
model.addObject("userName",userName);
return
model;
}
補(bǔ)充1:Timestamp類型對象在保存到數(shù)據(jù)的時候 毫秒精度會丟失。比如:2013-10-08 10:29:10.234 存到mysql數(shù)據(jù)庫的時候 變成 2013-10-08 10:29:10.0。時間變得不相同了,sid 匹配的時候不會相等。 所以我做了忽略精度的操作。
補(bǔ)充2:解決linux下面title中文亂碼
sun.misc.BASE64Encoder enc = new sun.misc.BASE64Encoder(); mailMessage.setSubject(MimeUtility.encodeText(mailInfo.getSubject(), "UTF-8", "B")); //解決linux郵件title亂碼
補(bǔ)充3:怎么不直接把sid插入到user表呢。驗證的時候直接比較sid就ok了。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java Runtime類詳解_動力節(jié)點Java學(xué)院整理
Runtime類封裝了運行時的環(huán)境。每個 Java 應(yīng)用程序都有一個 Runtime 類實例,使應(yīng)用程序能夠與其運行的環(huán)境相連接。下面通過本文給大家分享Java Runtime類詳解,需要的朋友參考下吧2017-04-04
Springboot整合阿里巴巴SMS的實現(xiàn)示例
本文主要介紹了Springboot整合阿里巴巴SMS的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-12-12
Mybatis-Plus集成Sharding-JDBC與Flyway實現(xiàn)多租戶分庫分表實戰(zhàn)
這篇文章主要為大家介紹了Mybatis-Plus集成Sharding-JDBC與Flyway實現(xiàn)多租戶分庫分表實戰(zhàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
Java的Hibernate框架中Criteria查詢使用的實例講解
這篇文章主要介紹了Java的Hibernate框架中Criteria查詢使用的實例講解,Hibernate是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2016-01-01
springboot中的controller參數(shù)映射問題小結(jié)
這篇文章主要介紹了springboot中的controller參數(shù)映射問題小結(jié),本文通過實例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2024-12-12

