邮箱认证流程

在日常生活中,我们在一个网站中注册一个账户时,往往在提交个人信息后,网站还要我们通过手机或邮件来验证,邮件的话大概会是下面这个样子的:

用户通过点击链接从而完成注册,然后才能登录。

也许你会想,为什么要这么麻烦直接提交注册不就行了吗?这其中很大一部分原因是为了防止恶意注册。接下来让我们一起来使用最简单的JSP+Servlet的方式来完成一个通过邮箱验证注册的小案例吧。

准备工作

前提知识

动手实践之前,你最好对以下知识有所了解:

如果对邮件收发过程完全不了解的话,可以花三分钟的时间到慕课网了解一下,讲得算是非常清楚了,这里就不赘述了。放张图回忆一下:

邮件收发过程

邮箱准备

在了解的上述内容之后,要实现这个案例,首先我们还得有两个邮箱账号,一个用来发送邮件,一个用来接收邮件。本案例使用QQ邮箱向163邮箱发送激活邮件,因此需要登录QQ邮箱,在设置->账户面板中开启POP3/SMTP服务,以允许我们通过第三方客户端发送邮件:

还要注意的是,登录以下服务: POP3/IMAP/SMTP/Exchange/CardDAV/CalDAV服务时,需要用到授权码而不是QQ密码,授权码是用于登录第三方邮件客户端的专用密码。因此我们需要获得授权码,以在后面的程序中使用。

好了,到此准备工作就差不多了,下面开始动手吧。

实现注册Demo

创建Maven工程

本次案例基于Maven,因此你要先创建一个Maven的Web工程,并引入相关依赖:

<dependencies>
        <!-- JavaEE依赖 -->
        <dependency>
            <groupId>javaee</groupId>
            <artifactId>javaee-api</artifactId>
            <version>5</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>taglibs</groupId>
            <artifactId>standard</artifactId>
            <version>1.1.2</version>
        </dependency>
        <!-- mysql驱动依赖 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.40</version>
        </dependency>
        <!-- c3p0依赖 -->
        <dependency>
            <groupId>c3p0</groupId>
            <artifactId>c3p0</artifactId>
            <version>0.9.1.2</version>
        </dependency>
        <!-- JavaMail相关依赖 -->
        <dependency>
            <groupId>javax.mail</groupId>
            <artifactId>mail</artifactId>
            <version>1.4.7</version>
        </dependency>
        <dependency>
            <groupId>javax.activation</groupId>
            <artifactId>activation</artifactId>
            <version>1.1.1</version>
        </dependency>

</dependencies>

创建数据库表

接下来使用MySQL创建一张简单的用户表:

create table `user`(
    id int(11) primary key auto_increment comment ‘用户id‘,
    username varchar(255) not null comment ‘用户名‘,
    email varchar(255) not null comment ‘用户邮箱‘,
    password varchar(255) not null comment ‘用户密码‘,
    state int(1) not null default 0 comment ‘用户激活状态:0表示未激活,1表示激活‘,
    code varchar(255) not null comment ‘激活码‘
)engine=InnoDB default charset=utf8;

其中要注意的地方是state字段(用来判断用户账号是否激活)和code字段(激活码)。

创建注册页面

使用JSP创建一个最简单的注册页面(请自行忽略界面):

嗯,果然够简单。

主要的业务逻辑

先想一下,我们的整个流程应该是这样的:

  1. 用户填写相关信息,点击注册按钮
  2. 系统先将用户记录保存到数据库中,其中用户状态为未激活
  3. 系统发送一封邮件并通知用户去验证
  4. 用户登录邮箱并点击激活链接
  5. 系统将用户状态更改为已激活并通知用户注册成功

搞清楚了整个流程,实现起来应该就不难了。下图是我建立的包结构:

ps:完整代码请见后文链接,这里只讨论主要的思路

首先是,用户提交注册信息后,相应的servlet会将相关信息传给service层去处理,在service中需要做的就是讲记录保存到数据库中(调用dao层),然后再给用户发送一封邮件,UserServiceImpl相关代码如下:

public boolean doRegister(String userName, String password, String email) {
    // 这里可以验证各字段是否为空

    //利用正则表达式(可改进)验证邮箱是否符合邮箱的格式
    if(!email.matches("^\\[email protected](\\w+\\.)+\\w+$")){
        return false;
    }
    //生成激活码
    String code=CodeUtil.generateUniqueCode();
    User user=new User(userName,email,password,0,code);
    //将用户保存到数据库
    UserDao userDao=new UserDaoImpl();
    //保存成功则通过线程的方式给用户发送一封邮件
    if(userDao.save(user)>0){
        new Thread(new MailUtil(email, code)).start();;
        return true;
    }
    return false;
}

需要注意的是,应该新建一个线程去执行发送邮件的任务,不然被骂估计是免不了了。
数据库的操作比较简单,此处就不贴出来了,无非是将用户记录插到数据库中。值得一提的是,此处使用c3p0来作为数据源来替代DriverManager,在频繁获取释放数据库连接时效率会大大提高,c3p0最简单的配置如下:

<?xml version="1.0" encoding="UTF-8"?>
<c3p0-config>
    <named-config name="mysql">
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql://127.0.0.1:3306/test1?useSSL=false</property>
        <property name="user">root</property>
        <property name="password">123456</property>
        <!-- 初始化时一个连接池尝试获得的连接数量,默认是3,大小应该在maxPoolSize和minPoolSize之间 -->
        <property name="initialPoolSize">5</property>
        <!-- 一个连接最大空闲时间(单位是秒),0意味着连接不会过时 -->
        <property name="maxIdleTime">30</property>
        <!-- 任何指定时间的最大连接数量 ,默认值是15 -->
        <property name="maxPoolSize">20</property>
        <!-- 任何指定时间的最小连接数量 ,默认值是3 -->
        <property name="minPoolSize">5</property>
    </named-config>
</c3p0-config>

提供一个工具类DBUtil以获取,释放连接:

<pre>
public class DBUtil {
private static ComboPooledDataSource cpds=null;

static{
    cpds=new ComboPooledDataSource("mysql");
}

public static Connection getConnection(){
    Connection connection=null;
    try {
        connection = cpds.getConnection();
    } catch (SQLException e) {
        e.printStackTrace();
    }
    return connection;
}

public static void close(Connection conn,PreparedStatement pstmt,ResultSet rs){
    try {
        if(rs!=null){
            rs.close();
        }
        if(pstmt!=null){
            pstmt.close();
        }
        if(rs!=null){
            rs.close();
        }
    } catch (SQLException e) {
        e.printStackTrace();
    }

}

}
</pre>

要特别注意的一点是:即使是使用连接池,使用完Connection后调用close方法,当然这不意味着关闭与数据库的TCP 连接,而是将连接还回到池中去,如果不close掉的话,这个连接将会一直被占用,直到连接池中的连接耗尽为止。

使用JavaMail发送邮件

使用JavaMail发送邮件非常简单,也是三步曲:

  1. 创建连接对象javax.mail.Session
  2. 创建邮件对象 javax.mail.Message
  3. 发送邮件

直接看代码,详细的注释在代码中,MailUtil代码如下:

    public class MailUtil implements Runnable {
    private String email;// 收件人邮箱
    private String code;// 激活码

    public MailUtil(String email, String code) {
        this.email = email;
        this.code = code;
    }

    public void run() {
        // 1.创建连接对象javax.mail.Session
        // 2.创建邮件对象 javax.mail.Message
        // 3.发送一封激活邮件
        String from = "[email protected]";// 发件人电子邮箱
        String host = "smtp.qq.com"; // 指定发送邮件的主机smtp.qq.com(QQ)|smtp.163.com(网易)

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

        properties.setProperty("mail.smtp.host", host);// 设置邮件服务器
        properties.setProperty("mail.smtp.auth", "true");// 打开认证

        try {
            //QQ邮箱需要下面这段代码,163邮箱不需要
            MailSSLSocketFactory sf = new MailSSLSocketFactory();
            sf.setTrustAllHosts(true);
            properties.put("mail.smtp.ssl.enable", "true");
            properties.put("mail.smtp.ssl.socketFactory", sf);

            // 1.获取默认session对象
            Session session = Session.getDefaultInstance(properties, new Authenticator() {
                public PasswordAuthentication getPasswordAuthentication() {
                    return new PasswordAuthentication("[email protected]", "xxx"); // 发件人邮箱账号、授权码
                }
            });

            // 2.创建邮件对象
            Message message = new MimeMessage(session);
            // 2.1设置发件人
            message.setFrom(new InternetAddress(from));
            // 2.2设置接收人
            message.addRecipient(Message.RecipientType.TO, new InternetAddress(email));
            // 2.3设置邮件主题
            message.setSubject("账号激活");
            // 2.4设置邮件内容
            String content = "<html><head></head><body><h1>这是一封激活邮件,激活请点击以下链接</h1><h3><a href=‘http://localhost:8080/RegisterDemo/ActiveServlet?code="
                    + code + "‘>http://localhost:8080/RegisterDemo/ActiveServlet?code=" + code
                    + "</href></h3></body></html>";
            message.setContent(content, "text/html;charset=UTF-8");
            // 3.发送邮件
            Transport.send(message);
            System.out.println("邮件成功发送!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

ps:需要把上面的账号、授权码进行相应修改。

完成后,再有用户提交注册信息时,应该就能收到验证邮件了:

用户点击链接后,我们要做的工作就是根据code(可以利用UUID生成)更改数据库中相应用户的状态,然后提示用户注册结果了。

总结

简单介绍了如何使用JavaMail完成了一个带邮箱验证的注册案例,当然在实际开发中还有许多细节要注意,例如对用户提交信息的校验,密码进行加密等,此处的简单案例并未详尽处理这些细节。

代码

作者:SnDragon
链接:https://www.jianshu.com/p/8f8d7a46888f
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

原文地址:https://www.cnblogs.com/waves-a/p/9797961.html

时间: 2025-01-17 21:52:12

邮箱认证流程的相关文章

【爱江山越野跑】ITRA积分认证流程

背景:目前在越野跑领域,高级别的赛事有很多,比如UTMB,TDG等,而想报名参与这些赛事需要一定的积分(ITRA积分), 而这些积分的获得,需要参与获得ITRA认证的赛事,赛事难度不同,获得的积分也不同.所以对于赛事组委会而言,需要成为ITRA的认证赛事,就需要成为他们的会员,并提交赛事资料进行认证.从提交到认证结果下来大概需要1个月时间. 赛事认证流程 一.注册为ITRA成员 登录ITRA官网:http://www.i-tra.org/.由于目前(2016年)ITRA赛事认证系统正在更改,所以

企业微信认证需要什么资料(最新认证流程)

微信公众平台认证后具有权威性.名人效应.品牌特性.可信度提高,营销价值也将大幅提升.可申请微信认证的主体是企业.网店商家.媒体.政府及事业单位.其他组织.社会团体.民办非企业.其他盈利组织等. 微信认证所需资料 1.微信认证的名称:如上海某公司.某旗舰店.某部队医院.某品牌 2.手机号码:用于审核,不对用户开放 3.固定电话:用于审核,不对用户开放,没有可不提供 5.邮箱地址: 6.公司地址:个人可提供住址等 7.身份证:要求能看清名字和身份证号码 8.企业logo:可作为头像,如没有可以后补充

邮箱认证新用户注册,并用验证回传参数完成用户的注册

邮箱认证新用户注册,也可用于重置密码 第一部分:发送邮箱 1.当用户注册完的时候,提示用户打开邮箱验证 2.当用户填写信息后,用户信息插入数据库(ischeck设置为0 未验证),并发送邮件(内容用微信的方式进行拼接和回调参数) public static void SendMail(string smtpserver/*SMTP服务器smtp.qq.com*/, int enablessl/*是否启用SSL加密*/, string userName/*登录帐号*/, string pwd/*登

Shiro权限控制框架入门1:Shiro的认证流程以及基本概念介绍

前言:我在最开始学习Shiro这个框架时,在网上搜索到的一个介绍比较全面的教程是:<跟我学Shiro>系列教程.但是在我看了他写的前几篇文章后,我发现虽然他在这个系列教程中把shiro的一些特性介绍地非常全面详细,但是整个教程的叙述方式还是有很大缺陷的.文章与文章之间并没有很好地串联起来,每篇文章介绍的东西都过于分散了,如果是对shiro完全不了解的新手来看的话完全是一场噩梦.就像一个网友评价的这样: 看了看这个教程,看完之后都想放弃shiro了,完全看不懂,后来百度了很多别的资料才理解了sh

优化的网站注册时[验证邮箱]的流程

你通常见到的一个流程大概是这样的:1. 填写用户名.密码.重复密码.Email.……,提交:2. 收取邮件,点击激活链接:3. 进一步完善资料或使用网站. 照上面的流程,请大家思考:“在邮箱激活链接未得到使用之前,网站就已经将该Email地址与一个用户名和密码绑定了关系,这是否存在问题?” 这当然是存在问题的,就像银行不会还没看你身份证时,就先发一张银行卡和密码给你.既然要验证邮箱是否有效,就不应该在邮箱未验证成功之前就在该邮箱上绑定上用户名.密码等信息.正常的流程应该是:用户通过邮箱获得了激活

身份认证流程描述

1.1.1          身份认证 每个用户都持有可唯一标识用户身份的私钥和由CA签发的公钥证书,用户私有信息保存在电子钥匙(USBKEY)或其它安全载体中.在访问应用系统时,用户出示自已的身份证明(签名和证书),实现基于证书的身份认证过程. 身份认证分为单向或双向认证方式,在一般情况下采用服务器认证客户的单向认证方式,在必要时可采用双向认证方式. 应用系统中,应用服务程序直接通过安全接口调用认证服务器对用户进行身份认证,用户身份信息由应用系统自身进行维护和管理. 应用系统服务器端和客户端各

微信服务号认证流程

一.申请微信认证 1.申请微信认证入口. 1) “设置->账户信息->认证情况“点击“申请微信认证”进入. 2) “服务->服务中心“点击“微信认证”进入. 2.申请微信认证操作全攻略. 1) 签署<微信公众平台认证服务协议>,勾选同意,点击下一步. 2) 同意<公众号命名规则>,勾选同意,点击下一步. 3) 填写资料:根据账号类型选择企业或媒体(目前只支持这两种类型),并如实填写认证资料. 特别说明:企业对公银行帐户为必填项,在审核过程中第三方审核公司会向该帐户

Shiro learning - 认证流程

Shiro认证流程 在学习认证流程之前,你应该先了解Shiro的基本使用流程 认证 身份认证: 证明用户是谁.用户需要提供相关的凭证principals(身份标识)和Credentials (凭证,证明你是这个用户,可以理解成密码) Principals: 用户的属性,可以有多个.但是至少有一个属性能唯一用户 Credentials: 证明信息,密码或者证书之类的 认证流程 token传给Subject.login(Token).在调用login方法时候,内部完成了认证和授权 实际上的认证是由S

企业邮箱申请流程注意哪些方面?

如今不少企业都会使用企业邮箱,一方面是能让企业走向正规标准化,另一方面是各方面功能都能满足公司工作的需求.企业邮箱申请流程需要注意什么?现在和大家说说.1.确定相关域名,一般都是以公司名称设定域名内容.企业邮箱一般以公司域名组成后缀,能帮助公司开展网络营销. 2.确定公司需要使用人数和年限,企业邮箱针对不同的年限和用户数的优惠有所不同.建议用户数和年限可以多买一些,企业邮箱一般都会长期使用,日后公司规模扩大也方便分配. 3.官网登记之后会有品牌或者经销商前来联系的.我们公司那时申请免费试用网易企