忘记密码 (通过邮箱来进行重新设置密码)

忘记密码 (通过邮箱来进行重新设置密码)

*总的流程

1.点击忘记密码

image-20220327195530868

页面的代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>登录页面</title>
</head>
<body>

<div class="reg_form">
    <form action="userServlet?type=1" method="post">
        <ul>

            <li>
                <label >用户名:</label><input type="text" name="uname">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <label >密码:</label><input type="password" name="pwd">
                <span class=""><i class=""></i></span>
            </li>

            <li>
                <a href="update.html">忘记密码</a>
            </li>

            <li>
                <input class="btn" type="submit" value="登录">
            </li>
        </ul>
    </form>
</div>
</body>
</html>

2.进入到身份验证环节(身份校验)

image-20220327195550998

一.页面的代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>忘记密码</title>
    <script src="demo/js/jquery-3.6.0.js"></script>
    <script src="demo/js/jquery.min.js"></script>
    <script>
        function sendEmail() {
            alert(12);
            var uname=$("#uname").val();
            var email=$("#email").val();
            $.ajax({
                url:"userServlet?type=11",
                type:"post",
                dataType:"json",
                data:{"uname":uname,"email":email},
                success:function (data) {
                    alert(data.msg);
                    if (data.istrue==1){
                        location.href="inputYzm.html";
                    }
                } ,
                error:function (data) {
                    alert(data.msg);
                    if (data.istrue==2){
                        location.href="update.html";
                    }
                }
            });
        }
    </script>
</head>
<body>
<div class="reg_form">
    <form >
        <ul>

            <li>
                <label >账号:</label><input type="text"  id="uname">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <label >邮箱:</label><input type="text"  id="email">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <input class="btn" type="button" onclick="sendEmail()" value="确认">
            </li>
        </ul>
    </form>
</div>
</body>
</html>

二.后台代码(servlet)

//单个查询+发送邮件 生成4位随机数字
    protected void selectOne(HttpServletRequest req, HttpServletResponse resp) throws Exception {

        PrintWriter out=resp.getWriter();
        String uname = req.getParameter("uname");
        System.out.println(uname);
        String email = req.getParameter("email");
        System.out.println(email);
        sys_users one = ss.selectOne(new sys_users(uname,email));//测试数据
        if(one!=null){
            System.out.println("cg");
            resultOne re=new resultOne();
            re.setIstrue(1);
            re.setMsg("该用户名邮箱下存在账号");
            String json= JSON.toJSONStringWithDateFormat(re, "yyyy-MM-dd");
            out.print(json);
            //验证身份成功之后 发送邮件给用户 验证码
            String pt = "1596481760@qq.com";
            String yzm22 = emailTest.getyzemail(email, pt).toString();
            req.getSession().setAttribute("yzm22",yzm22);

            System.out.println("单个查询成功存下的验证码"+yzm22);
        }else{
            System.out.println("sb");
            resultOne re=new resultOne();
            re.setIstrue(2);
            re.setMsg("该用户名邮箱下不存在账号");
            String json= JSON.toJSONStringWithDateFormat(re, "yyyy-MM-dd");
            out.print(json);
            //验证身份失败

        }
        out.flush();
        out.close();
    }

三.后台代码(serviceImpl)

 // 7: .单个查询 验证用户名  (写全单个查询 后面 还用的上)
    @Override
    public sys_users selectOne(sys_users u) {
        //编写单个查询的 sql语句
        String sql="select * from sys_users where uname='"+u.getUname()+"' and email='"+u.getEmail()+"'";
        sys_users users = sd.selectOne(sql);
        return users;
    }

四.后台代码(daoImpl)

  // 7: .单个查询 验证用户名  (写全单个查询 后面 还用的上)
    @Override
    public sys_users selectOne(String sql) {

        sys_users users=null;


        try {
            con= jdbcUtil.getConnection();
            ps=con.prepareStatement(sql);
            rs=ps.executeQuery();
            if (rs.next()){
                users=new sys_users(rs.getString("uname"));
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }

        return users;
    }

五.emailTest类中的 发送邮件的方法

public class emailTest {


    public static Integer getyzemail(String to,String from) {

        Integer random1 = 0;
        // 收件人电子邮箱


        // 发件人电子邮箱


        // 指定发送邮件的主机为 smtp.qq.com
        String host = "smtp.qq.com";  //QQ 邮件服务器

        // 获取系统属性
        Properties properties = System.getProperties();

        // 设置邮件服务器
        properties.setProperty("mail.smtp.host", host);

        properties.put("mail.smtp.auth", "true");
        // 获取默认session对象
        Session session = Session.getDefaultInstance(properties,new Authenticator(){
            public PasswordAuthentication getPasswordAuthentication()
            {
                //发件人邮件用户名、授权码【打开邮箱-设置-生成授权码==password】(自己的邮箱和授权码)
                return new PasswordAuthentication("xxxxxxx@qq.com", "uuekxrwqqyeffjde");
            }
        });

        try{
            // 创建默认的 MimeMessage 对象
            MimeMessage message = new MimeMessage(session);

            // Set From: 头部头字段
            message.setFrom(new InternetAddress(from));

            // Set To: 头部头字段
            message.addRecipient(Message.RecipientType.TO,
                    new InternetAddress(to));

            // Set Subject: 头部头字段
            message.setSubject("通知");

            //产生随机数

            random1 = (int) (Math.random()*(9999-1000+1))+1000;
            // 设置消息体
            message.setText("验证码是"+random1);

            // 发送消息
            Transport.send(message);

            System.out.println("发送成功");
        }catch (MessagingException mex) {
            mex.printStackTrace();
        }

        return random1;
    }


}

3.身份校验成功后(发送验证码)跳转到输入验证码页面

image-20220327195610282

输入验证码验证的页面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title></title>
    <script src="demo/js/jquery-3.6.0.js"></script>
    <script src="demo/js/jquery.min.js"></script>
    <script>
        function yzEmailyzm() {
            alert(22);
            var email=$("#email").val();
            $.ajax({
                url:"userServlet?type=10",
                type:"post",
                dataType:"json",
                data:{"email":email},
                success:function (data) {
                    alert(data.msg);
                    if (data.istrue==1){
                        location.href="hello.jsp";
                    }
                } ,
                error:function (data) {

                }
            });
        }
    </script>
</head>
<body>
<h1>请输入验证码</h1>
<div class="reg_form">
    <form >
        <ul>
            <li>
                <label >验证码:</label><input type="text" id="email">
                <span class=""><i class=""></i></span>
            </li>

            <li>
                <input class="btn" type="button" onclick="yzEmailyzm()" value="确认验证码">
            </li>
        </ul>
    </form>
</div>
        </body>
        </html>

输入验证码验证的后台

protected void yzemailyzm(HttpServletRequest req, HttpServletResponse resp) throws Exception {
        //自己输入验证码
        String yzm = req.getParameter("email");
        //自动生成的验证码
        String yzm2 = (String) req.getSession().getAttribute("yzm22");
        System.out.print(yzm + "==============" + yzm2);
        PrintWriter out = resp.getWriter();
        resultOne re = new resultOne();
        if (yzm.equals(yzm2)) {
            System.out.print("验证成功");
            //验证码验证成功 到修改页面 修改密码
            re.setIstrue(1);
            re.setMsg("验证码验证成功");
            String json = JSON.toJSONStringWithDateFormat(re, "yyyy-MM-dd");
            out.println(json);
        } else {
            System.out.print("验证失败");
        }
        out.flush();
        out.close();

    }

4.用户(通过邮箱收到)验证码后输入验证码

验证跳转到修改用户名的页面》》》》》》》

image-20220327195625104

进入页面后加载出单个查询的 账号和邮箱显示在页面上

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>修改密码 页面</title>
    <script src="demo/js/jquery-3.6.0.js"></script>
    <script src="demo/js/jquery.min.js"></script>
    <script>
        function selectInfo() {
            //通过ajax去后台查询数据
                $.ajax({
                    url:"userServlet?type=14",
                    dataType:"json",
                    type:"post",
                   // data:{},
                    success:function (data) {
                        $("#uname").val(data.uname);
                        $("#email").val(data.email);
                    } ,
                    error:function (data) {

                    }
                });
        }
    </script>
</head>
<body onload="selectInfo()">
<h1>修改密码页面</h1>
<div class="reg_form">
    <form>
        <ul>

            <li>
                <label>账号:</label><input type="text" id="uname">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <label>邮箱:</label><input type="text" id="email">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <label>密码:</label><input type="text" id="pwd">
                <span class=""><i class=""></i></span>
            </li>
            <li>
                <input class="btn" type="button" onclick="updatePwd()" value="确认">
            </li>
        </ul>
    </form>
</div>
</body>
</html>

显示在页面上的 方法

protected void one_infoName(HttpServletRequest req, HttpServletResponse resp) throws Exception {
        PrintWriter out=resp.getWriter();
        resultOne re=new resultOne();
        String selectone_uname = (String)req.getSession().getAttribute("selectone_uname");
        System.out.println(selectone_uname);
        String selectone_email = (String)req.getSession().getAttribute("selectone_email");
        System.out.println(selectone_email);
        re.setUname(selectone_uname);
        re.setEmail(selectone_email);

        String json = JSON.toJSONStringWithDateFormat(re, "yyyy-MM-dd");
        out.println(json);
        out.flush();
        out.close();
    }

5.修改密码

<think>好的,用户的问题是如何在Django应用中实现通过注册时使用的邮箱找回忘记密码,也就是密码重置功能。我需要逐步思考这个问题,确保回答正确且符合Django的最佳实践。 首先,Django内置了用户认证系统,其中包含密码重置的功能。用户提到他们想通过邮箱重置密码,这应该涉及到Django的django.contrib.auth模块中的相关视图和配置。我需要回忆一下Django官方文档中关于密码重置的部分。 第一步是检查settings.py中的配置。必须确保已经启用了django.contrib.auth和django.contrib.contenttypes应用,这两个通常是默认包含的。然后需要确认邮箱配置是否正确,比如EMAIL_BACKEND、EMAIL_HOST、EMAIL_PORT等。用户可能需要使用控制台后端来测试,或者配置真实的SMTP服务。这时候需要提醒用户设置这些参数,否则发送邮件会失败。 接下来是URL配置。Django的auth应用提供了PasswordResetView、PasswordResetDoneView、PasswordResetConfirmView和PasswordResetCompleteView这几个视图,需要将这些URL包含到项目的urls.py中。例如,path('password-reset/', auth_views.PasswordResetView.as_view(), name='password_reset'),以及其他的相关路径。需要确保用户正确设置了这些URL,并且模板路径正确。 然后是模板文件的创建。Django的密码重置流程需要四个模板:password_reset_form.html(用于输入邮箱)、password_reset_done.html(提交后的提示页面)、password_reset_confirm.html(输入新密码的表单)和password_reset_complete.html(重置完成页面)。这些模板需要放在指定的目录下,通常是registration目录下。需要提醒用户创建这些模板,并按照Django的要求命名,同时内容可以自定义,但要包含必要的表单元素。 关于邮件内容,Django会自动生成带有重置链接的邮件。用户需要自定义邮件的主题和内容,可以通过覆盖默认的模板来实现。例如,创建password_reset_email.html和password_reset_subject.txt模板文件,确保链接正确指向PasswordResetConfirmView的URL,并包含uidb64和token参数。 测试流程也很重要。用户需要启动开发服务器,访问密码重置页面,输入注册邮箱,然后检查是否收到邮件,点击链接后是否能跳转到设置密码的页面,并成功更新密码。这里可能需要提醒用户检查邮箱配置是否正确,尤其是在生产环境中,需要使用真实的SMTP服务器而不是控制台后端。 此外,用户可能需要自定义视图或表单,例如使用自定义的邮件模板或者添加额外的验证步骤。这时候可以建议用户继承Django提供的视图类,重写相关方法来实现自定义逻辑。例如,自定义邮件主题或者发件人地址。 最后,需要确保用户的Django版本兼容这些功能,虽然密码重置功能在较旧的版本中也存在,但最好确认用户使用的是较新的稳定版本,以避免兼容性问题。同时,提醒用户在生产环境中配置安全的邮箱服务,并处理可能的错误情况,比如无效的token或用户不存在的情况。 总结下来,步骤应该是:配置邮箱设置→包含URL路由→创建模板→测试流程→自定义需求(可选)。同时,需要引用Django的官方文档作为参考,确保用户能够查阅更多细节。</think>在Django中实现通过邮箱重置密码的功能需要以下步骤,结合系统内置的认证模块和邮件服务配置: ### 一、配置邮箱服务 在`settings.py`中添加SMTP配置: ```python EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' # 生产环境 EMAIL_HOST = 'smtp.example.com' # 邮箱服务商地址 EMAIL_PORT = 587 EMAIL_USE_TLS = True EMAIL_HOST_USER = 'your_email@example.com' # 发件邮箱 EMAIL_HOST_PASSWORD = 'your_password' # 邮箱授权码 ``` ### 二、配置URL路由 在项目`urls.py`中添加认证系统路由: ```python from django.contrib.auth import views as auth_views urlpatterns = [ path('password-reset/', auth_views.PasswordResetView.as_view( template_name='registration/password_reset_form.html', email_template_name='registration/password_reset_email.html', subject_template_name='registration/password_reset_subject.txt' ), name='password_reset'), path('password-reset/done/', auth_views.PasswordResetDoneView.as_view( template_name='registration/password_reset_done.html' ), name='password_reset_done'), path('password-reset-confirm/<uidb64>/<token>/', auth_views.PasswordResetConfirmView.as_view( template_name='registration/password_reset_confirm.html' ), name='password_reset_confirm'), path('password-reset-complete/', auth_views.PasswordResetCompleteView.as_view( template_name='registration/password_reset_complete.html' ), name='password_reset_complete'), ] ``` ### 三、创建模板文件 1. `templates/registration/password_reset_form.html`(邮箱输入表单): ```html <form method="post"> {% csrf_token %} {{ form.as_p }} <button type="submit">发送重置邮件</button> </form> ``` 2. `templates/registration/password_reset_email.html`(邮件内容模板): ```html {% autoescape off %} 您收到这封邮件是因为请求重置密码。请点击下方链接: {{ protocol }}://{{ domain }}{% url 'password_reset_confirm' uidb64=uid token=token %} {% endautoescape %} ``` ### 四、操作流程 1. 用户访问`/password-reset/`输入注册邮箱 2. 系统发送包含重置链接的邮件 3. 用户点击邮件链接进入密码重置页面 4. 用户输入新密码后完成重置 ### 五、自定义功能(可选) 在视图类中重写方法实现高级功能: ```python class CustomPasswordResetView(auth_views.PasswordResetView): def form_valid(self, form): # 添加自定义逻辑(如记录操作日志) return super().form_valid(form) ``` [^1]: Django官方文档建议使用`django.contrib.auth.views`中的预构建视图处理密码重置流程,可有效减少开发工作量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值