实现用户登录和注册功能。项目分为 Web 层、业务逻辑层、数据访问层。
将用户信息封装在 UserBean 对象中。用户信息有:username(String)、password(String)、nickname(String)、email(String)、birthday(Date)。
在注册页面设计时,我们应考虑用户输入数据的合法性,所以我们将用户输入的表单数据封装在 FormBean 中,由于用户提交的数据都是以字符串的形式传递,因此 FormBean 中的所有对象都是 String 类型。在 FormBean 中还维护了一个 Map 对象,用以存放错误提示信息。如果用户输入的数据不合法,我们可以将错误信息作为 value 将表单项的名作为 key 存放在 Map 中,再以 request 属性的方式转发给注册页面,实现动态错误提示功能。为了防止机器人注册,我们还应该在注册页面中加入验证码。生成验证码的 Servlet 作为工具类存放在 utils 包中。服务器产生的验证码数据存放在服务器 session 对象中,用以用户提交表单时验证。为了防止 表单重复提交,我们在表单中还应该增加一个表单令牌,以隐藏域的形式存放。同样服务器的 session 对象中也应该增加这个属性。
当用户提交注册请求的时候,我们利用 BeanUtils 的 populate 方法,将请求数据存放在 FormBean 中,如果验证合法,那么我们再将 FormBean 中的数据拷贝到 UserBean 中存储。在拷贝 Bean 对象时,我们利用 BeanUtils 的 copyBean 方法,将一个对象中的属性拷贝到另一个对象。注意,因为 FormBean 中的属性都是 String 类型,而 UserBean 中的生日是以 Date 类型存储,因此我们在拷贝属性时应该自定义一个转换器。在数据库中存储的用户密码应该是以 MD5 方式加密后的数据,我们在存储用户名密码时应该先用 MD5 运算后再存入数据库。
用户登录页面提交数据到 LoginServlet 时,我们先将提交的密码数据进行 MD5 运算,然后与数据库中的密码进行比对。如果比对成功,则在 Session 域中存储用户信息,再跳转至首页。如果比对失败则将错误信息存放在 request 域中带给客户机进行提示。
业务逻辑图:
包层次关系:
cn.dk.domain
package cn.dk.domain;
import java.util.Date;
public class User {
public User() {
super();
}
public User(String username, String password, String nickname,
String email, Date birthday) {
super();
this.username = username;
this.password = password;
this.nickname = nickname;
this.email = email;
this.birthday = birthday;
}
private String username;
private String password;
private String nickname;
private String email;
private Date birthday;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public String getNickname() {
return nickname;
}
public void setNickname(String nickname) {
this.nickname = nickname;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
}
cn.dk.dao
package cn.dk.dao;
import cn.dk.domain.User;
public interface IUserDao {
public void insertUser(User user);
public boolean seacherUserByUsername(String username);
public User searchUser(String username, String password);
}
package cn.dk.dao.impl;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.Node;
import com.sun.org.apache.xerces.internal.impl.dv.util.Base64;
import cn.dk.dao.IUserDao;
import cn.dk.domain.User;
import cn.dk.utils.XmlUtils;
public class UserDaoImpl implements IUserDao {
public void insertUser(User user) {
// 获取xml文档对象
Document document = XmlUtils.getDocument();
// 获取文档根元素
Element root = document.getRootElement();
// 向根元素中增加子元素
Element child = root.addElement("user");
// 向子元素中增加用户名和密码属性
child.addAttribute("username", user.getUsername());
// 对密码加密
String disgestPass = this.md5Disgest(user.getPassword());
child.addAttribute("password", disgestPass);
// 向子元素中增加昵称、email、生日属性
child.addElement("nickname").addText(user.getNickname());
child.addElement("email").addText(user.getEmail());
if (user.getBirthday() != null) {
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
String birthday = format.format(user.getBirthday());
child.addElement("birthday").addText(birthday);
}
// 保存xml到磁盘
XmlUtils.saveXml(document);
}
public User searchUser(String username, String password) {
// 获取xml文档对象
Document document = XmlUtils.getDocument();
// 获取文档根元素
Element root = document.getRootElement();
// 对密码进行md5运算
String disgestPass = this.md5Disgest(password);
// 匹配用户名和密码
Element child = (Element) root.selectSingleNode("//user[@username='"
+ username + "' and @password='" + disgestPass + "']");
User user = null;
if (child != null) {
user = new User();
user.setUsername(child.attributeValue("username"));
user.setPassword(child.attributeValue("password"));
user.setNickname(child.elementText("nickname"));
user.setEmail(child.attributeValue("email"));
String birthdayStr = child.attributeValue("birthday");
if (birthdayStr != null) {
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
try {
Date birthday = format.parse(child
.attributeValue("birthday"));
user.setBirthday(birthday);
} catch (ParseException e) {
throw new RuntimeException(e);
}
}
}
return user;
}
public boolean seacherUserByUsername(String username) {
// 获取xml文档对象
Document document = XmlUtils.getDocument();
// 获取文档根元素
Element root = document.getRootElement();
// 匹配用户名
Node selectSingleNode = root.selectSingleNode("//user[@username='"
+ username + "']");
if (sel