nodejs中利用expresss脚手架和bootstrap,数据库mongodb搭建的留言板案例

## 1. 先打开编辑器,创建一个项目

## 2. 再打开cmd命令提示符下载express脚手架

        express   项目名   --view=ejs 或express   -e    项目名

## 3. 在cmd中进入项目名(myapp)下载所需的依赖

         cd myapp  --------->cnpm  install 

## 4. 在下载mongoose(前提你电脑上要安装数据库的插件)

        cn cnpm mongoose  --save

## 5. 在myapp项目中在创建一个文件夹,里面在新建三个文件

         文件夹名  lib    ------->三个文件名  mongoose.js    schema,js     appModel.js

## 6. 在mongoose.js 中连接数据库

           //先引入mongoose模块

         var mongoose=require("mongoose");

          //连接数据库服务器

         mongoose.connect("mongodb://localhost/数据库名(bao)",function(error){

                       if(error){

                                 console.log("数据库连接失败")

                            }else{

                                 console.log("数据库连接成功")

                            }

        })

         //导出

      module.exports=mongoose;

## 7. 在schema.js 文件中定义schema

       //引入mongoose.js文件

     var mongoose=require("./mongoose.js")      //这里的.js可省略不写

     //定义schema

    var   schema=mongoose.Schema({

                 //这里是数据库自己创建的属性名:他的属性类型   如:

id:String,

name:String,

age:Number,

tel:Number

     })

     //导出

      module.exports=schema;

## 8. 在appModel.js 文件中定义模型

             //引入mongoose.js 文件

             var mongoose=require("./mongoose");

              //引入schema.js 文件

             var schema=require("./schema");

             //定义模型

              var appModel=mongoose.model("appModel",schema,"数据库中的集合名(app)");

              //导出

               module.exports=appModel;

## 9. 在views文件夹中找到index.els编辑

                  //先引入bootstrap的css样式,js样式,不过要把jquery的js插件引入在bootstrap.js的前面,bootstrap连接的样式可通过本地下载,也可网上连接地址即可

                         //本地下载    cnpm install bootstrap   --save

                //通过Bootstrap官网找到我们需要的样式直接复制粘贴


content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">

留言板练习

.box{

margin: 100px auto;

}

table{

margin: 100px auto;

}

table th{

text-align: center;

}


学号:

姓名:

年龄:

电话:

添加

重置


学号

姓名

年龄

电话

操作

1

张三

20

删除

修改

2

李四

20

删除-->

修改-->

全部删除


×

修改数据

学号:

姓名:

年龄:

电话:

关闭

保存

## 10. 在routes路由文件夹中创建两个文件   

                 index.js 渲染页面的二级路由   users.js增删改查的接口

## 11. 在app.js中引入这两个二级路由文件


var indexRouter = require('./routes/index');

var usersRouter = require('./routes/users');

app.use('/', indexRouter);

app.use('/api', usersRouter);

## 12. 在index.js文件中


//引入express模块

var express = require('express');

//获取路由

var router = express.Router();

//引入model.js文件

var mm=require("../lib/appModel");

//设置渲染页面路由

/* GET home page. */

//主页从数据库找到数据,返回前台,并渲染

router.get('/', function(req, res, next) {

mm.find({},function (err,docs) {

if(err){

console.log("查找数据库数据失败")

}else{

res.render('index', {

data:docs

});

}

})

});

//导出路由

module.exports = router;

## 13. 在index.ejs 文件中渲染的部分

<%for(var i=0;i

<%=data[i].id%>

<%=data[i].name%>

<%=data[i].age%>

<%=data[i].tel%>


" type="button" class="del btn btn-warning">删除

修改

<%}%>

</tbody>

## 14. 在users.js二级路由文件中设置增删改查

//增(也可以说是添加)

//引入express模块

var express=require("express");

//获取路由

var router=express.Router();

//引入model.js文件

var mm=require("../lib/model");

//设置数据接口路由

//添加数据到数据库

router.get("/write",function (req, res,next) {

var da=req.query;//获取前台请求的数据,返回来是一个对象

console.log(da)//{id:xh.name:xm,age:nl,tel:tel}

//实例化

var aa=new mm(da);//mm({id:..,name:..,age:..,tel:...})

//添加

aa.save(function (err) {

if(err){//如果失败就输出

res.send({

code:1,

message:"添加失败"

});

}else{//否则

res.send({

code:0,

message:"添加成功"

})

}

})

});

//导出

module.exprots=router;

## 15. 在index.ejs中的 增 代码

$(".add").click(function () {

//获取学号,姓名,年龄,性别的内容

var xh=$("#xh").val();

var xm=$("#xm").val();

var nl=$("#nl").val();

var tel=$("#tel").val();

// alert(xh); alert(xm); alert(nl); alert(tel)检测获取数据成功

$.ajax({

url:"/api/write",

data:{

id:xh,

name:xm,

age:nl,

tel:tel

},

success:function (ret) {//成功返回数据

console.log(ret)

},error:function (msg) {//失败返回数据

console.log(msg)

}

})

});

## 16. 解决点击添加按钮后,要刷新一次才能把添加的内容显示在页面

//1. 可以是添加完,是整个页面刷新("location.href="/"  "),但是有因为下面的表格式要异步更新的,所以不能整个页面刷新

//2. 我们要让它点击添加按钮后,tbody的里面内容变空$("tbody").html(" "),在通过ajax读取后台数据库里的数据,把他渲染,添加到当前的tbody里

//读取数据库里的数据

//读取数据库中的数据

router.get("/read",function (req, res, next) {

mm.find({},function (err,docs) {

if(err){

res.send({

code:2,

message:"读取数据失败"

});

}else{

res.send({

code:0,

data:docs,

message:"读取数据成功"

})

}

})

});

//通过ajax渲染添加到页面

/*添加数据,,,用ajax*/

$(".add").click(function () {

//获取学号,姓名,年龄,性别的内容

var xh=$("#xh").val();

var xm=$("#xm").val();

var nl=$("#nl").val();

var tel=$("#tel").val();

// alert(xh); alert(xm); alert(nl); alert(tel)检测获取数据成功

$.ajax({

url:"/api/write",

data:{

id:xh,

name:xm,

age:nl,

tel:tel

},

success:function (ret) {//成功返回数据

console.log(ret);

if(ret.code==0){//在这里判断后

//当后台返回数据是写入数据成功,让当前的tbody中的内容变空,

$("tbody").html("");

//在通过ajax读取后台在数据库中的内容,在渲染到tbody中

add(); //封装的函数

}

},error:function (msg) {//失败返回数据

console.log(msg)

}

})

});

//在这里因为后面都需要用到,所以封装到函数里,后面用的话直接调用就可以了

//包装函数

function add(){

$.ajax({

url:"/api/read",

success:function(ret){

if(ret.code==0){

var str="";

var data=ret.data;

for(var i in data){

str+=`

${data[i].id}

${data[i].name}

${data[i].age}

${data[i].tel}

删除

修改

`

}

$("tbody").html(str);

//当渲染后(局部更新后)让当前的学号,姓名,年龄,电话框都为空

$("#xh").val("");

$("#xm").val("");

$("#nl").val("");

$("#tel").val("");

}

},error:function (msg) {

console.log(msg)

}

})

}

## 17. 全部删除   users.js

//全部删除数据

router.get("/delAll",function (req, res, next) {

mm.remove({},function (err) {

if(err){

res.send({

code:3,

message:"全部删除失败"

});

}else{

res.send({

code:0,

message:"全部删除成功"

})

}

})

});

## 18. 全部删除   index.ejs

//全部删除

$(".delAll").click(function () {

$.ajax({

url:"/api/delAll",

success:function (ret) {

console.log(ret);

if(ret.code==0){

add();

}

},error:function (msg) {

console.log(msg);

}

})

});

## 19. 单行删除   users.js

//单行删除

router.get("/del",function (req, res, next) {

//获取前台请求的数据

var data=req.query;

mm.remove(data,function (err) {

if(err){

res.send({

code:4,

message:"单行删除失败"

});

}else{

res.send({

code:0,

message:"单行删除成功"

})

}

})

});

## 20. 单行删除   index.ejs

//单行删除,需要用到事件委托

$("tbody").on("click",".del",function () {

// alert(111)

//获取_id的值,来确定当前行的位置(如:id,name,age,del都可)

var a=$(this).attr("v");

// alert(a);//检测当前弹出的_id是否不同

$.ajax({

url:"/api/del",

data:{

_id:a

},

success:function (ret) {

console.log(ret);

if(ret.code==0){

add();

}

},error:function (msg) {

console.log(msg);

}

})

});

## 21. 修改数据  users.js

//修改数据库内容

router.get("/xg",function (req, res, next) {

//获取前台请求数据

var data=req.query;

console.log(data);

//获取前台的_id

console.log(data.f);

//获取前台的数据请求

console.log(data.da);

//修改数据

mm.update(data.f,data.da,function (err) {

if(err) {

res.send({

code: 5,

message: "修改数据失败"

});

}else{

res.send({

code:0,

message:"修改数据成功"

});

}

})

});

## 22. 修改数据  index.ejs

// 当点击修改时,获取他的兄弟元素删除的_id,渲染的都需要用到事件委托

$("tbody").on("click",".xg",function () {

var a=$(this).siblings().attr("v");

// alert(a)//检测他兄弟删除的_id

$(".bc").attr("v",a);//将他兄弟删除的属性也添加成了自己的属性

});

// 当点击弹框保存时,修改数据,有点问题,一次只能修改一次,下次要刷新

$(".bc").click(function () {

//attr一个参数是获取,两个参数是设置

var _id=$(this).attr("v");

alert(_id)

//获取弹框里的学号,姓名,年龄,性别的内容

//注意:这里要用class名,id名只能用一次

var xh=$(".xh").val();

var xm=$(".xm").val();

var nl=$(".nl").val();

var tel=$(".tel").val();

// alert(xh); alert(xm); alert(nl); alert(tel);

//ajax

$.ajax({

url:"/api/xg",

data:{

f:{

_id:_id

},

da:{

id:xh,

name:xm,

age:nl,

tel:tel

}

},

success:function (ret) {

console.log(ret);

if(ret.code==0){

$("tbody").html("");

add();

}

},error:function (msg) {

console.log(msg)

}

})

});

## 23. 到这里项目就结束了,说明一下:我也是一个刚学的小白,这个修改数据里有点小问题,等你自己去发现哟! 

             有哪位大佬解决了这个bug,请留言告诉我哟,在此不胜感激!!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容