用户在注册后会在数据库中插入一条UUID生成的64位的编码,并且将用户激活标志置0,然后在用户登陆时除了判断用户存在和密码正确外还要校验激活标志,以此来达成业务功能。
该项目的包结构如图
在MailUtil.java中封装了发送邮件功能的代码:
package me.reg.util;
import java.util.Date;
import java.util.Properties;
import javax.mail.Session;
import javax.mail.Transport;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeMessage;
public class MailUtil {
public static String myEmailAccount = "[email protected]";
public static String myEmailPassword = "123456";
// 发件人邮箱的 SMTP 服务器地址, 必须准确, 不同邮件服务器地址不同, 一般(只是一般, 绝非绝对)格式为: smtp.xxx.com
// 网易163邮箱的 SMTP 服务器地址为: smtp.163.com
public static String myEmailSMTPHost = "127.0.0.1";
// 收件人邮箱(替换为自己知道的有效邮箱)
// public static String receiveMailAccount = "[email protected]";
public static void sendTo(String body, String receiveMailAccount) throws Exception { //只需要传入正文和目标地址即可
Properties props = new Properties(); // 参数配置
props.setProperty("mail.transport.protocol", "smtp"); // 使用的协议(JavaMail规范要求)
props.setProperty("mail.smtp.host", myEmailSMTPHost); // 发件人的邮箱的 SMTP
// 服务器地址
props.setProperty("mail.smtp.auth", "true"); // 需要请求认证
// PS: 某些邮箱服务器要求 SMTP 连接需要使用 SSL 安全认证 (为了提高安全性, 邮箱支持SSL连接, 也可以自己开启),
// 如果无法连接邮件服务器, 仔细查看控制台打印的 log, 如果有有类似 “连接失败, 要求 SSL 安全连接” 等错误,
// 打开下面 /* ... */ 之间的注释代码, 开启 SSL 安全连接。
/*
* // SMTP 服务器的端口 (非 SSL 连接的端口一般默认为 25, 可以不添加, 如果开启了 SSL 连接, //
* 需要改为对应邮箱的 SMTP 服务器的端口, 具体可查看对应邮箱服务的帮助, // QQ邮箱的SMTP(SLL)端口为465或587,
* 其他邮箱自行去查看) final String smtpPort = "465";
* props.setProperty("mail.smtp.port", smtpPort);
* props.setProperty("mail.smtp.socketFactory.class",
* "javax.net.ssl.SSLSocketFactory");
* props.setProperty("mail.smtp.socketFactory.fallback", "false");
* props.setProperty("mail.smtp.socketFactory.port", smtpPort);
*/
// 2. 根据配置创建会话对象, 用于和邮件服务器交互
Session session = Session.getDefaultInstance(props);
session.setDebug(false); // 设置为debug模式, 可以查看详细的发送 log
// 4. 根据 Session 获取邮件传输对象
Transport transport = session.getTransport();
// 3. 创建一封邮件
MimeMessage message = createMimeMessage(session, myEmailAccount, receiveMailAccount, body);
// 4. 根据 Session 获取邮件传输对象
transport = session.getTransport();
// 5. 使用 邮箱账号 和 密码 连接邮件服务器, 这里认证的邮箱必须与 message 中的发件人邮箱一致, 否则报错
//
// PS_01: 成败的判断关键在此一句, 如果连接服务器失败, 都会在控制台输出相应失败原因的 log,
// 仔细查看失败原因, 有些邮箱服务器会返回错误码或查看错误类型的链接, 根据给出的错误
// 类型到对应邮件服务器的帮助网站上查看具体失败原因。
//
// PS_02: 连接失败的原因通常为以下几点, 仔细检查代码:
// (1) 邮箱没有开启 SMTP 服务;
// (2) 邮箱密码错误, 例如某些邮箱开启了独立密码;
// (3) 邮箱服务器要求必须要使用 SSL 安全连接;
// (4) 请求过于频繁或其他原因, 被邮件服务器拒绝服务;
// (5) 如果以上几点都确定无误, 到邮件服务器网站查找帮助。
//
// PS_03: 仔细看log, 认真看log, 看懂log, 错误原因都在log已说明。
transport.connect(myEmailAccount, myEmailPassword);
// 6. 发送邮件, 发到所有的收件地址, message.getAllRecipients()
// 获取到的是在创建邮件对象时添加的所有收件人,
// 抄送人, 密送人
transport.sendMessage(message, message.getAllRecipients());
// 7. 关闭连接
transport.close();
}
public static MimeMessage createMimeMessage(Session session, String sendMail, String receiveMail, String body)
throws Exception {
// 1. 创建一封邮件
MimeMessage message = new MimeMessage(session);
// 2. From: 发件人
message.setFrom(new InternetAddress(sendMail, "LB", "UTF-8"));
// 3. To: 收件人(可以增加多个收件人、抄送、密送)
message.setRecipient(MimeMessage.RecipientType.TO, new InternetAddress(receiveMail, "XX用户", "UTF-8"));
// 4. Subject: 邮件主题
message.setSubject("激活您的帐号", "UTF-8");
// 5. Content: 邮件正文(可以使用html标签)
message.setContent(body, "text/html;charset=UTF-8");
// 6. 设置发件时间
message.setSentDate(new Date());
// 7. 保存设置
message.saveChanges();
return message;
}
}
Controller的代码如下:
package me.reg.controller;
import javax.servlet.http.HttpServletRequest;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;
import me.reg.domain.User;
import me.reg.server.UserServer;
import me.reg.util.LogFlag;
@Controller
@RequestMapping(value = "/user")
public class UserController {
@Autowired
private UserServer us;
@RequestMapping(value = "/log.do")
public ModelAndView log(ModelAndView mv, HttpServletRequest request) {
// TODO 获取用户登录名 密码 组装一个user
String username = request.getParameter("username");
String password = request.getParameter("password");
User user = new User();
user.setPassword(password);
user.setUsername(username);
// TODO 调用us的登录方法判断用户情况,根据用户情况跳转不同页面
if (LogFlag.NOUSER.equals(us.log(user))) {
mv.setViewName("/WEB-INF/page/fail.jsp");
mv.addObject("msg", "未注册");
}
if (LogFlag.NOTCONFORM.equals(us.log(user))) {
mv.setViewName("/WEB-INF/page/fail.jsp");
mv.addObject("msg", "未确认注册");
}
if (LogFlag.OK.equals(us.log(user))) {
mv.setViewName("/WEB-INF/page/success.jsp");
}
if (LogFlag.PASSWORDERR.equals(us.log(user))) {
mv.setViewName("/WEB-INF/page/fail.jsp");
mv.addObject("msg", "密码错误");
}
// TODO 把user放入mv中
mv.addObject("user", user);
return mv;
}
@RequestMapping(value = "reg.do")
public ModelAndView reg(ModelAndView mv, HttpServletRequest request) {
User user = new User();
user.setEmail(request.getParameter("email"));
user.setUsername(request.getParameter("username"));
user.setPassword(request.getParameter("password"));
if (us.reg(user)) {
mv.setViewName("/user/log.jsp");
} else {
mv.addObject("msg", "注册失败");
mv.setViewName("/WEB-INF/page/fail");
}
return mv;
}
@RequestMapping(value = "regconf.do")
public ModelAndView regconf(ModelAndView mv, HttpServletRequest request) {
String code = request.getParameter("code");
us.regconf(code);
mv.setViewName("/user/log.jsp");
return mv;
}
}
顺便提一下CharFilter,在开始做这个小项目的时候忘了考虑中文乱码的问题于是被坑到了,提交的中文用户名将会出现问题。
package me.reg.filter;
import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
/**
* Servlet Filter implementation class CharFilter
*/
public class CharFilter implements Filter {
private static String charEncoding;
/**
* Default constructor.
*/
public CharFilter() {
// TODO Auto-generated constructor stub
}
/**
* @see Filter#destroy()
*/
public void destroy() {
// TODO Auto-generated method stub
}
/**
* @see Filter#doFilter(ServletRequest, ServletResponse, FilterChain)
*/
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
// TODO Auto-generated method stub
// place your code here
// pass the request along the filter chain
request.setCharacterEncoding(charEncoding);
response.setContentType("text/html;charset=utf-8");
chain.doFilter(request, response);
}
/**
* @see Filter#init(FilterConfig)
*/
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
charEncoding = fConfig.getInitParameter("charEncoding");
System.out.println("CharFilter is inited");
}
}
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
reg
注册
登录页面:
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
Insert title here
下面是实现邮箱验证的主要代码:
package me.reg.server.impl;
import java.util.UUID;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import me.reg.domain.User;
import me.reg.mapper.UserMapper;
import me.reg.server.UserServer;
import me.reg.util.LogFlag;
import me.reg.util.MailUtil;
@Service
public class UserServerImp implements UserServer {
@Autowired
private UserMapper um;
@Override
public boolean reg(User user) {
// TODO Auto-generated method stub
String code = UUID.randomUUID().toString() + UUID.randomUUID().toString();
code = code.replaceAll("-", "");
user.setCode(code);
if (um.createUSeer(user) == 1) {
try {
MailUtil.sendTo("激活帐号",
user.getEmail());
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return true;
}
return false;
}
@Override
public LogFlag log(User user) {
// TODO Auto-generated method stub
User rUser = um.queryUserByName(user.getUsername());
if (null == rUser) {
return LogFlag.NOUSER;
}
if (rUser.getState() == 0) {
return LogFlag.NOTCONFORM;
}
if (!rUser.getPassword().equals(user.getPassword())) {
return LogFlag.PASSWORDERR;
}
return LogFlag.OK;
}
@Override
public void regconf(String code) {
// TODO Auto-generated method stub
User user = um.queryUser(code);
user.setState(1);
user.setCode(code);
um.modifyUser(user);
}
}
最终的效果如图