• Jsp邮件找回密码全攻略


    【来源网络  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 (2)Servlet代码 就是根据思路对应的处理
     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 
    64 //重置密码
    65 else if (method.equals(reset)){
    66 
    67 String userId = request.getParameter(userid);
    68 String password = request.getParameter(password1);
    69 try {
    70 Connection con = dbUtil.getCon();
    71 userDao.updatePassword(con, userId,password);
    72 request.setAttribute(error, 修改成功,请重新登录!);
    73 request.getRequestDispatcher(login.jsp).forward(request, response);
    74 } catch (Exception e) {
    75 // TODO Auto-generated catch block
    76 e.printStackTrace();
    77 }
    78 
    79 
    80 }
      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 
     41 
     42 
     43 
     44 
     45 第二类是邮件发送类
     46 参考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即可。
     47 
     48 JAVA邮件发送的大致过程是这样的的:
     49 
     50 1、构建一个继承自javax.mail.Authenticator的具体类,并重写里面的getPasswordAuthentication()方法。此类是用作登录校验的,以确保你对该邮箱有发送邮件的权利。
     51 
     52 2、构建一个properties文件,该文件中存放SMTP服务器地址等参数。
     53 
     54 3、通过构建的properties文件和javax.mail.Authenticator具体类来创建一个javax.mail.Session。Session的创建,就相当于登录邮箱一样。剩下的自然就是新建邮件。
     55 
     56 4、构建邮件内容,一般是javax.mail.internet.MimeMessage对象,并指定发送人,收信人,主题,内容等等。
     57 
     58 5、使用javax.mail.Transport工具类发送邮件。
     59 
     60 
     61 
     62 这里我参考写出来的邮件类是只支持SMTP 而不支持另外两种的,所以也没发时间去写工厂类了。不过SMTP应该就够了吧。 今天TX有BUG发送不出去换了个邮箱就OK了。
     63 1、首先是继承自javax.mail.Authenticator的一个具体类。getPasswordAuthentication()方法也就是构建一个PasswordAuthentication对象并返回,有点费解JAVA Mail这样的设计意图,可能是javax.mail.Authenticator为我们提供了附加的保证安全的验证措施吧。
     64 
     65 package com.homeSeller.util;
     66 import javax.mail.Authenticator;
     67 import javax.mail.PasswordAuthentication;
     68 
     69 
     70 public class MailAuthenticator extends Authenticator{
     71 
     72 private String username;
     73 
     74 
     75 private String password;
     76 
     77 
     78 public MailAuthenticator(String username,String password)
     79 {
     80 this.username = username;
     81 this.password = password;
     82 }
     83 
     84 
     85 String getPassword(){
     86 return password;
     87 }
     88 
     89 @Override
     90 protected PasswordAuthentication getPasswordAuthentication(){
     91 return new PasswordAuthentication(username,password);
     92 }
     93 
     94 
     95 String getUsername(){
     96 return username;
     97 }
     98 
     99 
    100 public void setPassword(String password){
    101 this.password = password;
    102 }
    103 
    104 
    105 public void setUsername(String username){
    106 this.username = username;
    107 }
    108 }
      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 *
     24 * 发送邮件的props文件
     25 */
     26 
     27 
     28 private final transient Properties props = System.getProperties();
     29 
     30 /*
     31 * 邮件服务器登录验证
     32 */
     33 
     34 private transient MailAuthenticator authenticator;
     35 
     36 /**
     37 * 邮箱session
     38 */
     39 
     40 private transient Session session;
     41 
     42 /**
     43 * 初始化邮件发送器
     44 *
     45 * @param smtpHostName
     46 * SMTP邮件服务器地址
     47 * @param username
     48 * 发送邮件的用户名(地址)
     49 * @param password
     50 * 发送邮件的密码
     51 */
     52 
     53 public SimpleMailSender(final String smtpHostName,final String username,final String password)
     54 {
     55 init(username,password,smtpHostName);
     56 }
     57 
     58 
     59 /**
     60 * 初始化邮件发送器
     61 *
     62 * @param username
     63 * 发送邮件的用户名(地址),并以此解析SMTP服务器地址
     64 * @param password
     65 * 发送邮件的密码
     66 *
     67 */
     68 
     69 public SimpleMailSender(final String username,final String password){
     70 //通过邮箱地址解析出smtp服务器,对大多数邮箱都管用
     71 final String smtpHostName = smtp.+username.split(@)[1];
     72 init(username,password,smtpHostName);
     73 }
     74 
     75 
     76 /**
     77 * 初始化
     78 *
     79 * @param username
     80 * 发送邮件的用户名(地址)
     81 * @param password
     82 * 密码
     83 * @param smtpHostName
     84 * SMTP主机地址
     85 */
     86 
     87 private void init(String username,String password,String smtpHostName)
     88 {
     89 //初始化 props
     90 
     91 props.put(mail.smtp.auth,true);
     92 props.put(mail.smtp.host,smtpHostName);
     93 
     94 //验证
     95 authenticator = new MailAuthenticator(username,password);
     96 
     97 //创建session
     98 session = Session.getInstance(props,authenticator);
     99 }
    100 
    101 
    102 /**
    103 * 发送邮件
    104 *
    105 * @param recipient
    106 * 收件人邮箱地址
    107 * @param subject
    108 * 邮件主题
    109 * @param content
    110 * 邮件内容
    111 *
    112 * @throws AddressException
    113 * @throws MessagingException
    114 */
    115 
    116 public void send(String recipient,String subject,Object content) throws AddressException,MessagingException{
    117 
    118 //创建mime类型邮件
    119 final MimeMessage message = new MimeMessage(session);
    120 
    121 //设置发信人
    122 message.setFrom(new InternetAddress(authenticator.getUsername()));
    123 
    124 //设置收件人
    125 
    126 message.setRecipient(RecipientType.TO,new InternetAddress(recipient));
    127 
    128 //设置主题
    129 message.setSubject(subject);
    130 
    131 //设置邮件内容
    132 message.setContent(content.toString(),text/html;charset=utf-8);
    133 
    134 //发送
    135 Transport.send(message);
    136 }
    137 
    138 /**
    139 *
    140 * 群发邮件
    141 *
    142 * @param recipients
    143 * 收件人们
    144 * @param subject
    145 * 主题
    146 * @param content
    147 * 内容
    148 * throws AddressException
    149 * throws MessagingException
    150 */
    151 
    152 public void send(List recipients,String subject ,Object content) throws AddressException ,MessagingException{
    153 
    154 //创建Mime类型邮件
    155 final MimeMessage message = new MimeMessage(session);
    156 
    157 //设置发信人
    158 message.setFrom(new InternetAddress(authenticator.getUsername()));
    159 
    160 //设置收信人们
    161 final int num = recipients.size();
    162 InternetAddress[] addresses = new InternetAddress[num];
    163 
    164 for(int i=0;i recipients,SimpleMail mail) throws AddressException,MessagingException
    165 {
    166 send(recipients,mail.getSubject(),mail.getContent());
    167 }
    168 }
    169 
    170 
    171 
    172 
    173 
    174 3.POJO:SimpleMail
    175 
    176 package com.homeSeller.util;
    177 /*
    178  * SimpleMail
    179  * PROJ
    180  */
    181 public class SimpleMail {
    182 private String Content;
    183 private String Subject;
    184 
    185 
    186 public String getContent() {
    187 return Content;
    188 }
    189 public void setContent(String Content) {
    190 this.Content = Content;
    191 }
    192 public String getSubject() {
    193 return Subject;
    194 }
    195 public void setSubject(String Subject) {
    196 this.Subject = Subject;
    197 }
    198 
    199 }
     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 
    11 public class SendMail {
    12 
    13 
    14 public void send(String email,String url) throws AddressException, MessagingException
    15 {
    16 SimpleMailSender sms = new SimpleMailSender(tianxia00115@163.com”,”password);
    17 String recipients = email;
    18 sms.send(recipients, HomeSeller找回密码,尊敬的HomeSeller用户,为了找回您的密码,请在两分钟之内点击以下连接:+url+ 如果不是您本人操作,请忽略此消息。);
    19 }
    20 
    21 
    22 
    23 public static void main(String[] args) throws AddressException, MessagingException{
    24 SimpleMailSender sms = new SimpleMailSender(tianxia00115@163.com,383160100033);
    25 ArrayList recipients = new ArrayList();
    26 recipients.add(2867870421@qq.com);
    27 
    28 for(String recipient:recipients){
    29 sms.send(recipients, test测试,hello hrwhisper.);
    30 }
    31 }
    32 }

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

  • 相关阅读:
    针对当前用户 对方法加锁
    二叉树
    return与析构续
    return与析构函数
    拷贝构造函数
    引用
    返回引用的函数值和参数的引用
    使用引用返回函数值
    邻接表--图 部分代码
    dec,hex and oct
  • 原文地址:https://www.cnblogs.com/dixinyunpan/p/6041276.html
Copyright © 2020-2023  润新知