在平常生活中,咱们在一个网站中注册一个帐户时,每每在提交我的信息后,网站还要咱们经过手机或邮件来验证,邮件的话大概会是下面这个样子的:html
用户经过点击连接从而完成注册,而后才能登陆。java
也许你会想,为何要这么麻烦直接提交注册不就好了吗?这其中很大一部分缘由是为了防止恶意注册。接下来让咱们一块儿来使用最简单的JSP+Servlet的方式来完成一个经过邮箱验证注册的小案例吧。mysql
动手实践以前,你最好对如下知识有所了解:git
若是对邮件收发过程彻底不了解的话,能够花三分钟的时间到慕课网了解一下,讲得算是很是清楚了,这里就不赘述了。放张图回忆一下:github
邮件收发过程正则表达式
在了解的上述内容以后,要实现这个案例,首先咱们还得有两个邮箱帐号,一个用来发送邮件,一个用来接收邮件。本案例使用QQ邮箱向163邮箱发送激活邮件,所以须要登陆QQ邮箱,在设置->帐户面板中开启POP3/SMTP服务,以容许咱们经过第三方客户端发送邮件:sql
还要注意的是,登陆如下服务: POP3/IMAP/SMTP/Exchange/CardDAV/CalDAV服务时,须要用到受权码而不是QQ密码,受权码是用于登陆第三方邮件客户端的专用密码。所以咱们须要得到受权码,以在后面的程序中使用。数据库
好了,到此准备工做就差很少了,下面开始动手吧。api
本次案例基于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建立一个最简单的注册页面(请自行忽略界面):
嗯,果真够简单。
先想一下,咱们的整个流程应该是这样的:
搞清楚了整个流程,实现起来应该就不难了。下图是我创建的包结构:
ps:完整代码请见后文连接,这里只讨论主要的思路
首先是,用户提交注册信息后,相应的servlet会将相关信息传给service层去处理,在service中须要作的就是讲记录保存到数据库中(调用dao层),而后再给用户发送一封邮件,UserServiceImpl相关代码以下:
public boolean doRegister(String userName, String password, String email) { // 这里能够验证各字段是否为空 //利用正则表达式(可改进)验证邮箱是否符合邮箱的格式 if(!email.matches("^\\w+@(\\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发送邮件很是简单,也是三步曲:
直接看代码,详细的注释在代码中,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 = "xxx@qq.com";// 发件人电子邮箱 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("xxx@qq.com", "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
來源:简书
著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。
注:想要让别人可以成功的完成注册,须要提供一个公网ip,替换掉localhost。