图书销售系统 —— 书窝

书窝线上地址 GitHub地址 由于时间有限,事务较重,系统实现只好从简。


需求分析

需交课程设计报告和软件(源代码)。课程设计报告将存档。报告内容包括:需求分析、算法思想描述、数据流图、E-R图、数据字典、程序结构、收获与体会等。 功能要求:实现一个基于web的网上图书的销售管理系统,能提供多种条件的查询,还应具有会员管理、意见反馈、销售分析等功能。将留言板、图表分析、文件上传等思想纳入其中。

用户系统

需要用户系统,用户是系统的根源,是数据的源头,该系统的用户就是要求中所说的会员,所以需要提供如下功能:

  • 用户注册
  • 用户登录
  • 用户信息查看
  • 用户既可以是卖家,亦可以是买家

用户留言(信息反馈)

为了逐渐完善系统,提供用户与建站者的交流通道,并且该通道不仅限于用户与建站者,用户与用户之间也能够互相交流,提高趣味性。具体功能如下:

  • 用户留言
  • 留言查看

卖家买家系统

买卖离不开卖家买家,该系统亦是如此,需要如下功能列表:

  • 卖家
  • 图书上架
  • 图书修改
  • 图书查看
  • 销售情况图表
  • 买家
  • 图书查看购买
  • 买书记录查看
  • 图书评论与删除

设计文档

该部分将涉及整个系统从无到有的设计思路,自底向上有 1. 数据库选择和设计 2. 服务器框架选择和设计 3. 前端框架选择和设计

并且将针对具体细节给出相关表示,如数据字典,ER图,数据流图,算法设计等。

数据库选择和设计

选择与原因

该系统我选择了MySQL数据库,具体原因如下: 1. MySQL十分轻量,相比课堂上讲的SQL Server数据库,一个安装包2GB,MySQL的500MB简直小巫见大巫(最近看到属于NoSQLMongoDB居然只有100MB)。 2. 短时间需要完成该系统,同时对于MySQL十分熟悉,之前做在线订票系统,在线考试系统等都是建立在MySQL上。 3. 具有SQLYog这种强大方便的图形操作软件,轻松导入导出数据,轻松连接远程服务器传递数据。

基于以上原因,因此选择了MySQL

数据字典

用户

字段类型备注
usernamevarchar(12)用户名,唯一, 6-12位
emailvarchar(20)电子邮箱,唯一
passwordvarchar(12)密码,6-12位
registerDatedate注册日期

留言

字段类型备注
usernamevarchar(12)留言者用户名
contenttext留言内容
datetimedatetime留言时间

图书

字段类型备注
bookIDchar(8)图书ID,唯一,如BK123456
titlevarchar(25)书名
authorvarchar(25)作者
pressvarchar(25)出版社
pricedecimal(10,1)价格,保留小数点后一位
quantityint(11)库存量(本)
imagemediumblob封面图片
sellervarchar(12)卖家用户名
importDatedate上架日期

图书评论

字段类型备注
usernamevarchar(12)评论人用户名
bookIDchar(8)被评论图书
datetimedatetime评论时间
contenttext评论内容

交易 因为交易完成后,卖家仍然可以对图书进行修改,所以我将买进时图书信息都存放在此表中,表示买进时图书的信息。

字段类型备注
tradeIDchar(10)交易记录ID,唯一,如TD12345678
tradeDatedate交易日期
buyervarchar(12)买家用户名
bookIDchar(8)书本ID
titlevarchar(25)书名
authorvarchar(25)作者
pressvarchar(25)出版社
pricedecimal(10,1)买进价格
quantityint(11)库存
imagemediumblob封面
sellervarchar(12)卖家

E-R图

服务器框架选择和设计

选择与原因

作为JS动态语言的受益者,我服务器肯定就选择nodejs了。 1. 未使用过nodejs开发一个相对完整的系统,打算就将该系统作为入门了。 2. nodejs非常适合IO密集型的应用,采用的是异步事件队列的机制。 3. JS语言简洁灵活有趣。

nodejsexpress

  • nodejs 基于ChromeV8引擎,以JS作为宿主语言的一个虚拟环境,JSNodeJs可以看做是JavaJVMNodeJs现在正在不断发展中,目标是与Java一样,能够形成一套十分完备的库,目前NodeJs的生态环境非常好,第三方模块层出不穷,而且由于npm,这些包是否方便管理与下载。
  • express 介绍完nodejs,那么express是什么呢?expressnodejs的一个第三方Web框架,开发者可以用该框架非常方便有效的建立HTTP服务。

结构设计

文件结构
`db/` 所有数据库操作代码 `public/` 静态资源库,如js,css `routes/` 路由操作diam,对应url `utils/` 工具包代码 `views/` Jade模板文件 `app.js` 系统入口 `mysql.sql` mysql数据文件,可导入
部分数据流图
  • 用户留言
  • 卖家添加图书
  • 买家购书
部分算法设计

首先对于留言评论,防止恶意用户刷留言评论,设置了定时销毁器。 算法设计如下:

var _timer = {};

var Timer = {
    set : function (key,mill) { //设置定时器的关键字与销毁时间
        this.remove(key);
        _timer[key] ={mill:mill};
        _timer[key].code = setTimeout(function () {
            delete _timer[key];
        },mill);
    },
    isExist : function(key){
        return !!_timer[key];
    },
    remove : function(key){
        if(this.isExist(key)){
            clearTimeout(_timer[key].code);
            delete _timer[key];
            return true;
        }
        return false;
    }
}

该系统具有销售分析,如某卖家所有销售情况的折线图,x轴表示日期,y轴表示该日售出书籍数目。 算法设计如下:

// all 表示所有卖家销售记录,0-N 时间从现在到以前
function makeLineChart(all) {
    if(all==null || all.length==0)
        return;
    //返回数据中 keys表示日期数组,data表示对应keys日期的销售量
    return all.reduceRight((p,n)=>{ // 从右向左归并
        var date = n.tradeDate;
        if(p.keys[p.keys.length-1]!=date){
            p.keys.push(date);
            p.data.push(1);
        }else{
            p.data[p.data.length-1]++;
        }
        return p;         	
    },{keys:[],data:[]});
}

销售分析中还具有周最受欢迎图书饼图,表示一周内各个图书销售情况。 算法设计如下:

// all 表示所有卖家销售记录,0-N 时间从现在到以前
function makePieChart(all) {
    if(all==null || all.length==0)
        return;
    var pivot = all[all.length-1];  // 取出最后一个交易记录,也就是最近的
    var end = new Date(pivot.tradeDate).setHours(24); // 根据最近时间的时间得到第二天0点时间
    var start = new Date(end - 1000*60*60*24*7); // 减去7天毫秒数,得到七天前时间
    var data = {keys:[pivot.title],data:[1]},titleMap = {};
    titleMap[pivot.title]=0; //初始化参数
    for(var i=all.length-2;i>=0;i--){
        if(new Date(all[i].tradeDate)<start) // 如果时间再七天之前,跳出循环
            break;
        var title = all[i].title;
        if(titleMap[title]==null){
            titleMap[title] = data.keys.length;
            data.keys.push(title);
            data.data.push(1);
        }else
            data.data[titleMap[title]]++;
    }
    return data;
}

前端框架选择和设计

选择与原因

针对于用户之间接触的前端界面,我选择了Bootstrap3.0 UIjQuerymarked.jshighlight.jspace.js,下面做出相关介绍与说明:

  • BootStrap3.0 一套完备的UI框架,包括美观的css样式和一些基于jQuery的组件。可以让开发者用最少的时间建立一个美观的界面。
  • jQuery 因为BootStrap组件需要,而且能够方便进行DOM操作,强大的选择器与链式调用。
  • marked.js 一个将markdown文本翻译为HTML的库,api简单。
  • highlight.jsmarked.js配套使用,能够将代码段解析为具有样式类的库。
  • pace.js 通过检查浏览器加载的状态,判断文档加载进度,并且提供了一系列的css样式,能够在页面加载的时候显示页面加载进度。

文件结构

public/
├── javascripts/
│   ├── addbook.js
│   ├── Ajaxdelete.js
│   ├── indexjs.js
│   ├── msgAnimate.js
│   ├── msgjs.js
│   ├── popImage.js
│   ├── search.js
│   ├── selljs.js
│   ├── utils.js
└── stylesheets/
    ├── bootstrap/
    ├── style.less
    ├── style.css
    ├── hljs-github-min.css
    └── markdown.less

stylesheets/文件夹放的是样式文件,其中的style.less是入口,style.css是利用node模块压缩后真正传送的样式文件

特殊功能说明

  1. 提示框动态固定显示
    方法一:(固定width)
.messages{
  .box-shadow(0px 0px 8px 3px #bbb);
  z-index: 10000;
  position: fixed;
  width: 340px;
  height: 50px;
  left: 50%;
  top : -50px;
  margin-left: -170px;

  -webkit-transition: all 2s;
  -moz-transition: all 2s;
  -ms-transition: all 2s;
  -o-transition: all 2s;
  transition: all 2s;
  opacity:0;
}
window.onload = function () {
    $('.messages')
        .css({'transform':'translate(0,120px)',opacity:1})
}

方法二:width自适应

.messages{
  .box-shadow(0px 0px 8px 3px #bbb);
  z-index: 10000;
  position: fixed;
  left: 50%;
  top : 0px;
  -webkit-transform: translate(-50%,-50%);
  -moz-transform: translate(-50%,-50%);
  -ms-transform: translate(-50%,-50%);
  -o-transform: translate(-50%,-50%);
  transform: translate(-50%,-50%);

  -webkit-transition: all 2s;
  -moz-transition: all 2s;
  -ms-transition: all 2s;
  -o-transition: all 2s;
  transition: all 2s;
  opacity:0;
}
window.onload = function () {
    $('.messages')
        // .css({'transform':'translate(0,120px)',opacity:1})
        .css({'margin-top':'140px',opacity:1})
}
  1. 跳至评论,评论区高亮
@keyframes blinking {
    0% {
        //opacity: 0;
    }
    30%{
        background-color: #fcf8e3;
    }
    100% {
        background-color: #fcf8e3;
        //opacity: 1;
    }
}
.blink{
    .animation(blinking 2.8s)
}
$('[role=link-msg]').click(function (e) {
        var x = $(this.hash).next().children().removeClass('blink')
        setTimeout(function () {
            x.addClass('blink').children('textarea').focus();
        },0)
    });
  1. markdown编辑区可粘贴网络图片
input.onpaste = function (e) {
        var clipboardData, pastedData;

        // Get pasted data via clipboard API
        clipboardData = e.clipboardData || window.clipboardData;
        var text = clipboardData.getData('text/plain');
        if(!text){
            var img = clipboardData.getData('text/html');
            img.replace(/<img.+src="(.+?)"/g,(m,c)=>{
                e.preventDefault();
                // 调用
                document.execCommand('insertText', false, "![ClipboardImage]("+c+")");
            })
        }
    };
  1. 代码段右上角显示语言

系统截图


收获与体会

  • 感受到了nodejsexpress的魅力
  • 增强了系统的整体把控架构能力
  • 掌握了一些常见具体问题的处理方式
  • 不足在于后端异步结构代码比较冗杂,难于管理

参考资料