使用MySQL数据库持久化:
Node.js学生管理系统(Express+MySQL)
使用MongoDB数据库持久化:
Node.js学生管理系统(Express+MongoDB)
目录
一、完成功能:实现对“学生信息”的增删改查。并且实现了数据持久化,
二、使用到的技术:基于Express框架和art-template模板引擎,并且将路由模块分离出来,更利于开发和维护
一、完成功能:实现对“学生信息”的增删改查。并且实现了数据持久化,
虽然是存在json文件中,但是,这样更可以锻炼自我的编码能力(emmmmm),虽然页面难看了点
二、使用到的技术:基于Express框架和art-template模板引擎,并且将路由模块分离出来,更利于开发和维护
"art-template": "^4.12.2",
"body-parser": "^1.18.3",
"bootstrap": "^4.1.3",
"express": "^4.16.3",
"express-art-template": "^1.0.1",
三、编写代码
1.首先初始化项目(package.json)
npm init -y
2.安装要用到的包
# Express框架
npm i -S express
# 模板引擎
npm i -S art-template
# 模板引擎与Express的关联包
npm i -S express-art-template
# post请求解析工具
npm i -S body-parser
3.编写核心模块app.js
/**
* app.js 入门模块
* 职责:
* 创建服务
* 做一些服务相关配置
* 模板引擎
* body-parser 解析表单 post 请求体
* 提供静态资源服务
* 挂载路由
* 监听端口启动服务
*/
var express = require('express');
var router = require('./router');
var bodyParser = require('body-parser');
var app = express();
app.use('/node_modules/',express.static('./node_modules/'));
app.use('/public',express.static('./public/'));
// 配置模板引擎和 body-parser 一定要在 app.use(router) 挂载路由之前
// parse application/x-www-form-urlencoded
app.engine('html',require('express-art-template'));
app.use(bodyParser.urlencoded({extended:false}));
// parse application/json
app.use(bodyParser.json());
//挂载路由到app服务中
app.use(router);
app.listen(8888,function () {
console.log('running 8888 ....')
})
4.路由模块router.js
/**
* router.js 路由模块
* 职责:
* 处理路由
* 根据不同的请求方法+请求路径设置具体的请求处理函数
* 模块职责要单一,不要乱写
* 我们划分模块的目的就是为了增强项目代码的可维护性
* 提升开发效率
*/
var fs = require('fs');
var Student = require('./student');
// Express 提供了一种更好的方式
// 专门用来包装路由的
var express = require('express');
// 1、创建一个路由容器
var router = express.Router();
//2.把路由挂载到router路由中
/*
渲染学生列表界面
*/
router.get('/students', function (req, res) {
if (req.query.id != null && req.query.id!=undefined &&req.query.id!='' ){
Student.findById(req.query.id,function (err,student) {
if (err) {
return res.status(500).send('Server error');
}
if (student ==undefined){
res.render('index.html', {
count:0,
student: student
})
}else {
res.render('index.html', {
count:1,
student: student
})
}
})
}else {
Student.find(function (err, students) {
if (err) {
return res.status(500).send('Server error');
}
if (students.length >= 3) {
var top = [
students[0],
students[1],
students[2],
]
}
res.render('index.html', {
top: top,
students: students
})
})
}
})
/**
* 渲染添加学生的页面
*/
router.get('/students/new', function (req, res) {
res.render('new.html');
});
/*
* 处理添加学生
*/
router.post('/students/new', function (req, res) {
Student.save(req.body, function (err) {
if (err) {
return res.status(500).send('Server error');
}
res.redirect('/students')
})
})
/*
* 渲染编辑学生页面
*/
router.get('/students/edit', function (req, res) {
// 1. 在客户端的列表页中处理链接问题(需要有 id 参数)
// 2. 获取要编辑的学生 id
//
// 3. 渲染编辑页面
// 根据 id 把学生信息查出来
// 使用模板引擎渲染页面
Student.findById(parseInt(req.query.id), function (err, student) {
if (err) {
return res.status(500).send('Server error.');
}
res.render('edit.html', {
student: student
})
})
})
/*
* 处理编辑学生
*/
router.post('/students/edit', function (req, res) {
// 1. 获取表单数据
// req.body
// 2. 更新
// Student.updateById()
// 3. 发送响应
Student.updateById(req.body, function (err) {
if (err) {
return res.status(500).send('Server error.')
}
res.redirect('/students')
})
})
/*
* 处理删除学生
*/
router.get('/students/delete', function (req, res) {
// 1. 获取要删除的 id
// 2. 根据 id 执行删除操作
// 3. 根据操作结果发送响应数据
Student.deleteById(req.query.id, function (err) {
if (err) {
return res.status(500).send('Server error.')
}
res.redirect('/students')
})
})
// 3. 把 router 导出
module.exports = router;
5.编写学生处理模块student.js
/**
* student.js
* 数据操作文件模块
* 职责:操作文件中的数据,只处理数据,不关心业务
*
* 这里才是我们学习 Node 的精华部分:奥义之所在
* 封装异步 API
*/
var fs = require('fs');
//统一路径管理
var dbPath = './db.json';
/**
* 获取学生列表
* @param callback
*/
exports.find = function (callback) {
fs.readFile(dbPath, 'utf8', function (err, data) {
if (err) {
return callback(err);
}
callback(null, JSON.parse(data).students);
})
}
/**
* 根据id 获取学生信息对象
* @param id
* @param callback
*/
exports.findById = function (id, callback) {
fs.readFile(dbPath, 'utf8', function (err, data) {
if (err) {
return callback(err);
}
var students = JSON.parse(data).students;
var ret = students.find(function (item) {
return item.id === parseInt(id);
})
callback(null, ret);
})
}
/**
* 添加保存学生
* @param {Object} student 学生对象
* @param {Function} callback 回调函数
*/
exports.save = function (student, callback) {
fs.readFile(dbPath, 'utf8', function (err, data) {
if (err) {
return callback(err);
}
var students = JSON.parse(data).students;
//添加id ,唯一不重复
student.id = students[students.length - 1].id + 1;
//把用户传递的对象保存到数组中
students.push(student);
var fileData = JSON.stringify({
students: students
})
//把字符串保存到文件中
fs.writeFile(dbPath,fileData, function (err) {
if (err) {
//错误就是把错误对象传递给它
return callback(err)
}
//成功就没错,所以错误对象是null
callback(null);
})
});
}
/**
* 更新学生
*/
exports.updateById = function (student, callback) {
fs.readFile(dbPath, 'utf8', function (err, data) {
if (err) {
return callback(err)
}
var students = JSON.parse(data).students;
//注意:这里记得把id统一转换为数字类型
student.id = parseInt(student.id);
// 你要修改谁,就需要把谁找出来
// EcmaScript 6 中的一个数组方法:find
// 需要接收一个函数作为参数
// 当某个遍历项符合 item.id === student.id 条件的时候,find 会终止遍历,同时返回遍历项
var stu = students.find(function (item) {
return item.id === student.id
})
// 遍历拷贝对象
for (var key in student) {
stu[key] = student[key];
}
// 把对象数据转换为字符串
var fileData = JSON.stringify({
students: students
})
// 把字符串保存到文件中
fs.writeFile(dbPath,fileData, function (err) {
if (err) {
return callback(err)
}
callback(null)
})
})
}
/**
* 删除学生
*/
exports.deleteById = function (id, callback) {
fs.readFile(dbPath, 'utf8', function (err, data) {
if (err) {
return callback(err);
}
var students = JSON.parse(data).students
// findIndex 方法专门用来根据条件查找元素的下标
var deleteId = students.findIndex(function (item) {
return item.id === parseInt(id)
})
// 根据下标从数组中删除对应的学生对象
students.splice(deleteId, 1)
// 把对象数据转换为字符串
var fileData = JSON.stringify({
students: students
})
// 把字符串保存到文件中
fs.writeFile(dbPath, fileData, function (err) {
if (err) {
// 错误就是把错误对象传递给它
return callback(err)
}
// 成功就没错,所以错误对象是 null
callback(null)
})
})
}
6.视图index.html
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<!-- 上述3个meta标签*必须*放在最前面,任何其他内容都*必须*跟随其后! -->
<meta name="description" content="">
<meta name="author" content="">
<link rel="icon" href="../../favicon.ico">
<title>学生管理系统</title>
<!-- Bootstrap core CSS -->
<link href="/node_modules/bootstrap/dist/css/bootstrap.min.css" rel="stylesheet">
<!-- Custom styles for this template -->
<link href="/public/css/main.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="navbar-header">
<a class="navbar-brand" href="#">学生管理系统</a>
</div>
<div class="row">
</div>
<div class="col-sm-9 col-sm-offset-3 col-md-12 col-md-offset-2 main">
<h1 class="page-header">Top3</h1>
<div class="row placeholders">
{{ each top }}
<div class="col-xs-6 col-sm-4 placeholder">
<img src="/public/img/head.jpg"
width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
<h4>{{ $value.name }}</h4>
<span class="text-muted">{{ $value.hobbies }}</span>
</div>
{{ /each }}
</div>
<h2 class="sub-header">学生列表</h2>
<form class="form-inline" action="/students">
<div class="form-group">
<label for="id" style="margin:5px">搜索:</label>
<input type="text" style="margin:5px ;width: 400px" class="form-control" name="id" id="id"
placeholder="请输入ID">
</div>
<div class="form-inline">
<button type="submit" class="btn btn-default" style="width: 100px ;margin:5px">搜索</button>
<a class="btn btn-success" href="/students/new">添加学生</a>
</div>
</form>
<div class="table-responsive" style="margin-top: 50px">
<table class="table table-striped">
<thead>
<tr>
<th>#</th>
<th>姓名</th>
<th>性别</th>
<th>年龄</th>
<th>爱好</th>
<th>操作</th>
</tr>
</thead>
<tbody>
{{if count == 0}}
<tr>
</tr>
<td>
未找到该ID,请重新查询!
</td>
</tbody>
{{else if count == 1}}
<tr>
<td>{{ student.id }}</td>
<td>{{ student.name }}</td>
{{if student.gender==0 }}
<td>男</td>
{{else}}
<td>女</td>
{{/if}}
<td> {{ student.age }}</td>
<td>{{ student.hobbies }}</td>
<td>
<a href="/students/edit?id={{ student.id }}">编辑</a>
<a href="javascript:if(confirm('确实要删除?'))location='/students/delete?id={{ student.id }}'">删除</a>
</td>
</tr>
{{else}}
{{each students}}
<tr>
<td>{{ $value.id }}</td>
<td>{{ $value.name }}</td>
{{if $value.gender==0 }}
<td>男</td>
{{else}}
<td>女</td>
{{/if}}
<td> {{ $value.age }}</td>
<td>{{ $value.hobbies }}</td>
<td>
<a href="/students/edit?id={{ $value.id }}">编辑</a>
<a href="javascript:if(confirm('确实要删除?'))location='/students/delete?id={{ $value.id }}'">删除</a>
</td>
</tr>
{{ /each }}
{{/if}}
</tbody>
</table>
</div>
</div>
</div>
</div>
</body>
</html>