在平常生活中,咱們在一個網站中註冊一個帳戶時,每每在提交我的信息後,網站還要咱們經過手機或郵件來驗證,郵件的話大概會是下面這個樣子的: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。