【来源网络  http://www.2cto.com/kf/201502/376374.html】

一般大型网站我们登录的时候,密码忘了都有个功能可以找回密码。
细数下大致的方法:
1.直接把密码发送到你的邮箱去。一般是临时密码。 2.短信验证,成本较高。 3.密保问题 4.发送一个链接到你邮箱点击即可更改密码。
个人认为第四种方法最经济实惠,这次也主要都是在搞这个。

搞了一个晚上,单单邮件发送功能写了快300行,虽然很多是注释和空格,被舍友一说,用python只写了20几行,不禁膜拜PYTHON了!不过不管怎样写出来了,封装好了,下次要用就方便了。代码大部分是参考网上的和一些自己写的。

首先说下思路:
参考:http://blog.yidongzhifu.net/2014/03/07/邮箱找回密码功能的实现/
用户填写自己的邮箱时,需要查看该邮箱是否与用户ID绑定的邮箱想匹配,只有当匹配的时候才会发送邮件。 这封邮件中最重要的是一个链接地址: url =
baseUrl + “?uid=” + uid + “&validkey=” + validkey;
这个地址含有两个参数,id用户的id,validkey验证code,这是一个通过MD5加密过的字符串。
效验MD5就是用来确保文件在传输过程中未被修改用的,这个加密过的字符串应该包含用户的id+过期时间+随机数 validkey=md5(uid +
“|” + outdate + “|” + secretKey);

说白了,就是你要找回密码,你就得先输入你的帐号和邮箱。然后系统去判断,帐号和邮箱是匹配的。那么就在这一瞬间再在另外一张表写下信息(usrId,outdate,url)即你的ID,过期的时间(用现在的时间加上X分钟的有效期),URL即发送的链接。
发送的链接 = baseUrl + “?uid=” + uid + “&validkey=” + validkey;

比如说我在本地的地址是localhost:8080/homeSeller/resetPassword.jsp这个是我重置密码的页面地址。那我再在地址栏加上’?’
再在后面填写传入的属性与对应的值即可传值。 比如我的一次链接:
localhost:8080/homeSeller/resetPassword.jsp?uid=zhuang123&validkey=36B0F10812DE6D2B0D3B2DC044F9A27D
意思就是传入id,以及vaildkey vaildkey在之前我们已经写入数据库了!

填写成功后,利用JAVAMAIL发送邮件到指定邮箱。然后你点击那个链接,并且传值。这个时候就在JSP中判断一下对应userId的validkey是不是和数据库中的一样,以及currentTime是不是比outdate大即是否过期。如果都满足的话就跳转到更改密码的页面。更改密码就是简单的SQL,这里就不讲了。

  1. ok,下面讲下代码:
  2. 首先是
  3. 1)数据库层(DAO): 上面说的把信息插入数据库的代码(代码都是挺简单的增删改查
  4.  
  5. //找回密码,插入信息 ,这里的date是java.sql.Date
  6. public int insertInfor(Connection con,String userId,String email,Timestamp date,String signature) throws SQLException
  7. {
  8. String sql = insert into findPass(userId,email,outdate,signature) values(?,?,?,?);
  9. PreparedStatement pstmt = con.prepareStatement(sql);
  10. pstmt.setString(1, userId);
  11. pstmt.setString(2, email);
  12. pstmt.setTimestamp(3,date);
  13. pstmt.setString(4, signature);
  14.  
  15. int res = pstmt.executeUpdate();
  16. pstmt.close();
  17. con.close();
  18. return res;
  19. }
  20.  
  21. //找回密码,查询是否可以修改密码
  22. public boolean isChangePass(String userId,String validkey) throws Exception
  23. {
  24. DbUtil dbUtil = new DbUtil();
  25. Connection con = dbUtil.getCon();
  26. String sql = select * from findPass where userId = ?;
  27. PreparedStatement pstmt = con.prepareStatement(sql);
  28. pstmt.setString(1, userId);
  29. ResultSet res = pstmt.executeQuery();
  30. if(res.last())
  31. {
  32. String signature = res.getString(signature);
  33. if(!validkey.equals(signature)){
  34. pstmt.close();
  35. con.close();
  36. return false;
  37. }
  38. else{
  39. long current = System.currentTimeMillis();
  40. long time = res.getTimestamp(outdate).getTime();
  41. if(current> time){
  42. pstmt.close();
  43. con.close();
  44. return false;
  45. }
  46. else{
  47. pstmt.close();
  48. con.close();
  49. return true;
  50. }
  51. }
  52. }
  53. else{
  54. pstmt.close();
  55. con.close();
  56. return false;
  57. }
  58. }
  1. 2Servlet代码 就是根据思路对应的处理
  2.  
  3. public void doPost(HttpServletRequest request, HttpServletResponse response)
  4. throws ServletException, IOException {
  5.  
  6. //分割处理
  7. String method = request.getParameter(method);
  8.  
  9. if(method.equals(find)){
  10. String userId = request.getParameter(userId);
  11. String userEmail = request.getParameter(userEmail);
  12.  
  13. Connection con = null;
  14. try {
  15. con = dbUtil.getCon();
  16. } catch (Exception e) {
  17. // TODO Auto-generated catch block
  18. e.printStackTrace();
  19. }
  20.  
  21. boolean flag = false;
  22. try {
  23. flag = userDao.judgeUserEamil(userId, userEmail);
  24. } catch (Exception e1) {
  25. // TODO Auto-generated catch block
  26. e1.printStackTrace();
  27. }
  28.  
  29. if(flag){
  30. long currentTime = System.currentTimeMillis() + 120000;
  31. Date time = new Date(currentTime);
  32. Timestamp ts = new Timestamp(time.getTime());
  33. Random random = new Random();
  34. String key = userId + | + ts + | + random.nextInt();
  35. String signature = MD5Util.MD5(key);
  36.  
  37. try {
  38. int res = userDao.insertInfor(con, userId, userEmail, ts, signature);
  39. if(res==1)
  40. {
  41. SendMail sendmail = new SendMail();
  42. String url = localhost:8080/homeSeller/resetPassword.jsp+?uid= + userId + &validkey= + signature;
  43. sendmail.send(userEmail, url);
  44. }
  45. } catch (SQLException e) {
  46. // TODO Auto-generated catch block
  47. e.printStackTrace();
  48. } catch (AddressException e) {
  49. // TODO Auto-generated catch block
  50. e.printStackTrace();
  51. } catch (MessagingException e) {
  52. // TODO Auto-generated catch block
  53. e.printStackTrace();
  54. }
  55. }
  56. else
  57. {
  58. request.setAttribute(error, 用户名和邮箱不匹配,请重新输入!);
  59. }
  60. }
  61.  
  62. //重置密码
  63. else if (method.equals(reset)){
  64.  
  65. String userId = request.getParameter(userid);
  66. String password = request.getParameter(password1);
  67. try {
  68. Connection con = dbUtil.getCon();
  69. userDao.updatePassword(con, userId,password);
  70. request.setAttribute(error, 修改成功,请重新登录!);
  71. request.getRequestDispatcher(login.jsp).forward(request, response);
  72. } catch (Exception e) {
  73. // TODO Auto-generated catch block
  74. e.printStackTrace();
  75. }
  76.  
  77. }
  1. 3)工具类
  2. 工具类一共有两类 MD5加密,实现上面说的validkey的加密处理防止人工识别出来。
  3.  
  4. package com.homeSeller.util;
  5. import java.security.MessageDigest;
  6. import java.security.MessageDigest;
  7. public class MD5Util {
  8. public final static String MD5(String s) {
  9. char hexDigits[]={'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
  10. try {
  11. byte[] btInput = s.getBytes();
  12. // 获得MD5摘要算法的 MessageDigest 对象
  13. MessageDigest mdInst = MessageDigest.getInstance(MD5);
  14. // 使用指定的字节更新摘要
  15. mdInst.update(btInput);
  16. // 获得密文
  17. byte[] md = mdInst.digest();
  18. // 把密文转换成十六进制的字符串形式
  19. int j = md.length;
  20. char str[] = new char[j * 2];
  21. int k = 0;
  22. for (int i = 0; i < j; i++) {
  23. byte byte0 = md[i];
  24. str[k++] = hexDigits[byte0 >>> 4 & 0xf];
  25. str[k++] = hexDigits[byte0 & 0xf];
  26. }
  27. return new String(str);
  28. } catch (Exception e) {
  29. e.printStackTrace();
  30. return null;
  31. }
  32. }
  33. public static void main(String[] args) {
  34. System.out.println(MD5Util.MD5(20121221));
  35. System.out.println(MD5Util.MD5(加密));
  36. }
  37. }
  38.  
  39. 第二类是邮件发送类
  40. 参考http://www.cnblogs.com/codeplus/archive/2011/10/30/2229391.html JAVA MAIL是利用现有的邮件账户发送邮件的工具,比如说,我在网易注册一个邮箱账户,通过JAVA Mail的操控,我可以不亲自登录网易邮箱,让程序自动的使用网易邮箱发送邮件。这一机制被广泛的用在注册激活和垃圾邮件的发送等方面。 JavaMail可以到http://www.oracle.com/technetwork/java/javamail/index-138643.html进行下载,并将mail.jar添加到classpath即可。
  41.  
  42. JAVA邮件发送的大致过程是这样的的:
  43.  
  44. 1、构建一个继承自javax.mail.Authenticator的具体类,并重写里面的getPasswordAuthentication()方法。此类是用作登录校验的,以确保你对该邮箱有发送邮件的权利。
  45.  
  46. 2、构建一个properties文件,该文件中存放SMTP服务器地址等参数。
  47.  
  48. 3、通过构建的properties文件和javax.mail.Authenticator具体类来创建一个javax.mail.SessionSession的创建,就相当于登录邮箱一样。剩下的自然就是新建邮件。
  49.  
  50. 4、构建邮件内容,一般是javax.mail.internet.MimeMessage对象,并指定发送人,收信人,主题,内容等等。
  51.  
  52. 5、使用javax.mail.Transport工具类发送邮件。
  53.  
  54. 这里我参考写出来的邮件类是只支持SMTP 而不支持另外两种的,所以也没发时间去写工厂类了。不过SMTP应该就够了吧。 今天TXBUG发送不出去换了个邮箱就OK了。
  55. 1、首先是继承自javax.mail.Authenticator的一个具体类。getPasswordAuthentication()方法也就是构建一个PasswordAuthentication对象并返回,有点费解JAVA Mail这样的设计意图,可能是javax.mail.Authenticator为我们提供了附加的保证安全的验证措施吧。
  56.  
  57. package com.homeSeller.util;
  58. import javax.mail.Authenticator;
  59. import javax.mail.PasswordAuthentication;
  60.  
  61. public class MailAuthenticator extends Authenticator{
  62.  
  63. private String username;
  64.  
  65. private String password;
  66.  
  67. public MailAuthenticator(String username,String password)
  68. {
  69. this.username = username;
  70. this.password = password;
  71. }
  72.  
  73. String getPassword(){
  74. return password;
  75. }
  76.  
  77. @Override
  78. protected PasswordAuthentication getPasswordAuthentication(){
  79. return new PasswordAuthentication(username,password);
  80. }
  81.  
  82. String getUsername(){
  83. return username;
  84. }
  85.  
  86. public void setPassword(String password){
  87. this.password = password;
  88. }
  89.  
  90. public void setUsername(String username){
  91. this.username = username;
  92. }
  93. }
  1. 2、邮件发送类,剩下的步骤都是在这个类实现的。代码中的SimpleMail是封装了邮件主题和内容的一个POJO。觉得在一个方法参数中既包含主题又包含内容,不太合适,故重载了此方法。还有就是因为大多数邮箱的SMTP服务器地址都是可以通过邮箱地址算出来,简单起见,提供了一个不需要SMTP服务器地址的构造器。
  2.  
  3. package com.homeSeller.util;
  4. import java.util.List;
  5. import java.util.Properties;
  6.  
  7. import javax.mail.MessagingException;
  8. import javax.mail.Session;
  9. import javax.mail.Transport;
  10. import javax.mail.internet.AddressException;
  11. import javax.mail.internet.InternetAddress;
  12. import javax.mail.internet.MimeMessage;
  13. import javax.mail.internet.MimeMessage.RecipientType;
  14.  
  15. public class SimpleMailSender {
  16.  
  17. /*
  18. * 简单邮件发送器,可单发,群发
  19. */
  20.  
  21. /**
  22. *
  23. * 发送邮件的props文件
  24. */
  25.  
  26. private final transient Properties props = System.getProperties();
  27.  
  28. /*
  29. * 邮件服务器登录验证
  30. */
  31.  
  32. private transient MailAuthenticator authenticator;
  33.  
  34. /**
  35. * 邮箱session
  36. */
  37.  
  38. private transient Session session;
  39.  
  40. /**
  41. * 初始化邮件发送器
  42. *
  43. * @param smtpHostName
  44. * SMTP邮件服务器地址
  45. * @param username
  46. * 发送邮件的用户名(地址)
  47. * @param password
  48. * 发送邮件的密码
  49. */
  50.  
  51. public SimpleMailSender(final String smtpHostName,final String username,final String password)
  52. {
  53. init(username,password,smtpHostName);
  54. }
  55.  
  56. /**
  57. * 初始化邮件发送器
  58. *
  59. * @param username
  60. * 发送邮件的用户名(地址),并以此解析SMTP服务器地址
  61. * @param password
  62. * 发送邮件的密码
  63. *
  64. */
  65.  
  66. public SimpleMailSender(final String username,final String password){
  67. //通过邮箱地址解析出smtp服务器,对大多数邮箱都管用
  68. final String smtpHostName = smtp.+username.split(@)[1];
  69. init(username,password,smtpHostName);
  70. }
  71.  
  72. /**
  73. * 初始化
  74. *
  75. * @param username
  76. * 发送邮件的用户名(地址)
  77. * @param password
  78. * 密码
  79. * @param smtpHostName
  80. * SMTP主机地址
  81. */
  82.  
  83. private void init(String username,String password,String smtpHostName)
  84. {
  85. //初始化 props
  86.  
  87. props.put(mail.smtp.auth,true);
  88. props.put(mail.smtp.host,smtpHostName);
  89.  
  90. //验证
  91. authenticator = new MailAuthenticator(username,password);
  92.  
  93. //创建session
  94. session = Session.getInstance(props,authenticator);
  95. }
  96.  
  97. /**
  98. * 发送邮件
  99. *
  100. * @param recipient
  101. * 收件人邮箱地址
  102. * @param subject
  103. * 邮件主题
  104. * @param content
  105. * 邮件内容
  106. *
  107. * @throws AddressException
  108. * @throws MessagingException
  109. */
  110.  
  111. public void send(String recipient,String subject,Object content) throws AddressException,MessagingException{
  112.  
  113. //创建mime类型邮件
  114. final MimeMessage message = new MimeMessage(session);
  115.  
  116. //设置发信人
  117. message.setFrom(new InternetAddress(authenticator.getUsername()));
  118.  
  119. //设置收件人
  120.  
  121. message.setRecipient(RecipientType.TO,new InternetAddress(recipient));
  122.  
  123. //设置主题
  124. message.setSubject(subject);
  125.  
  126. //设置邮件内容
  127. message.setContent(content.toString(),text/html;charset=utf-8);
  128.  
  129. //发送
  130. Transport.send(message);
  131. }
  132.  
  133. /**
  134. *
  135. * 群发邮件
  136. *
  137. * @param recipients
  138. * 收件人们
  139. * @param subject
  140. * 主题
  141. * @param content
  142. * 内容
  143. * throws AddressException
  144. * throws MessagingException
  145. */
  146.  
  147. public void send(List recipients,String subject ,Object content) throws AddressException ,MessagingException{
  148.  
  149. //创建Mime类型邮件
  150. final MimeMessage message = new MimeMessage(session);
  151.  
  152. //设置发信人
  153. message.setFrom(new InternetAddress(authenticator.getUsername()));
  154.  
  155. //设置收信人们
  156. final int num = recipients.size();
  157. InternetAddress[] addresses = new InternetAddress[num];
  158.  
  159. for(int i=0;i recipients,SimpleMail mail) throws AddressException,MessagingException
  160. {
  161. send(recipients,mail.getSubject(),mail.getContent());
  162. }
  163. }
  164.  
  165. 3.POJOSimpleMail
  166.  
  167. package com.homeSeller.util;
  168. /*
  169. * SimpleMail
  170. * PROJ
  171. */
  172. public class SimpleMail {
  173. private String Content;
  174. private String Subject;
  175.  
  176. public String getContent() {
  177. return Content;
  178. }
  179. public void setContent(String Content) {
  180. this.Content = Content;
  181. }
  182. public String getSubject() {
  183. return Subject;
  184. }
  185. public void setSubject(String Subject) {
  186. this.Subject = Subject;
  187. }
  188.  
  189. }
  1. 4.最终的用来发送的类
  2.  
  3. package com.homeSeller.util;
  4. import java.util.List;
  5. import java.util.ArrayList;
  6.  
  7. import javax.mail.MessagingException;
  8. import javax.mail.internet.AddressException;
  9.  
  10. public class SendMail {
  11.  
  12. public void send(String email,String url) throws AddressException, MessagingException
  13. {
  14. SimpleMailSender sms = new SimpleMailSender(tianxia00115@163.com”,”password);
  15. String recipients = email;
  16. sms.send(recipients, HomeSeller找回密码,尊敬的HomeSeller用户,为了找回您的密码,请在两分钟之内点击以下连接:+url+ 如果不是您本人操作,请忽略此消息。);
  17. }
  18.  
  19. public static void main(String[] args) throws AddressException, MessagingException{
  20. SimpleMailSender sms = new SimpleMailSender(tianxia00115@163.com,383160100033);
  21. ArrayList recipients = new ArrayList();
  22. recipients.add(2867870421@qq.com);
  23.  
  24. for(String recipient:recipients){
  25. sms.send(recipients, test测试,hello hrwhisper.);
  26. }
  27. }
  28. }

到这里基本OK 所有东西串接起来,就很好的实现了密码找回了。
以前不懂原理,看到邮件发送来的都不知道是什么东西,现在懂了自然高兴,学习应该建立在这种不断学习不断满足的过程!

Jsp邮件找回密码全攻略的更多相关文章

  1. JDK - Tomcat - JSP - Servlet 配置运行全攻略(转)

    http://www.cnblogs.com/myqiao/archive/2005/08/29/225497.html 花了将近两个月的时间,从 JDK 开始一步一步摸索,历经千辛万苦,终于让第一个 ...

  2. JDK - Tomcat - Eclipse - JSP - Servlet 配置运行全攻略

    花了将近两个月的时间,从 JDK 开始一步一步摸索,历经千辛万苦,终于让第一个 Servlet 运行起来了,创建第一个 Servlet  程序确实要比创建第一个 Asp.net 程序困难多了,但是不要 ...

  3. [置顶] 创建GitHub技术博客全攻略

    [置顶] 创建GitHub技术博客全攻略 分类: GitHub2014-07-12 13:10 19710人阅读 评论(21) 收藏 举报 githubio技术博客网站生成 说明: 首先,你需要注册一 ...

  4. TestLink安装全攻略

    TestLink安装全攻略 此文章转自该链接--http://www.cnblogs.com/Tcorner/archive/2011/07/26/2117296.html 安装前准备 需要下载xam ...

  5. 取代奶瓶Minidwep-gtk破解WPA 全攻略

    取代奶瓶Minidwep-gtk 破 WPA 全攻略  目录 1. CDlinux 下使用 minidwepgtk 获取握手包并使用自带的字典破解 2. 自带的字典破解不出密码时使用 U 盘外挂字典继 ...

  6. 用C#制作PDF文件全攻略

    用C#制作PDF文件全攻略 目  录 前    言... 3 第一部分 iText的简单应用... 4 第一章 创建一个Document 4 第一步 创建一个Document实例:... 5 第二步 ...

  7. 【转】轻松搞定FTP之FlashFxp全攻略

    转载网址:http://www.newhua.com/2008/0603/39163.shtml 轻松搞定FTP之FlashFxp全攻略 导读: FlashFXP是一款功能强大的FXP/FTP软件,融 ...

  8. MyEclipse优化全攻略

    (0) 吐槽 Eclipse仅仅是个半成品有木有?什么都须要自己安装插件,新手非常难用有木有? 安装上插件以后了版本号兼容和各种问题烦死人有木有? 都怪碎片和版本号乱公布有木有? IntelliJ I ...

  9. 开发小白也毫无压力的hexo静态博客建站全攻略 - 躺坑后亲诉心路历程

    目录 基本原理 方法1 - 本机Windows下建站 (力荐) 下载安装node.js 用管理员权限打开命令行,安装hexo-cli和hexo 下载安装git 初始化hexo 使用hexo gener ...

随机推荐

  1. mongodb可视化客户端下载

    网站:https://www.mongodbmanager.com/files/ 直接下载:https://www.mongodbmanager.com/files/mongodbmanagerpro ...

  2. vcf2maf

    1.https://github.com/mskcc/vcf2maf 2.https://github.com/cbare/vcf2maf

  3. java 位数补0处理(转)

    数据库中表的ID为4位数字,如:0001,0012,0123,1234 如果插入的值不满足4位,需要进行转换,有两种方法 第一种方法以: int n = 1;         NumberFormat ...

  4. H5唤起app

    H5唤起app 1.判断是否在微信中打开 无论是在哪个平台的客户端Android/IOS,在微信的平台上访问都有一个问题,那就是无法启动客户端,这是微信为了安全性考虑的限制,android这边屏蔽sc ...

  5. 【转】Scikit-learn技巧(拓展)总结

    最近看了<Python数据挖掘入门与实战>,网上有说翻译地不好的,但是说实话,我觉得这本书还是相当不错的.作者Robert Layton是sklearn的开发者之一,书中介绍了很多skle ...

  6. 基于dapper的通用泛型分页

    1.定义一个用来装载适合所有类的分页结果类 public class PageDataView<T> { private int _TotalNum; public PageDataVie ...

  7. day27 CRM delete& action& 嵌入CRM

    课程目录:deleteactionpop up window嵌入crm项目 权限(未讲)学员交作业发邮件 代码路径:https://github.com/liyongsan/git_class/tre ...

  8. Element-UI 实现下拉树

    组件调用 <template> <!-- 行模式 --> <el-form inline> <el-form-item label="inline ...

  9. laravel中的validate验证的使用案例:

    第一个是设置,第二个是直接调用.

  10. 快速切题 poj 1002 487-3279 按规则处理 模拟 难度:0

    487-3279 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 247781   Accepted: 44015 Descr ...