最近项目中有一个功能,注册的时候需要验证邮箱,发送激活连接,激活之后才能登陆,在写的时候也是遇到了一些小阻碍,这里写下来给大家借鉴借鉴。
项目环境
我直接使用的是我们项目环境来写的,使用ssm框架,所以呢dao层访问数据库的部分我就省略啦,这里大家直接使用原生的jdbc或者其他的orm框架都是没有问题的。好了,不多说啦,直接进入重点吧。
代码
1、实体类User
这里实体类和数据库是对应的,省略get和set方法以及构造器。
public class User {
private Integer id;
private String username;
private String password;
private String email;
/*
* 0 未激活
* 1 已激活
* */
private Integer state;
/*
* 激活码
* */
private String code;
}
2、注册界面
注册界面写的很简单啦,<form>+<table>的组合,界面简单,只是为了测试。
<form action="/register" method="post">
<table>
<tr>
<td>用户名:</td>
<td><input type="text" id="username" name="username"></td>
</tr>
<tr>
<td>密码:</td>
<td><input type="password" name="password"></td>
</tr>
<tr>
<td>邮箱:</td>
<td><input type="text" name="email"></td>
</tr>
<tr >
<td aria-colspan="2"><input type="submit" value="注册"></td>
</tr>
</table>
</form>
3、注册Controller
注册的功能还是一样的,把实体类封装存到数据库里,只不过在controller(也就是Servlet啦,spring MVC里面要叫controller)里面需要给用户设定激活状态State和激活码code。因为我使用的ssm框架,所以user帮我自动封装了,使用Servlet的小伙伴请手动封装。
@RequestMapping("/register")
public String register(User user, ModelMap modelMap) throws Exception {
//封装数据,默认状态为未激活
user.setState(0);
//设置激活码
user.setCode(UUID.randomUUID().toString().replaceAll("-",""));
System.out.println(user);
//调用业务层处理数据
userService.register(user);
//页面跳转
modelMap.addAttribute("msg","激活码已发送,请去邮箱激活账户");
return "success";
}
4、Service
Service层也就是对数据库的一些操作,我们要实现邮箱激活的功能需要进行的数据库操作有:添加一个用户,查询用户,修改用户。
@Service
public class UserServiceImpl implements UserService {
@Autowired
UserMapper userMapper;
@Override
public void register(User user) throws Exception {
//将数据存入数据库
userMapper.register(user);
//发送一封激活邮件
EmailUtil.sendEmailByLocal(user.getEmail(),user.getCode());
}
//查询该激活码对应的用户,并返回该用户
@Override
public User activeUser(String code) {
return userMapper.activeUser(code);
}
//更新用户状态
@Override
public int updateUser(User user) {
return userMapper.updateUser(user);
}
}
5、EmailUtil
重点部分来了,前面都是在给做铺垫,这个部分才是我今天想要分享的重点,如何发送给邮件,如果你使用163或者qq邮箱,请在邮箱设置里面开启smtp哟,这个我用的163邮箱,开启会让你设置授权码,授权码一定要记住呀,因为我们后面会用到。
请看一下代码。
public class EmailUtil {
/**
* 发件人的邮箱和密码
* 某些邮箱服务器为了增加邮箱本身密码的安全性
* 给SMTP,客户端设置了独立密码(有的邮箱称为“授权码”)
* 比如163,这里我用163邮箱测试的,
* 设置了独立密码的邮箱,所以这里的邮箱密码必需使用这个独立密码
**/
public static String myEmailAccount = "******";
public static String myEmailPassword = "********";
/**
* 发件人邮箱的 SMTP 服务器地址必须准确, 不同邮件服务器地址不同,
* 网易163邮箱的 SMTP 服务器地址为: smtp.163.com
*/
public static String myEmailSMTPHost = "smtp.163.com";
/**
* 使用别人家的邮箱服务器,例如163邮箱,qq邮箱等
* @param to 给谁发邮件,也就是发送邮件的地址
* @param code 激活码
*/
public static void sendEmailByWeb(String to,String code) throws Exception {
// 1. 创建参数配置, 用于连接邮件服务器的参数配置
Properties props = new Properties();
props.setProperty("mail.transport.protocol", "smtp"); // 使用的协议(JavaMail规范要求)
props.setProperty("mail.smtp.host", myEmailSMTPHost); // 发件人的邮箱的 SMTP 服务器地址
props.setProperty("mail.smtp.auth", "true"); // 需要请求认证
//2.创建链接对象,链接到邮箱服务器
Session session = Session.getDefaultInstance(props);
//这里是为了在控制台可以看到详细发送的信息
session.setDebug(true);
//3.创建邮件对象
Message message = createMimeMessage(session,myEmailAccount,to,code);
// 4. 根据 Session 获取邮件传输对象
Transport transport = session.getTransport();
//使用邮箱账号和密码连接邮件服务器, 这里认证的邮箱必须与 message 中的发件人邮箱一致, 否则报错
transport.connect(myEmailAccount, myEmailPassword);
// 6. 发送邮件, 发到所有的收件地址, message.getAllRecipients() 获取到的是在创建邮件对象时添加的所有收件人, 抄送人, 密送人
transport.sendMessage(message, message.getAllRecipients());
// 7. 关闭连接
transport.close();
}
/**
* 使用自己本地的邮箱服务器,这里我用的是易邮邮箱服务器
* @param to
* @param code
* @throws Exception
*/
public static void sendEmailByLocal(String to,String code) throws Exception {
//1、连接配置,本地不用配置
Properties props = new Properties();
//2、创建连接对象
Session session = Session.getDefaultInstance(props, new Authenticator() {
@Override
protected PasswordAuthentication getPasswordAuthentication() {
//这里是本地的邮箱账户和密码,域名可以在邮箱服务器自己设定
return new PasswordAuthentication("server@sinnamm.con","server");
}
});
//获取邮件对象
Message message = createMimeMessage(session,"server@sinnamm.com",to,code);
//发送邮件
Transport.send(message);
}
/**
* 创建邮件
* @param session 和服务器交互的会话
* @param sendMail 发件人邮箱
* @param receiveMail 收件人邮箱
* @return
* @throws Exception
*/
public static MimeMessage createMimeMessage(Session session, String sendMail, String receiveMail,String code) throws Exception {
// 1. 创建一封邮件
MimeMessage message = new MimeMessage(session);
// 2. From: 发件
message.setFrom(new InternetAddress(sendMail, "findlover", "UTF-8"));
// 3. To: 收件人(可以增加多个收件人、抄送、密送)
message.setRecipient(MimeMessage.RecipientType.TO, new InternetAddress(receiveMail, "用户", "UTF-8"));
// 4. Subject: 邮件主题
message.setSubject("激活邮件", "UTF-8");
// 5. Content: 邮件正文(可以使用html标签)
message.setContent("<h1>请点击以下链接激活你的账号</h1>" +
"<h3><a href='http://localhost:8080/ActiveServlet?code="+code+"'>http://localhost:8080/ActiveServlet?code="+code+"</a></h3>","text/html;charset=UTF-8");
// 6. 设置发件时间
message.setSentDate(new Date());
// 7. 保存设置
message.saveChanges();
return message;
}
}
在工具类中我写了两种发送邮件的方法
- 直接使用163邮箱或者qq邮箱来进行发送,这个需要在邮箱的设置里面开启smtp这个选项,不然不能发送。
- 本地下载安装易邮邮箱服务器,然后新建几个邮箱账号就可以当我们的测试邮箱了。对了,这里还需要使用邮箱客户端,我用的是Foxmail客户端,把本地的账号添加进去就可以接收邮件了。在本地测试,写这个的原因是为了方便测试啦,总觉得本地的东西比联网方便。
6、激活controller
最后一步啦,邮箱已经成功发送了,剩下的就是激活啦。用户点击链接会将请求发送给我们的Servlet来进行处理
@RequestMapping("/ActiveServlet")
public String ActiveServlet(@Param("code")String code,ModelMap modelMap) throws Exception {
//接收激活码,Spring MVC封装好了
//根据激活码查询用户
User user = userService.activeUser(code);
//查询到用户,修改用户的状态
if (user!=null){
//查询到用户
user.setState(1);
user.setCode(null);
//更新用户信息
userService.updateUser(user);
modelMap.addAttribute("msg","激活成功,请登录");
return "success";
}else {
//激活码没有查询到该用户
modelMap.addAttribute("msg","激活码失效,请重新获取");
return "success";
}
}
这样一个简单的邮件激活就写完啦,写这么多其实重点只有发送邮件工具类的编写上,其他的都是一些注册代码和用户信息的处理。读者们测试的时候可以直接写工具类里面的内容就可以了,一样能发送成功!
是不是简单呢,是的没错,赶快动手操作一下吧。