本旅游网站的架构设计主要分为可以3层,主要有Web层,业务层,Model层。其中web层还包括View层和Controller层,Model层包括元数据扩展层和数据访问层。
系统架构如下图所示。
图4-1 系统架构
旅游网站总体分为前台用户模块和后台管理员模块。
两个模块表现上是分别独立存在,但是访问的数据库是一样的。每一个模块的功能都是根据先前完成的需求分析,并查阅相关资料后整理制作的。
综上所述,系统功能结构图如下图所示。
图4-2 系统功能结构图
登录模块:登录模块是进入系统的入口,所有用户必须登录后才能访问系统。登录需要输入用户名和密码,如果多次尝试登录需要输入验证码。登录时需要选择用户的角色,是一般用户还是管理员登录等。登录成功后,会通过数据库获取用户的权限,并跳转至用户的主页面。
管理员用户管理模块:管理员管理包括:管理员的添加,修改和删除操作。添加管理员时,先判断用户添加的管理员是否是admin,如果不是则添加成功。修改时候,可以修改所有管理员的信息的操作。
景点信息:用户可以根据景点类型浏览景点的信息,并选中某个景点查看详情,例如:景点名称、景点编号、景点类型、景点介绍等。
景点购票管理模块:景点购票管理分为订单号、行动编号、景点名称、票价、购买数量、总金额。景点信息由管理员进行详情、删除操作;景点门票购买由用户来执行。
酒店信息管理模块:酒店管理分为酒店名称、酒店编号、房型、客户电话、照片、价格、酒店介绍、便利设施客房图片、酒店等级、客房类型修改。酒店信息由管理员进行详情、修改、添加、删除操作;酒店预订由用户来执行。
从前面可以分析到数据库中最重要的是用户信息,景点信息、酒店信息、景点购票、酒店预订,分析可以得到如下数据描述:
注册用户:用于记录注册用户的各种信息,包括用户名、审核状态等数据项。
管理员:记录管理员的登录信息。包括用户名,密码,权限等数据项。
景点购票:存放给景点购票的内容,包括订单号、景点编号、景点名称、票价、购买数量、总金额、联系电话、支付状态、支付类型等数据项。
景点信息:存储各景点信息。包括景点编号、景点名称、照片、票价、地址、交通指南、特色项目、点击数、点赞数等数据项。
酒店预订:存储用户的酒店预订信息。包括订单号、酒店名称、房型、价格、预订天数、预订数量、总价格、用户名、联系电话、入住日期、支付状态、支付类型等数据项。
酒店信息。存储用户的酒店信息。包括酒店信息的酒店编号、酒店名称、房型、地址、客户电话、价格、照片、便利设施、到店指南、酒店介绍、点击数、点赞数等数据项。
根据前面的数据流程图,结合系统的功能模块设计,设计出符合系统的各信息实体。
系统ER图如下图所示。
图4-3 系统ER图
旅游网站所拥有的数据表有以下:用户信息表,热门景点信息表,门票购买信息表,线路预订信息表,旅游线路表,酒店预订表。
由于数据表较多,只展示系统主要数据表,如下表所示。
scenic_spot_ticket_purchase表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
scenic_spot_ticket_purchase_id | int | 11 | 是 | 是 | 景点购票ID |
order_number | varchar | 64 | 否 | 否 | 订单号 |
attraction_number | varchar | 64 | 否 | 否 | 景点编号 |
name_of_scenic_spot | varchar | 64 | 否 | 否 | 景点名称 |
ticket_price | varchar | 64 | 否 | 否 | 票价 |
purchase_quantity | int | 11 | 否 | 否 | 购买数量 |
total_amount | varchar | 64 | 否 | 否 | 总金额 |
user_name | int | 11 | 否 | 否 | 用户名 |
contact_number | varchar | 64 | 否 | 否 | 联系电话 |
pay_state | varchar | 16 | 是 | 否 | 支付状态 |
pay_type | varchar | 16 | 否 | 否 | 支付类型宝、网银 |
recommend | int | 11 | 是 | 否 | 智能推荐 |
create_time | datetime | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
scenic_spot_information表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
scenic_spot_information_id | int | 11 | 是 | 是 | 景点信息ID |
attraction_number | varchar | 64 | 否 | 否 | 景点编号 |
name_of_scenic_spot | varchar | 64 | 否 | 否 | 景点名称 |
photo | varchar | 255 | 否 | 否 | 照片 |
ticket_price | int | 11 | 否 | 否 | 票价 |
address | varchar | 64 | 否 | 否 | 地址 |
opening_hours | varchar | 64 | 否 | 否 | 开放时间 |
traffic_guide | text | 0 | 否 | 否 | 交通指南 |
characteristic_project | text | 0 | 否 | 否 | 特色项目 |
introduction_to_scenic_spots | longtext | 0 | 否 | 否 | 景点介绍 |
hits | int | 11 | 是 | 否 | 点击数 |
praise_len | int | 11 | 是 | 否 | 点赞数 |
recommend | int | 11 | 是 | 否 | 智能推荐 |
create_time | datetime | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
registered_user表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
registered_user_id | int | 11 | 是 | 是 | 注册用户ID |
user_name | varchar | 64 | 是 | 否 | 用户名 |
examine_state | varchar | 16 | 是 | 否 | 审核状态 |
recommend | int | 11 | 是 | 否 | 智能推荐 |
user_id | int | 11 | 是 | 否 | 用户ID |
create_time | datetime | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
hotel_reservation表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
hotel_reservation__id | int | 11 | 是 | 是 | 酒店预订ID |
order_number | varchar | 64 | 否 | 否 | 订单号 |
hotel_name | varchar | 64 | 否 | 否 | 酒店名称 |
layout_of_a_house_or_an_apartment | varchar | 64 | 否 | 否 | 房型 |
price | varchar | 64 | 否 | 否 | 价格 |
booking_quantity | int | 11 | 否 | 否 | 预订数量 |
booking_days | int | 11 | 否 | 否 | 预订天数 |
total_price | varchar | 64 | 否 | 否 | 总价格 |
user_name | int | 11 | 否 | 否 | 用户名 |
contact_number | varchar | 64 | 否 | 否 | 联系电话 |
check_in_date | date | 0 | 否 | 否 | 入住日期 |
pay_state | varchar | 16 | 是 | 否 | 支付状态 |
pay_type | varchar | 16 | 否 | 否 | 支付类型: |
recommend | int | 11 | 是 | 否 | 智能推荐 |
create_time | datetime | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
Xianluyuding表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
hotel_information_id | int | 11 | 是 | 是 | 酒店信息ID |
hotel_number | varchar | 64 | 否 | 否 | 酒店编号 |
hotel_name | varchar | 64 | 否 | 否 | 酒店名称 |
layout_of_a_house_or_an_apartment | varchar | 64 | 否 | 否 | 房型 |
address | varchar | 64 | 否 | 否 | 地址 |
customer_telephone | varchar | 64 | 否 | 否 | 客户电话 |
price | int | 11 | 否 | 否 | 价格 |
photo | varchar | 255 | 否 | 否 | 照片 |
amenities | text | 0 | 否 | 否 | 便利设施 |
store_guide | text | 0 | 否 | 否 | 到店指南 |
hotel_introduction | longtext | 0 | 否 | 否 | 酒店介绍 |
hits | int | 11 | 是 | 否 | 点击数 |
praise_len | int | 11 | 是 | 否 | 点赞数 |
recommend | int | 11 | 是 | 否 | 智能推荐 |
create_time | datetime | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
comment表:
名称 | 类型 | 长度 | 不是null | 主键 | 注释 |
comment_id | int | 11 | 是 | 是 | 评论ID |
user_id | int | 11 | 是 | 否 | 评论人ID |
reply_to_id | int | 11 | 是 | 否 | 回复评论ID |
content | longtext | 0 | 否 | 否 | 内容 |
nickname | varchar | 255 | 否 | 否 | 昵称 |
avatar | varchar | 255 | 否 | 否 | 头像地址 |
create_time | timestamp | 0 | 是 | 否 | 创建时间 |
update_time | timestamp | 0 | 是 | 否 | 更新时间 |
source_table | varchar | 255 | 否 | 否 | 来源表 |
source_field | varchar | 255 | 否 | 否 | 来源字段 |
source_id | int | 10 | 是 | 否 | 来源ID |
系统中应用的开发工具总结如下表所示。
表5-1开发工具
名称 | 工具 | 版本 |
操作系统 IDE(Integrated Development Environment) | windows VisualStudio | 7/8/10 2010 |
服务器 | IIS | 9.10.7 |
数据库 浏览器 界面工具 | SQL Server 谷歌 Photoshop | 2008 6.0 2016 |
-
- 注册模块的实现
未注册用户在用户注册界面uerRegister.aspx.cs,输入新用户信息,点击注册后,新用户信息由界面传入控制层userController,调用addUser(user)方法,通过user的操作方法向数据库用户表插入用户信息,由于用户账号是自动生成的,若成功则反馈注册成功,并提示新用户的用户账号;若失败则回到登录界面,并显示登录失败。
注册界面如下所示。
图5-1用户注册流程图
用户注册界面如下图所示。
图5-2用户注册界面
用户注册关键代码如下所示。
/**
* 注册
* @param user
* @return
*/
@PostMapping("register")
public Map<String, Object> signUp(@RequestBody User user) {
// 查询用户
Map<String, String> query = new HashMap<>();
query.put("username",user.getUsername());
List list = service.select(query, new HashMap<>()).getResultList();
if (list.size()>0){
return error(30000, "用户已存在");
}
user.setUserId(null);
user.setPassword(service.encryption(user.getPassword()));
service.save(user);
return success(1);
}
/**
* 用户ID:[0,8388607]用户获取其他与用户相关的数据
*/
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "user_id")
private Integer userId;
/**
* 账户状态:[0,10](1可用|2异常|3已冻结|4已注销)
*/
@Basic
@Column(name = "state")
private Integer state;
/**
* 所在用户组:[0,32767]决定用户身份和权限
*/
@Basic
@Column(name = "user_group")
private String userGroup;
/**
* 上次登录时间:
*/
@Basic
@Column(name = "login_time")
private Timestamp loginTime;
/**
* 手机号码:[0,11]用户的手机号码,用于找回密码时或登录时
*/
@Basic
@Column(name = "phone")
private String phone;
/**
* 手机认证:[0,1](0未认证|1审核中|2已认证)
*/
@Basic
@Column(name = "phone_state")
private Integer phoneState;
/**
* 用户名:[0,16]用户登录时所用的账户名称
*/
@Basic
@Column(name = "username")
private String username;
/**
* 昵称:[0,16]
*/
@Basic
@Column(name = "nickname")
private String nickname;
/**
* 密码:[0,32]用户登录所需的密码,由6-16位数字或英文组成
*/
@Basic
@Column(name = "password")
private String password;
/**
* 邮箱:[0,64]用户的邮箱,用于找回密码时或登录时
*/
@Basic
@Column(name = "email")
private String email;
/**
* 邮箱认证:[0,1](0未认证|1审核中|2已认证)
*/
@Basic
@Column(name = "email_state")
private Integer emailState;
/**
* 头像地址:[0,255]
*/
@Basic
@Column(name = "avatar")
private String avatar;
/**
* 创建时间:
*/
@Basic
@Column(name = "create_time")
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private Timestamp createTime;
@Basic
@Transient
private String code;
}
主要由两部分组成,登录前的登录界面以及登录后的用户功能界面。登录界面,要求用户输入用户名和密码,当用户名和密码其中一个输入为空时,给出提示“用户名,密码不能为空”。获取用户名和密码后到数据库中查找,如果用户名存在,以及对应的密码正确,则登录成功,否则登录失败。登录失败后给出提示,并把焦点停在文本框中。登录成功后将该次会话的全局变量username设置为用户名。登录成功后进入会员的功能模块,主要有会员基本信息修改,已经发布景点信息管理,发布信息,和退出功能。退出功能是清除全局变量username的值,并跳回到首页。
登录流程图如下图所示。
图5-3登录流程图
用户登录界面如下图所示。
图5-4用户登录界面
登录关键代码如下所示。
/**
* 登录
* @param data
* @param httpServletRequest
* @return
*/
@PostMapping("login")
public Map<String, Object> login(@RequestBody Map<String, String> data, HttpServletRequest httpServletRequest) {
log.info("[执行登录接口]");
String username = data.get("username");
String email = data.get("email");
String phone = data.get("phone");
String password = data.get("password");
List resultList = null;
Map<String, String> map = new HashMap<>();
if(username != null && "".equals(username) == false){
map.put("username", username);
resultList = service.select(map, new HashMap<>()).getResultList();
}
else if(email != null && "".equals(email) == false){
map.put("email", email);
resultList = service.select(map, new HashMap<>()).getResultList();
}
else if(phone != null && "".equals(phone) == false){
map.put("phone", phone);
resultList = service.select(map, new HashMap<>()).getResultList();
}else{
return error(30000, "账号或密码不能为空");
}
if (resultList == null || password == null) {
return error(30000, "账号或密码不能为空");
}
//判断是否有这个用户
if (resultList.size()<=0){
return error(30000,"用户不存在");
}
User byUsername = (User) resultList.get(0);
Map<String, String> groupMap = new HashMap<>();
groupMap.put("name",byUsername.getUserGroup());
List groupList = userGroupService.select(groupMap, new HashMap<>()).getResultList();
if (groupList.size()<1){
return error(30000,"用户组不存在");
}
UserGroup userGroup = (UserGroup) groupList.get(0);
//查询用户审核状态
if (!StringUtils.isEmpty(userGroup.getSourceTable())){
String sql = "select examine_state from "+ userGroup.getSourceTable() +" WHERE user_id = " + byUsername.getUserId();
String res = String.valueOf(service.runCountSql(sql).getSingleResult());
if (res==null){
return error(30000,"用户不存在");
}
if (!res.equals("已通过")){
return error(30000,"该用户审核未通过");
}
}
//查询用户状态
if (byUsername.getState()!=1){
return error(30000,"用户非可用状态,不能登录");
}
String md5password = service.encryption(password);
if (byUsername.getPassword().equals(md5password)) {
// 存储Token到数据库
AccessToken accessToken = new AccessToken();
accessToken.setToken(UUID.randomUUID().toString().replaceAll("-", ""));
accessToken.setUser_id(byUsername.getUserId());
tokenService.save(accessToken);
// 返回用户信息
JSONObject user = JSONObject.parseObject(JSONObject.toJSONString(byUsername));
user.put("token", accessToken.getToken());
JSONObject ret = new JSONObject();
ret.put("obj",user);
return success(ret);
} else {
return error(30000, "账号或密码不正确");
}
}
用户登录/注册成功之后可以修改自己的基本信息。修改页面的表单中每一个input的name值都要与实体类中的参数相匹配,在用户点击修改页面的时候,如果改后用户名与数据库里面重复了,页面会提示该用户名已经存在了,否则通过Id来查询用户,并将用户的信息修改为表单提交的数据。
酒店信息功能需要考虑高并发,防止出现酒店重复预订状态显示出错等情况,特对酒店这一共享数据增加锁机制。在乐观锁、悲观锁以及线程锁中,综合考虑性能效率和错误的可接受性选择了乐观锁机制。乐观锁的实现方式是使用版本标识来确定读到的数据与提交时的数据是否一致,提交后修改版本标识,不一致时可以采取丢弃和再次尝试的策略。在数据库酒店表(对应酒店实体)设计中增加了version字段,每次数据提交时(更改酒店状态)会判断version是否匹配,若不匹配停止本次提交,若匹配则提交成功并增加version的值。
酒店信息功能整体流程:用户浏览酒店名称信息时,同时会显示酒店的状态,系统会在其显示详细信息的页面时便会判断酒店的状态,若景点状态为可资讯,则会显示酒店的链接按钮。在用户点击酒店信息按钮时,会先通过拦截器判断用户是否登录,若未登录,会跳转至登录页面,提示用户先登录,若为登录用户就会跳转至填写酒店信息的页面,填写好酒店信息之后,点击提交按钮,酒店预订成功之后返回提示信息,告知用户成功。
酒店信息流程图如下图所示。
图5-5酒店信息流程图
酒店信息界面如下所示。
图5-6酒店信息买界面
5.6酒店预订管理模块的实现
用户提交酒店后后台会自动生成酒店预订列表。从session中取出该用户信息,前台发起请求,将对应的用户信息、dingdanpId参数信息从前台传递酒店预订控制类里,匹配到create()方法,create()方法调用酒店预订逻辑层的createDingdan()方法获取数据,调用本类的getCartDingdanItem()方法得到酒店订单列表。
酒店预订管理界面如下所示。
图5-7酒店预订管理界面
@RequestMapping(value = "/del")
@Transactional
public Map<String, Object> del(HttpServletRequest request) {
service.delete(service.readQuery(request), service.readConfig(request));
return success(1);
}
@Transactional
public void delete(Map<String,String> query,Map<String,String> config){
StringBuffer sql = new StringBuffer("DELETE FROM ").append("`").append(table).append("`").append(" ");
sql.append(toWhereSql(query, "0".equals(config.get(FindConfig.GROUP_BY))));
log.info("[{}] - 删除操作:{}",table,sql);
Query query1 = runCountSql(sql.toString());
query1.executeUpdate();
}
5.7景点信息管理模块的实现
此页面的关键是编写景点信息,包括景点编号、照片、票价、地址、开放时间,景点名称等。单击提交按钮以完成信息的添加。如果未写入完整的景点信息,例如,如果未写入景点名称,系统将给出相应的错误提示,并且无法成功输入。数据以概念的形式以onsubmit =“return checkForm()”的形式写入以进行检查,checkForm()函数是一种用于写入数据的不同类型的校对方法,是不是为空也是经过form表单中的οnsubmit=”return checkForm()来检查。
景点信息流程图如下图所示。
图5-8景点信息流程图
景点信息管理界面如下图所示。
图5-9景点信息管理界面
景点添加界面如下图所示。
图5-10景点添加界面
景点信息管理关键代码如下所示。
@PostMapping("/add")
@Transactional
public Map<String, Object> add(HttpServletRequest request) throws IOException {
service.insert(service.readBody(request.getReader()));
return success(1);
}
@Transactional
public Map<String, Object> addMap(Map<String,Object> map){
service.insert(map);
return success(1);
}
public Map<String,Object> readBody(BufferedReader reader){
BufferedReader br = null;
StringBuilder sb = new StringBuilder("");
try{
br = reader;
String str;
while ((str = br.readLine()) != null){
sb.append(str);
}
br.close();
String json = sb.toString();
return JSONObject.parseObject(json, Map.class);
}catch (IOException e){
e.printStackTrace();
}finally{
if (null != br){
try{
br.close();
}catch (IOException e){
e.printStackTrace();
}
}
}
return null;
}
public void insert(Map<String,Object> body){
StringBuffer sql = new StringBuffer("INSERT INTO ");
sql.append("`").append(table).append("`").append(" (");
for (Map.Entry<String,Object> entry:body.entrySet()){
sql.append("`"+humpToLine(entry.getKey())+"`").append(",");
}
sql.deleteCharAt(sql.length()-1);
sql.append(") VALUES (");
for (Map.Entry<String,Object> entry:body.entrySet()){
Object value = entry.getValue();
if (value instanceof String){
sql.append("'").append(entry.getValue()).append("'").append(",");
}else {
sql.append(entry.getValue()).append(",");
}
}
sql.deleteCharAt(sql.length() - 1);
sql.append(")");
log.info("[{}] - 插入操作:{}",table,sql);
Query query = runCountSql(sql.toString());
query.executeUpdate();
}
5.8景点购票管理模块的实现
此页面的关键是编写景点购票,包括景点编号、订单号,景点名称,票价、购买数量、总金额、用户名等。单击提交按钮以完成信息的添加。如果未写入完整的景点信息,例如,如果未写入景点名称,系统将给出相应的错误提示,并且无法成功输入。数据以概念的形式以onsubmit =“return checkForm()”的形式写入以进行检查,checkForm()函数是一种用于写入数据的不同类型的校对方法,是不是为空也是经过form表单中的οnsubmit=”return checkForm()来检查。
景点购票管理流程图如下图所示。
图5-11景点购票管理流程图
景点购票界面如下图所示。
图5-12景点购票管理界面
景点购票管理关键代码如下所示。
@PostMapping("/set")
@Transactional
public Map<String, Object> set(HttpServletRequest request) throws IOException {
service.update(service.readQuery(request), service.readConfig(request), service.readBody(request.getReader()));
return success(1);
}
public Map<String,String> readConfig(HttpServletRequest request){
Map<String,String> map = new HashMap<>();
map.put(FindConfig.PAGE,request.getParameter(FindConfig.PAGE));
map.put(FindConfig.SIZE,request.getParameter(FindConfig.SIZE));
map.put(FindConfig.LIKE,request.getParameter(FindConfig.LIKE));
map.put(FindConfig.ORDER_BY,request.getParameter(FindConfig.ORDER_BY));
map.put(FindConfig.FIELD,request.getParameter(FindConfig.FIELD));
map.put(FindConfig.GROUP_BY,request.getParameter(FindConfig.GROUP_BY));
map.put(FindConfig.MAX_,request.getParameter(FindConfig.MAX_));
map.put(FindConfig.MIN_,request.getParameter(FindConfig.MIN_));
return map;
}
public Map<String,String> readQuery(HttpServletRequest request){
String queryString = request.getQueryString();
if (queryString != null && !"".equals(queryString)) {
String[] querys = queryString.split("&");
Map<String, String> map = new HashMap<>();
for (String query : querys) {
String[] q = query.split("=");
map.put(q[0], q[1]);
}
map.remove(FindConfig.PAGE);
map.remove(FindConfig.SIZE);
map.remove(FindConfig.LIKE);
map.remove(FindConfig.ORDER_BY);
map.remove(FindConfig.FIELD);
map.remove(FindConfig.GROUP_BY);
map.remove(FindConfig.MAX_);
map.remove(FindConfig.MIN_);
return map;
}else {
return new HashMap<>();
}
}
@Transactional
public void update(Map<String,String> query,Map<String,String> config,Map<String,Object> body){
StringBuffer sql = new StringBuffer("UPDATE ").append("`").append(table).append("`").append(" SET ");
for (Map.Entry<String,Object> entry:body.entrySet()){
Object value = entry.getValue();
if (value instanceof String){
sql.append("`"+humpToLine(entry.getKey())+"`").append("=").append("'").append(value).append("'").append(",");
}else {
sql.append("`"+humpToLine(entry.getKey())+"`").append("=").append(value).append(",");
}
}
sql.deleteCharAt(sql.length()-1);
sql.append(toWhereSql(query,"0".equals(config.get(FindConfig.LIKE))));
log.info("[{}] - 更新操作:{}",table,sql);
Query query1 = runCountSql(sql.toString());
query1.executeUpdate();
}
public String toWhereSql(Map<String,String> query, Boolean like) {
if (query.size() > 0) {
try {
StringBuilder sql = new StringBuilder(" WHERE ");
for (Map.Entry<String, String> entry : query.entrySet()) {
if (entry.getKey().contains(FindConfig.MIN_)) {
String min = humpToLine(entry.getKey()).replace("_min", "");
sql.append("`"+min+"`").append(" >= '").append(URLDecoder.decode(entry.getValue(), "UTF-8")).append("' and ");
continue;
}
if (entry.getKey().contains(FindConfig.MAX_)) {
String max = humpToLine(entry.getKey()).replace("_max", "");
sql.append("`"+max+"`").append(" <= '").append(URLDecoder.decode(entry.getValue(), "UTF-8")).append("' and ");
continue;
}
if (like == true) {
sql.append("`"+humpToLine(entry.getKey())+"`").append(" LIKE '%").append(URLDecoder.decode(entry.getValue(), "UTF-8")).append("%'").append(" and ");
} else {
sql.append("`"+humpToLine(entry.getKey())+"`").append(" = '").append(URLDecoder.decode(entry.getValue(), "UTF-8")).append("'").append(" and ");
}
}
sql.delete(sql.length() - 4, sql.length());
sql.append(" ");
return sql.toString();
} catch (UnsupportedEncodingException e) {
log.info("拼接sql 失败:{}", e.getMessage());
}
}
return "";
}
对任何系统而言,测试都是必不可少的环节,测试可以发现系统存在的很多问题,所有的软件上线之前,都应该进行充足的测试之后才能保证上线后不会Bug频发,或者是功能不满足需求等问题的发生。下面分别从单元测试,功能测试和用例测试来对系统进行测试以保证系统的稳定性和可靠性。
下表是景点管理功能的测试用例,检测了旅游网站景点管理中对景点信息的增加,删除,修改,查询操作是否成功运行。观察系统的响应情况,得出该功能也达到了设计目标,系统运行正确。
前置条件;用户登录系统。
表6-1 景点管理的测试用例
功能描述 | 用于景点管理 | |
测试目的 | 检测景点管理时的各种操作的运行情况 | |
测试数据以及操作 | 预期结果 | 实际结果 |
点击添加景点,必填项合法输入,点击保存 | 提示添加成功 | 与预期结果一致 |
点击添加景点,必填项输入不合法,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击修改景点,必填项修改为空,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击修改景点,必填项输入不合法,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击删除景点,选择景点删除 | 提示删除成功 | 与预期结果一致 |
点击搜索景点,输入存在的景点名 | 查找出景点 | 与预期结果一致 |
点击搜索景点,输入不存在的景点名 | 不显示景点 | 与预期结果一致 |
下表是酒店预订管理功能的测试用例,检测了酒店预订管理中对酒店信息的增加,删除,修改,查询操作是否成功运行。观察系统的响应情况,得出该功能也达到了设计目标,系统运行正确。
前置条件;用户登录系统。
表6-2 酒店预订管理的测试用例
功能描述 | 用于酒店预订管理 | |
测试目的 | 检测酒店预订管理时的各种操作的运行情况 | |
测试数据以及操作 | 预期结果 | 实际结果 |
点击添加酒店,必填项合法输入,点击保存 | 提示添加成功 | 与预期结果一致 |
点击添加酒店,必填项输入不合法,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击修改酒店,必填项修改为空,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击修改酒店,必填项输入不合法,点击保存 | 提示必填项不能为空 | 与预期结果一致 |
点击删除酒店,选择酒店信息删除 | 提示删除成功 | 与预期结果一致 |
点击搜索酒店,输入存在的酒店名 | 查找出酒店 | 与预期结果一致 |
点击搜索酒店,输入不存在的酒店名 | 不显示酒店 | 与预期结果一致 |
下表是购票管理功能的测试用例,检测了购票管理中购票单的操作是否成功运行。观察系统的响应情况,得出该功能也达到了设计目标,系统运行正确。
前置条件;用户登录系统。
表6-3 购票管理的测试用例
功能描述 | 用于购票管理 | |
测试目的 | 检测购票管理时各种操作的情况 | |
测试数据以及操作 | 预期结果 | 实际结果 |
未选择景点,点击提交 | 提示请选择景点 | 与预期结果一致 |
未输入文字,点击提交 | 提示请输入文字 | 与预期结果一致 |
未选择时间,点击提交 | 提示请选择时间 | 与预期结果一致 |
-
- 性能测试
使用阿里云PTS(Performance Testing Service)性能测试服务对线上系统进行压力测试。线上服务器环境为:1核心CPU,1G内存,1Mbps公网带宽,Centos7.0操作系统。
压测过程中使用了2台并发机器,每台机器20个用户并发,对系统主页,登录,数据查询和数据维护等模块进行并发访问,测试结果是有40个用户并发时,数据管理相关页面的响应时间甚至达到了7s,通过查看服务器出网流量发现已经达到1381kb/s,可以看出服务器的带宽已经达到峰值,如果系统使用5Mbps的带宽,系统的响应时间和TPS将会大大增加。在整个测试的过程中,CPU的使用率占用仅8%,也提现出带宽瓶颈对系统的影响非常严重。
建造网站需要多少钱近期时事新闻博业建站网正品购物app排行榜前十名白云区做网站网站设计的国际专业流程是什么长春专业做网站制作小程序的流程免费网站安全软件网络品牌推广策略物联网系统开发客服管理系统大连seo建站网站建设服务商有哪些在线作图加字图片龙岩市新罗区疫情最新消息软件外包专业网站建设公司小程序个人网页制作成品 设计怎么编写app软件免费b站推广网站入口2020延安网站制作郑州企业建设网站网站seo策划方案设计b站推广网站2024mmm兰州网页推广网站赚钱网站制作二维码网络营销中的seo是指私密浏览器在线观看传奇霸业网页版