NodeJS_05

 

NodeJS七天课程学习笔记_第5天

课程内容概要:

1. pakcage-lock.json的细节

2. Array的新增方法find内部细节

3. MongoDB的安装、配置和使用

4. CRUD增删改查

5. Promise异步编程

推荐了一个Sublime Text 插件: HTML-CSS-JS Prettify

sublime 通过命令,直接手动设置当前文档的高亮语法种类

Sublime同时选中多行内容等实用操作:
1. ctrl + command + 上下箭头,作用是:上移或下移当前行
2. command + D, 作用是:选中光标所在单词,重复按可同时选中多个相同的单词
3. alt + 左右箭头, 作用是:跳至单词的起始或结束位置
4. ctrl + 左右箭头, 切换屏幕,如全屏的视频 与 编辑器之间切换

NodeJS七天课程主讲老师:李鹏周

JS 奥义: 单线程+事件循环
哪怕setTimeout延迟是0秒,甚至不传第2个延时的参数,
也一定要等主线程运行结束后,才执行定时器中的方法
setTimeout(function(){NSLog(‘keke’)},0)

如图所示:

另一个回调函数的例子:

代码如下:

function add(x,y,callback){
  setTimeout(function(){
    var result = x + y
    callback(result)
  },1000)
}

var x = 6
var y = 7
add(x,y,function(result){
  NSLog(x + ' + ' + y + ' = ' + result)
})

效果如下:

又一个回调函数的例子:

代码如下:

<!DOCTPYE html>  
<html lang="zh">  
<head>  
    <link rel="icon" href="beyond2.jpg" type="image/x-icon"/>
    <meta charset="UTF-8">
    <meta name="author" content="beyond">
    <meta http-equiv="refresh" content="520">
    <meta name="description" content="未闻花名-免费零基础教程-beyond">
    <meta name="viewport" content="width=device-width, 
    initial-scale=1.0, maximum-scale=1.0,minimum-scale=1.0,user-scalable=0" />
    <meta name="keywords" content="HTML,CSS,JAVASCRIPT,JQUERY">
    <title>beyond心中の动漫神作</title>
    <link rel="stylesheet" type="text/css" href="beyondbasestylewhite3.css">
    <script type="text/javascript" src="nslog.js"></script>

    <!--[if lt IE 9]>
        <script src="//apps.bdimg.com/libs/html5shiv/3.7/html5shiv.min.js"></script>


        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/1.10.2/jquery.js">
        </script>
    <![endif]-->

    <style type="text/css">
        body{
            font-size: 100%; 
            /*声明margin和padding是个好习惯*/  
            margin: 0;  
            padding: 0; 
            background-image: url("sakura4.png");  
            background-repeat: no-repeat;  
            background-position: center center;  
        }
    </style>
    <!-- 绿色按钮的css效果 -->
    <link rel="stylesheet" type="text/css" href="beyondbuttongreen.css">

    <!-- 引入 jquery 2.1.4 -->
    <!--[if gte IE 9]><!--> 
    <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/2.1.4/jquery.js">
    </script>
    <!--<![endif]-->


    
</head>  
  
<body>  
        <h1 style="color:white;text-shadow:2px 2px 4px #000;letter-spacing:5px;" class="sgcontentcolor sgcenter">  
            未闻花名
        </h1>

        <button id="id_btn" class="class_btn class_btn_green" type="button" style="display:block;margin:auto;margin-top:10px;">点我试试</button>
        <script type="text/javascript">

            // 自己封装一个get请求
            function beyondGet (url,callback) {
                   // 自己实现一个ajax请求
                   // 1. 创建xhr
                   var xmlHttpRequest = new XMLHttpRequest()
                   // 2. 请求url资源
                   xmlHttpRequest.onload = function () {
                      // 5. 核心代码,回调结果
                      callback(xmlHttpRequest.responseText)
                   }
                   // 3.连接资源
                   xmlHttpRequest.open("get",url,true)
                   // 4.发送请求
                   xmlHttpRequest.send()
            }
            $(document).ready(function () {
               $(".class_btn_green").click(function () {
                  // 使用自己封装的ajax请求
                  beyondGet("anohana2.txt",function (responseText) {
                    // body...
                    NSLog('responseText: \n' + responseText)
                  })
                   
               })
            })
        </script>
        <p class="sgcenter">
          <b>注意:</b> 使用自己封装的get函数发送ajax请求
        </p>
        <footer id="copyright">
            <p style="font-size:14px;text-align:center;font-style:italic;">  
            Copyright © <a id="author">2018</a> Powered by <a id="author">beyond</a>  
            </p>        
        </footer>
    
</body>  
</html>  

效果如下:

另一种写法,效果和上面是一样的:

<script type="text/javascript">

            // 自己封装一个get请求
            function beyondGet (url,callback) {
                   // 自己实现一个ajax请求
                   // 1. 创建xhr
                   var xmlHttpRequest = new XMLHttpRequest()
                   // 3.连接资源
                   xmlHttpRequest.open("GET",url,true)

                   // 2. 请求url资源
                   xmlHttpRequest.onreadystatechange = function () {
                      if (xmlHttpRequest.readyState === 4 && xmlHttpRequest.status === 200) {
                          // 5. 核心代码,回调结果
                          callback(xmlHttpRequest.responseText)
                      }
                   }
                   
                   // 4.发送请求
                   xmlHttpRequest.send(null)
            }
            $(document).ready(function () {
               $(".class_btn_green").click(function () {
                  // 使用自己封装的ajax请求
                  beyondGet("anohana2.txt",function (responseText) {
                    // body...
                    NSLog('responseText: \n' + responseText)
                  })
                   
               })
            })
        </script>

注意:凡是需得到一个函数内部异步操作的结果时,必须使用回调函数
如 setTimeout,readFile,writeFile,ajax

打个比方:你想追一部正在播出的动漫,每天都不厌其烦,一天N次去看它有更新了没有,这样非常没有效率
但是,只要你一旦订阅了之后,每当它有更新时,它会主动通知你,这就是在回调

众所周知,JavaScript天生不支持模块化
Node环境,通过CommonJS对JS实现了支持模块化

浏览器中,也可以像Node.js一样模块化编程
只需要使用第3方库:
1. Require.js (俗称AMD,即异步 模块 定义,全称是Asynchromous Module Definition)
官网:requirejs.com

2. Sea.js     (俗称CMD,即公共 模块 定义,全称是Common Module Definition)

3. UMD,即通用 模块 定义,全称是Universal Module Definition)

4. ECMAScript6 中官方的 Modules 规范(后面的Vue课程中会讲编译工具)

(前面课程已经教过)Less 编译–> CSS
ES6  编译–> ES5

在npm5以前,是没有package-lock.json这个文件

npm5以后,才增加了package-lock.json  (关键字:锁定版本+提高下载速度)
1. 当你安装第3方包的时候,都会生成或更新这个package-lock.json文件
2. 新版本中,不用写 –save,也能够将第3方包的安装记录,自动添加到包描述文件package.json里的dependences属性里
3. package-lock.json中记录了node_modules中所有包的 版本 以及 下载地址
4. 从package-lock.json文件名来看,有一个lock,作用就是(在不小心删除node_modules目录时,重新npm install时)锁定住特定的版本号的功能,防止自动升级到新版本

// 在不升级Node的情况下,npm自己升级自己
npm install –global npm

// 下载安装时,指定版本@1.11.1
npm install jquery@1.11.1

// package.json中jquery的^1.11.1中的^符号表示,可以安装该版本之上的版本

自己为数组实现一个find函数,参数是一个匿名函数,
find函数内部遍历数组,并让每个元素作为参数传递给参数里的匿名函数,
如果条件成立,则返回 当前的元素
Array.prototype.myFind = function(conditionFunction){
for(var i = 0; i < this.length; i++){
if(conditionFunction(this[i],i)){
return this[i]
}
}
// 如果没有找到符合条件的元素,则undefine
}

// 使用方法
var girlArr = [‘面码’,’mathilda’,’逢坂大河’]
var menma = girlArr.myFind(function(item,index){
return item === ‘面码’
})
NSLog(menma)

运行效果,如图所示

express 默认情况下,提示Can’t get xxx
如何自定义一个漂亮的404页面?

只需要要在自己的路由之后,增加一个
appServer.use(function(request,response){
// 这样所有的未处理的请求,都会跑到这儿来
// 在这儿,定制404页面即可
})

补充了一下,数组所有的与遍历相关的方法
forEach
find,findIndex
every
some
map
includes

reduce 强大

MongoDB

关系型数据库 与 非关系型数据库

NoSQL

关系型数据库:
1. 表就是关系,表与表之间存在关系
2. 所有关系型数据库都通过sql语言进行操作数据库
3. 所有的关系型数据库最初都要设计表结构(包括约束:唯一,主键,非空,默认值)

非关系型数据库:
1. 非常得灵活
2. 有的仅仅就是key – value键值对
3. MongoDB是长得最像关系型数据库的非关系型数据库
3.1 MongoDB里面,也有数据库
3.2 MongoDB里面,表 就叫做 集合/数据
3.3 MongoDB里面,记录 就叫做 文档对象
3.4 MongoDB里面,不需要设计 表结构 ,你可以往里面任意地存入数据,没有结构这么一个概念

MongoDB官网:mongodb.org,

点击download,然后选择社区版Community Server

https://fastdl.mongodb.org/osx/mongodb-osx-ssl-x86_64-3.6.3.tgz

mongodb-osx-ssl-x86_64-3.6.3.tgz  大小 87.7M

32位的下载地址: mongodb.org/dl/win32/i386

安装过程:

1.  解压缩

2. 将文件夹mongodb-osx-x86_64-3.6.3 重命名为 mongodb

3. 将mongodb目录整个移动到 /usr/local目录

4. 将mongodb目录下的bin目录的路径永久添加到path环境变量中

4.1 打开 open -t ~/.bash_profile

4.2 在文档的最后面追加下面两行

MONGODB=/usr/local/mongodb/bin
export PATH=$PATH:$MONGODB

注意: 下面这个如果是在命令行中,那么仅仅是临时添加到环境变量中

export PATH=/usr/local/mongodb/bin:$PATH

5. 核心步骤:在系统根目录下创建/data/db  (若缺少这一步,服务无法启动)

MongoDB的数据存储在磁盘根目录下的data目录的db目录下,但是这个目录在安装过程不会自动创建,所以你需要手动创建data目录,并在data目录中创建db目录。
sudo mkdir -p /data/db
注意:/data/db 是 MongoDB 默认的启动的数据库路径(--dbpath)。

也可以使用下面的命令来 修改默认的存储目录

mongod --dbpath=数据存储目录

6. 打开终端,输入mongod –version 查看版本

7. sudo mongod 启动数据库服务,如图所示: (必须加sudo)

8. 下面使用mongo命令,连接本机的数据库

再打开一个终端, cd /usr/local/mongodb/bin/ ,  再执行  ./mongo

(因为加入了 环境变量,直接敲 mongo 也是可以的)

可以看到连接信息,如图所示:

此时前面一个终端就会收到连接信息,如下所示:

2018-04-13T19:54:02.196+0800 I NETWORK  [initandlisten] waiting for connections on port 27017
2018-04-13T19:58:16.181+0800 I NETWORK  [listener] connection accepted from 127.0.0.1:64944 #1 (1 connection now open)
2018-04-13T19:58:16.183+0800 I NETWORK  [conn1] received client metadata from 127.0.0.1:64944 conn: 
{ application: { name: "MongoDB Shell" }, driver: { name: "MongoDB Internal Client", version: "3.6.3" }, os: { type: "Darwin", name: "Mac OS X", architecture: "x86_64", version: "16.6.0" } }

9. 如果要退出连接, 输入 exit 即可

如果要关闭服务, 只要关闭终端即可,或者 Ctrl + C 取消

注意: windows系统下面,也要找到安装目录下的bin目录,将它添加到环境变量中

如windows下的 c:\Program Files\MongoDB\Server\3.4\bin

之后,再打开终端: 输入命令 mongod –version

MongodDB数据库的表现形式如下:

1. 可以有多个数据库

2. 一个库,可以有多个表(集合)

3. 一个表(集合),可以有多个记录(文档)

4. 一个记录(文档),就是一个对象,由键值对构成

由此可见: 文档结构相当滴灵活,木有任何限制;

不需要像关系型数据库,事先设计库、表、约束等

{
  db2:{
    girls:[
      {
        "girlName" : "面码",
        "girlAge" : 15,
        "girlAnime" : "未闻花名"
      },
      {
        "girlName" : "mathilda",
        "girlAge" : 12,
        "girlAnime" : "这个杀手不太冷"
      },
      {
        "girlName" : "逢坂大河",
        "girlAge" : 16,
        "girlAnime" : "龙与虎"
      }
    ],
  // 另外两张表(集合)  
  animes:[{},{},{}],
  songs: [{},{},{}]
  },
// 另外的一个数据库,以此类推
db2:{...}
}

常用命令和操作:

1.  显示所有的数据库

> show dbs

效果如下:

2.  使用指定数据库(如果没有会创建)

> use db2

3.  显示当前在哪一个数据库

> db

4.  尝试在db2这个数据库的girl表(这儿叫集合),插入一条记录(这儿叫文档 )

> db.girl.inserOne({

“girlName” : “面码”,

“girlAge” : 15

})

上面的语句在插入一条记录(文档)的同时,也新建了一个girl表(集合)

5.  显示当前的db2数据库下面,所有的表(集合)

> show collections

6.  查询当前的db2数据库下面girl表(集合)里面的所有记录(文档)

> db.girl.find()

7.  更新一条记录(文档)

> 有点复杂,emmmm….

8.  删除一条记录(文档),条件是: girlAge = 15

>db.girl.remove({“girlAge”:15})

删除所有的记录(文档)

>db.girl.remove({})

9.  删除girl表(集合)

> db.girl.drop()

在Node中如何操作MongoDB数据库,

1.  使用MongoDB官方提供的包进行操作

https://github.com/mongodb/node-mongodb-native

2.  推荐使用第3方包 mongoose 进行操作,mongoose在MongoDB官方的包上面再进行了一层封装,使用起来更加方便

http://mongoosejs.com/

第1步:  安装第3方包

npm install mongoose --save

第2步:  按照官方的demo,向db2数据库中的girl表(集合)中插入一条记录(文档)

node_34_add.js代码如下:

function NSLog(loli,needLogo=true) {console.log(loli);if(needLogo){console.log('\nCopyright © 2018 Powered by beyond')};}  


// 导入框架
const mongoose = require('mongoose')
// 向db2这个数据库插入记录(没有则会创建)
const dbUrl = 'mongodb://localhost/db2'
mongoose.connect(dbUrl)

// 向girl表(集合)插入记录(文档)
// 定义girl表(集合)的规范
var girlSchema = mongoose.Schema({
	girlName: {
		type:String,
		// 非空字段
		required: true
	},
	girlAge: Number,
	girlAnime: String
	// isLoli: Boolean
	// pubTime: Date
	// messages:[{title:String,content:String,pubTime:Date}]
	// lover:{name:String,age:Number}
})
// 通过schema转成一个构造函数
// 参数1必须是: 首字母大写
// mongoose会自动将它转成小写并变成复数,作为表(集合)名
// 例如这儿的Girl,会自动变成girls表(集合)
// model方法的返回值是: 构造函数
const Girl = mongoose.model('Girl',girlSchema)
// typeof Girl: function 构造函数
NSLog('typeof Girl: ' + typeof(Girl))
// 使用 构造函数,创建一个记录(文档)对象,用于插入
const tiger = new Girl({
	girlName: "面码",
	girlAge: 15,
	girlAnime: "未闻花名"
})

// 插入一个对象
tiger.save().then(
	() => NSLog('插入成功')
)

效果如下: (插入成功后,再通过mongo命令连接数据库,查看是否真的有添加成功)

查询所有的记录(文档)node_35_find.js代码如下:

find方法

// 查找所有,第1个参数不写,就是查找所有
Girl.find(function (error,girls) {
	if (error) {
		return NSLog('查询出错: ' + error)
	}
	NSLog('查询结果: ' + girls)
})

效果如下:

按条件,查询指定的记录(文档)node_35_find.js代码如下:

find方法

// ----------------条件查询------------------

// 条件查询,第1个参数就是查询条件之一
var condition = {girlName:"面码"}
Girl.find(condition,function (error,girls) {
	if (error) {
		return NSLog('查询出错: ' + error)
	}
	NSLog('条件查询: ' + girls)
})

条件查询,效果如下:

另一个按条件查询,年龄大于12岁,node_35_find.js代码如下:

find方法

// ----------------条件查询------------------
// 条件查询,第1个参数就是查询条件之一
// 姓名为 面码, 并且 年龄 大于(great than) 12
var condition = {
				  girlName:"面码",
				  girlAge: {$gt: 12}
				}
Girl.find(condition,function (error,girls) {
	if (error) {
		return NSLog('查询出错: ' + error)
	}
	NSLog('条件查询: ' + girls)
})

效果如下:

根据id查找,并更新记录node_35_update.js代码如下:

findByIdAndUpdate

// ----------------根据Id查找并更新记录(文档)------------------
var idString = '5ad1732aa33423ed19e7ae2e'
var newObj = {girlAge: 13}
Girl.findByIdAndUpdate(idString,newObj,function (error) {
	if (error) {
		NSLog('更新失败: ' + error)	
	}
	NSLog('更新成功')
})

效果如下:

根据指定条件,更新一个:
Model.findOneAndUpdate([conditions],[updateObj],[options],[callback])

// ------------把girlName为'面码'更新为'menma'--------
var newObj = {girlName: 'menma'}
// 参数1: conditions不写 就是更新所有
var conditions = {girlName: '面码'}
// 参数3: 写不写好像都可以...emmmm
var options = {strict: true}
Girl.findOneAndUpdate(conditions,newObj,options,function (error) {
	if (error) {
		NSLog('更新失败: ' + error)	
	}
	NSLog('更新成功')
})

根据条件更新所有:

Model.updateMany(conditions,newDoc,[options],[callback])

还有一个只更新一个的方法是: Model.update(conditions,newDoc,[options],[callback])

// ------------把所有的girl年龄更新为13岁---------
var newObj = {girlAge: 13}
// 参数1: conditions不写 就是更新所有
var conditions = {}
// 参数3: 写不写好像都可以...emmmm
var options = {strict: false}
Girl.updateMany(conditions,newObj,options,function (error) {
	if (error) {
		NSLog('更新失败: ' + error)	
	}
	NSLog('更新成功')
})

条件删除node_35_remove.js代码如下:

remove方法

// ----------------条件删除------------------
Girl.remove({girlAge: 15},function (error) {
	if (error) {
		return NSLog('删除失败: ' + error)
	}
	NSLog('success: 条件删除')
})

删除全部node_35_remove.js代码如下:

remove方法

// ----------------删除所有------------------
Girl.remove({},function (error) {
	if (error) {
		return NSLog('删除失败: ' + error)
	}
	NSLog('success: 删除全部')
})

删除数据:
Model.findOneAndRemove(conditions,[options],[callback])

// ----------------条件删除------------------
Girl.findOneAndRemove({girlAge: 13},function (error) {
	if (error) {
		return NSLog('删除失败: ' + error)
	}
	NSLog('success: 条件删除')
})

根据id删除一个
Model.findByIdAndRemove(id,[options],[callback])

// ----------------条件删除------------------
Girl.findOneAndRemove('5ad40f38d1a2972dfb3fcf94',function (error) {
	if (error) {
		return NSLog('删除失败: ' + error)
	}
	NSLog('success: 条件删除')
})

下面开始把前面的使用文件作为数据库的CRUD项目Node_32_index.js,改造成使用mongodb

这是原来项目的结构:

使用mongod作为数据库,  新的项目node_36_index.js结构如下:

运行效果如下:

3个完整js代码如下:

首页app入口: node_36_index.js代码如下:

function NSLog(loli,needLogo=true) {console.log(loli);if(needLogo){console.log('\nCopyright © 2018 Powered by beyond')};}  


// 导入框架
var express = require('express')
// 创建服务器对象
var appServer = express()
// 监听端口,并启动服务
appServer.listen(5267,function (error) {
	if (error) {
		return NSLog('启动失败: ' + error)
	}
	NSLog('服务启动成功')
})
// -----------------------------------

// 静态资源请求时的 staticFileUrlPrefix
var staticFileUrlPrefix = '/public/'
// var staticFileUrlPrefix = '/public'

// 访问也只能使用 localhost:5267/public/img/beyond.jpg
// 磁盘上的静态资源目录
var staticFilePath = './public/'   
// var staticFilePath = 'public' 

var callbackFunction = express.static(staticFilePath)
appServer.use(staticFileUrlPrefix,callbackFunction)

// -----------------------------------

// 指明:对于 所有后缀为html 的模板文件 使用模板引擎
var templateFileSuffix = 'html'
appServer.engine(templateFileSuffix,require('express-art-template'))
// 下面这一句参数配置,可有可无
appServer.set('view options',{
	debug: process.env.NODE_ENV !== 'production'
})
// 注意:如果不想把模板文件放在默认的views目录下,则可以通过下面代码更改设置
// appServer.set('views','其他目录')

// -----------------------------------
// 使用middleware中间件body-parser进行post请求体中数据解析
var bodyParser = require('body-parser')
// 设置解析 application/x-www-form-urlencoded
appServer.use(bodyParser.urlencoded({extended: false}))        
// 设置解析 application/json
appServer.use(bodyParser.json())
// -----------------------------------
// 自定义路由设计的目的是:
// 1.让主入口程序的职责更加单一,代码更加简洁
//     1.1 创建服务
//     1.2 做一些服务相关的配置,比如:
//           1.2.1 静态资源配置
//           1.2.2 模板引擎配置
//           1.2.3 body-parse 解析表单
//           1.2.4 挂载自定义路由
//           1.2.5 监听端口,启动服务
// 使用自定义的路由模块 必须使用./
// 注意: 配置模板引擎和body-parser, 一定要在挂载路由之前
var beyondRouter = require('./node_36_router')
appServer.use(beyondRouter)

路由模块node_36_router.js代码如下:

function NSLog(loli) {console.log(loli);return 'Copyright © 2018 Powered by beyond';};  
/*
	自定义路由模块的职责是:
		专门处理所有的路由
		根据不同的请求方式和路径,采取相应的处理方法
*/ 
// express 专门提供了路由的处理方法
var express = require('express')
// 1.使用express专门提供的路由器处理路由
var router = express.Router()

// ----------------引入dao模块-------------------
// 先对dao初始化
var GirlDaoFunction = require('./node_36_dao')
// 时间格式化
var BeyondDateFormatFunction = require('./BeyondDateFormat')

// ----------------首页-------------------
router.get('/',function (request,response) {
	// 至于请求参数可以这样:
	// var queryObj = request.query

	// ----------------查找所有------------------
	// 查找所有,第1个参数不写,就是查找所有
	GirlDaoFunction.find(function (error,girls) {
	if (error) {
		return NSLog('查询出错: ' + error)
	}

	// 使用模板引擎渲染
		// 注意: 模板文件默认是放在views目录下
		// 为此,我们在views目录下  分别为不同的业务模块创建了不同的文件夹
		// 如 login登录 admin后台管理 index前台首页 article文章 comment评论
		response.render('index/node_36_index.html',{girlArr:girls})

	})

})

// ----------------添加的表单页面-------------------
router.get('/add',function (request,response) {
	response.render('index/node_36_add.html')
})


// ----------------增加一条记录-------------------
router.post('/insert',function (request,response) {
	// 1.body-parser得到obj
	var girlObj = request.body
	// NSLog(girlObj)
	// 1.调用girlDao写到文件数据库
	girlObj.pubTime = BeyondDateFormatFunction(new Date(),'yyyy-MM-dd')
	var newGirl = new GirlDaoFunction(girlObj)
	// 2.保存到数据库
	newGirl.save(function (error) {
		if (error) {
			// 有错误
			return response.status(500).send(error)
		}
		// 没有错误,跳转到首页	
		response.redirect('/')
	})
})

// ----------------修改页面------------------- 
router.get('/edit',function (request,response) {
	// 查询的girlID 不知怎么滴,首尾有引号
	//  "5ad42675f917fa32e250a58a"
	// 使用正则,把引号去掉, 如果不使用g的话,仅仅只是去掉第1个引号
	var pureId = request.query._id.replace(/"/g,"")
	// NSLog('query: ' + pureId)
	GirlDaoFunction.findById(pureId,function (error,girl) {
		if (error) {
			return response.status(500),send(error)
		}
		response.render('index/node_36_edit.html',{'girl': girl})
	})
})
// ----------------更新数据库------------------- 
router.post('/update',function (request,response) {
	// 1.请求体 id号 (前后多了两个引号,要手动去掉)
	var girlID = request.body._id
	var girlIDWithoutQuote = girlID.replace(/"/g,'')
	// 2.重新设置回去
	request.body._id = girlIDWithoutQuote
	// 3.根据id查询和更新
	GirlDaoFunction.findByIdAndUpdate(girlIDWithoutQuote,request.body,function (error) {
		if(error){
			// 如果保存出错了
			return response.status(500).send(error)
		}
		// 如果保存成功,回首页
		response.redirect('/')
	})
})

// ----------------删除一条记录------------------- 
router.get('/delete',function (request,response) {
	// 1.获取query对象中的_id
	var girlID = request.query._id
	var girlIDWithoutQuote = girlID.replace(/"/g,'')
	// 2.调用dao从数据库中删除一个对象
	GirlDaoFunction.findByIdAndRemove(girlIDWithoutQuote,function (error) {
		if (error) {
			// 有错误
			return response.status(500).send(error)
		}
		// 如果没有错误,跳转到首页
		return response.redirect('/')
	})
})


// 3.在模块文件最后,导出router
module.exports = router

使用了MongodDB的数据库操作模块node_36_dao.js代码如下:

function NSLog(loli) {console.log(loli);return 'Copyright © 2018 Powered by beyond';};  

// ----------------初始化------------------- 
var mongoose = require('mongoose')
// 连接数据库 db2,默认端口是: 27017
mongoose.connect('mongodb://localhost/db2')
var Schema = mongoose.Schema
// 仿着 文件数据库 node_32.db 设计表结构
var girlSchema = new Schema({
	girlName: {
		type: String,
		required: true
	},
	girlAge: {
		type: Number,
		enum: [8,9,10,11,12,13,14,15,16], // 好像没有什么效果
		default: 13
	},
	girlDescription: {
		type: String,
		required: true,
		default: ''

	},
	pubTime: {
		type: String,
		required: true,
		// 默认值 只是一个函数名,但不调用,即不加括号,因为一旦调用,会立刻执行
		default: ''
	},

})


// 核心代码: 直接导出 模型构造函数
// 参数1: Girl 首字母必须大写,且必须是单数; 
// 这样就能自动生成girls表(集合)
module.exports = mongoose.model('Girl',girlSchema)

3个html模板代码如下:

首页模板node_36_index.html代码如下:

<!DOCTPYE html>  
<html lang="zh">  
<head>  
    <link rel="icon" href="/public/img/beyond2.jpg" type="image/x-icon"/>
    <meta charset="UTF-8">
    <meta name="author" content="beyond">
    <meta http-equiv="refresh" content="520">
    <meta name="description" content="未闻花名-免费零基础教程-beyond">
    <meta name="viewport" content="width=device-width, 
    initial-scale=1.0, maximum-scale=1.0,minimum-scale=1.0,user-scalable=0" />
    <meta name="keywords" content="HTML,CSS,JAVASCRIPT,JQUERY,XML,JSON,C,C++,C#,OC,PHP,JAVA,JSP,PYTHON,RUBY,PERL,LUA,SQL,LINUX,SHELL,汇编,日语,英语,泰语,韩语,俄语,粤语,阿语,魔方,乐理,动漫,PR,PS,AI,AE">
    <title>beyond心中の动漫神作</title>
    <link rel="stylesheet" type="text/css" href="/public/css/beyondbasestylewhite5.css">
    <script type="text/javascript" src="/public/js/nslog.js"></script>

    <!--[if lt IE 9]>
        <script src="//apps.bdimg.com/libs/html5shiv/3.7/html5shiv.min.js"></script>


        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/1.10.2/jquery.js">
        </script>
    <![endif]-->

    <style type="text/css">
        body{
            font-size: 100%; 
            /*声明margin和padding是个好习惯*/  
            margin: 0;  
            padding: 0; 
            background-image: url("/public/img/sakura4.png");  
            background-repeat: no-repeat;  
            background-position: center center;  
        }
    </style>
    <!-- 绿色按钮的css效果 -->
    <link rel="stylesheet" type="text/css" href="/public/css/beyondbuttongreen.css">

    <!-- 引入 jquery 2.1.4 -->
    <!--[if gte IE 9]><!--> 
    <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/2.1.4/jquery.js">
    </script>
    <!--<![endif]-->


    <link rel="stylesheet" type="text/css" href="/public/lib/bootstrap/bootstrap.css">


    <!-- 分隔线 -->

    <!-- Bootstrap core CSS -->
    <link href="/public/lib/node30_337_bootstrap.css" rel="stylesheet">

    <!-- Custom styles for this template -->
    <link href="/public/css/node30dashboard.css" rel="stylesheet">

    

    <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->
    <!--[if lt IE 9]>
      <script src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script>
      <script src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script>
    <![endif]-->

</head>  
  
<body>

    <nav class="navbar navbar-inverse navbar-fixed-top">
      <div class="container-fluid">
        <div class="navbar-header">
          <button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
            <span class="sr-only">Toggle navigation</span>
            <span class="icon-bar"></span>
            <span class="icon-bar"></span>
            <span class="icon-bar"></span>
          </button>
          <a class="navbar-brand" href="#" style="color:white;">beyond</a>
        </div>
        <div id="navbar" class="navbar-collapse collapse">
          <ul class="nav navbar-nav navbar-right">
            <li><a href="#">Dashboard</a></li>
            <li><a href="#">Settings</a></li>
            <li><a href="#">Profile</a></li>
            <li><a href="#">Help</a></li>
          </ul>
          <form class="navbar-form navbar-right">
            <input type="text" class="form-control" placeholder="Search...">
          </form>
        </div>
      </div>
    </nav>

    <div class="container-fluid">
      <div class="row">
        <div class="col-sm-3 col-md-2 sidebar">
          <ul class="nav nav-sidebar">
            <li class="active"><a href="#">Overview <span class="sr-only">(current)</span></a></li>
            <li><a href="#">Reports</a></li>
            <li><a href="#">Analytics</a></li>
            <li><a href="#">Export</a></li>
          </ul>
          <ul class="nav nav-sidebar">
            <li><a href="">Nav item</a></li>
            <li><a href="">Nav item again</a></li>
            <li><a href="">One more nav</a></li>
            <li><a href="">Another nav item</a></li>
            <li><a href="">More navigation</a></li>
          </ul>
          <ul class="nav nav-sidebar">
            <li><a href="">Nav item again</a></li>
            <li><a href="">One more nav</a></li>
            <li><a href="">Another nav item</a></li>
          </ul>
        </div>
        <div class="col-sm-9 col-sm-offset-3 col-md-10 col-md-offset-2 main">
          <h1 class="page-header" style="color:white;text-shadow:2px 2px 4px #000;letter-spacing:5px;" class="sgcontentcolor sgcenter">未闻花名<small style="color:#ddd;"><a href="/add" style="color:white">添加记录</a>(Node+Express+MongoDB版)</small></h1>

          <!-- <div class="row placeholders">
            <div class="col-xs-6 col-sm-3 placeholder">
              <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
              <h4>Label</h4>
              <span class="text-muted">Something else</span>
            </div>
            <div class="col-xs-6 col-sm-3 placeholder">
              <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
              <h4>Label</h4>
              <span class="text-muted">Something else</span>
            </div>
            <div class="col-xs-6 col-sm-3 placeholder">
              <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
              <h4>Label</h4>
              <span class="text-muted">Something else</span>
            </div>
            <div class="col-xs-6 col-sm-3 placeholder">
              <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
              <h4>Label</h4>
              <span class="text-muted">Something else</span>
            </div>
          </div> -->

          <!-- <h2 class="sub-header">动漫神作列表 </h2> -->
          
          <div class="table-responsive" >
            <table class="table table-striped">
              <thead>
                <tr>
                  <th>id</th>
                  <th>芳名</th>
                  <th>年龄</th>
                  <th>动漫</th>
                  <th>日期</th>
                  <th>操作</th>
                </tr>
              </thead>
              <tbody>
              {{ each girlArr }}
                <tr>
                  <td style="vertical-align:middle;">{{ $index + 1 }}</td>
                  <td style="vertical-align:middle;">{{ $value.girlName }}</td>
                  <td style="vertical-align:middle;">{{ $value.girlAge }}</td>
                  <td style="vertical-align:middle;">{{ $value.girlDescription }}</td>
                  <td style="vertical-align:middle;">{{ $value.pubTime }}</td>
                  <td>
                    <button class="btn btn-success"><a href="/edit?_id={{ $value._id }}" style="color:white;">修改</a></button>
                    <button class="btn btn-danger"><a href="/delete?_id={{ $value._id }}" style="color:white;">删除</a></button>
                  </td>
                </tr>
            {{ /each }}
                
              </tbody>
            </table>
          </div>
        </div>
      </div>
    </div>

    <footer id="copyright">
            <p style="font-size:14px;text-align:center;font-style:italic;">  
            Copyright © <a id="author">2018</a> Powered by <a id="author">beyond</a>  
            </p>        
        </footer>
  </body>
</html>

添加girl的模板node_36_add.html代码如下:

<!DOCTPYE html>  
<html lang="zh">  
<head>  
  <link rel="icon" href="/public/img/beyond2.jpg" type="image/x-icon"/>
  <meta charset="UTF-8">
  <meta name="author" content="beyond">
  <meta http-equiv="refresh" content="520">
  <meta name="description" content="未闻花名-免费零基础教程-beyond">
  <meta name="viewport" content="width=device-width, 
  initial-scale=1.0, maximum-scale=1.0,minimum-scale=1.0,user-scalable=0" />
  <meta name="keywords" content="HTML,CSS,JAVASCRIPT,JQUERY,XML,JSON,C,C++,C#,OC,PHP,JAVA,JSP,PYTHON,RUBY,PERL,LUA,SQL,LINUX,SHELL,汇编,日语,英语,泰语,韩语,俄语,粤语,阿语,魔方,乐理,动漫,PR,PS,AI,AE">
  <title>beyond心中の动漫神作</title>
  <link rel="stylesheet" type="text/css" href="/public/css/beyondbasestylewhite5.css">
  <script type="text/javascript" src="/public/js/nslog.js"></script>

    <!--[if lt IE 9]>
        <script src="//apps.bdimg.com/libs/html5shiv/3.7/html5shiv.min.js"></script>


        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/1.10.2/jquery.js">
        </script>
        <![endif]-->

        <style type="text/css">
          body{
            font-size: 100%; 
            /*声明margin和padding是个好习惯*/  
            margin: 0;  
            padding: 0; 
            background-image: url("/public/img/sakura4.png");  
            background-repeat: no-repeat;  
            background-position: center center;  
          }
        </style>
        <!-- 绿色按钮的css效果 -->
        <link rel="stylesheet" type="text/css" href="/public/css/beyondbuttongreen.css">

        <!-- 引入 jquery 2.1.4 -->
        <!--[if gte IE 9]><!--> 
        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/2.1.4/jquery.js">
        </script>
        <!--<![endif]-->


        <link rel="stylesheet" type="text/css" href="/public/lib/bootstrap/bootstrap.css">


        <!-- 分隔线 -->

        <!-- Bootstrap core CSS -->
        <link href="/public/lib/node30_337_bootstrap.css" rel="stylesheet">

        <!-- Custom styles for this template -->
        <link href="/public/css/node30dashboard.css" rel="stylesheet">



        <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->
    <!--[if lt IE 9]>
      <script src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script>
      <script src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script>
      <![endif]-->

    </head>  

    <body>

      <nav class="navbar navbar-inverse navbar-fixed-top">
        <div class="container-fluid">
          <div class="navbar-header">
            <button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
              <span class="sr-only">Toggle navigation</span>
              <span class="icon-bar"></span>
              <span class="icon-bar"></span>
              <span class="icon-bar"></span>
            </button>
            <a class="navbar-brand" href="#" style="color:white;">beyond</a>
          </div>
          <div id="navbar" class="navbar-collapse collapse">
            <ul class="nav navbar-nav navbar-right">
              <li><a href="#">Dashboard</a></li>
              <li><a href="#">Settings</a></li>
              <li><a href="#">Profile</a></li>
              <li><a href="#">Help</a></li>
            </ul>
            <form class="navbar-form navbar-right">
              <input type="text" class="form-control" placeholder="Search...">
            </form>
          </div>
        </div>
      </nav>

      <div class="container-fluid">
        <div class="row">
          <div class="col-sm-3 col-md-2 sidebar">
            <ul class="nav nav-sidebar">
              <li class="active"><a href="#">Overview <span class="sr-only">(current)</span></a></li>
              <li><a href="#">Reports</a></li>
              <li><a href="#">Analytics</a></li>
              <li><a href="#">Export</a></li>
            </ul>
            <ul class="nav nav-sidebar">
              <li><a href="">Nav item</a></li>
              <li><a href="">Nav item again</a></li>
              <li><a href="">One more nav</a></li>
              <li><a href="">Another nav item</a></li>
              <li><a href="">More navigation</a></li>
            </ul>
            <ul class="nav nav-sidebar">
              <li><a href="">Nav item again</a></li>
              <li><a href="">One more nav</a></li>
              <li><a href="">Another nav item</a></li>
            </ul>
          </div>
          <div class="col-sm-9 col-sm-offset-3 col-md-10 col-md-offset-2 main">
            <h1 class="page-header" style="color:white;text-shadow:2px 2px 4px #000;letter-spacing:5px;" class="sgcontentcolor sgcenter">未闻花名<small><a href="/" style="color:white;">回到首页</a></small></h1>

            <div class="row placeholders">
              <!-- <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div> -->
            </div>

            <!-- <h2 class="sub-header">动漫神作列表 </h2> -->



            <form action="/insert" method="post">
              <div class="form-group">
                <label for="in_input_name">芳名</label>
                <input type="text" class="form-control" id="in_input_name" placeholder="请输入妳的名字" name="girlName">
              </div>
              <div class="form-group">
                <label for="id_input_age">年龄</label>
                <input type="text" class="form-control" id="id_input_age" placeholder="请输入妳的芳龄" name="girlAge">
              </div>

              <div class="form-group">
                <label for="id_input_anime">动漫推荐</label>
                <input type="text" class="form-control" id="id_input_anime" placeholder="请输入妳主演过的动漫" name="girlDescription">
              </div>
              <button type="submit" class="btn btn-default">Submit</button>
            </form>
          </div>
        </div>
        <footer id="copyright">
          <p style="font-size:14px;text-align:center;font-style:italic;">  
            Copyright © <a id="author">2018</a> Powered by <a id="author">beyond</a>  
          </p>        
        </footer>
      </body>
      </html>

修改girl的模板node_36_edit.html代码如下:

<!DOCTPYE html>  
<html lang="zh">  
<head>  
  <link rel="icon" href="/public/img/beyond2.jpg" type="image/x-icon"/>
  <meta charset="UTF-8">
  <meta name="author" content="beyond">
  <meta http-equiv="refresh" content="520">
  <meta name="description" content="未闻花名-免费零基础教程-beyond">
  <meta name="viewport" content="width=device-width, 
  initial-scale=1.0, maximum-scale=1.0,minimum-scale=1.0,user-scalable=0" />
  <meta name="keywords" content="HTML,CSS,JAVASCRIPT,JQUERY,XML,JSON,C,C++,C#,OC,PHP,JAVA,JSP,PYTHON,RUBY,PERL,LUA,SQL,LINUX,SHELL,汇编,日语,英语,泰语,韩语,俄语,粤语,阿语,魔方,乐理,动漫,PR,PS,AI,AE">
  <title>beyond心中の动漫神作</title>
  <link rel="stylesheet" type="text/css" href="/public/css/beyondbasestylewhite5.css">
  <script type="text/javascript" src="/public/js/nslog.js"></script>

    <!--[if lt IE 9]>
        <script src="//apps.bdimg.com/libs/html5shiv/3.7/html5shiv.min.js"></script>


        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/1.10.2/jquery.js">
        </script>
        <![endif]-->

        <style type="text/css">
          body{
            font-size: 100%; 
            /*声明margin和padding是个好习惯*/  
            margin: 0;  
            padding: 0; 
            background-image: url("/public/img/sakura4.png");  
            background-repeat: no-repeat;  
            background-position: center center;  
          }
        </style>
        <!-- 绿色按钮的css效果 -->
        <link rel="stylesheet" type="text/css" href="/public/css/beyondbuttongreen.css">

        <!-- 引入 jquery 2.1.4 -->
        <!--[if gte IE 9]><!--> 
        <script type="text/javascript" src="http://apps.bdimg.com/libs/jquery/2.1.4/jquery.js">
        </script>
        <!--<![endif]-->


        <link rel="stylesheet" type="text/css" href="/public/lib/bootstrap/bootstrap.css">


        <!-- 分隔线 -->

        <!-- Bootstrap core CSS -->
        <link href="/public/lib/node30_337_bootstrap.css" rel="stylesheet">

        <!-- Custom styles for this template -->
        <link href="/public/css/node30dashboard.css" rel="stylesheet">



        <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->
    <!--[if lt IE 9]>
      <script src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script>
      <script src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script>
      <![endif]-->

    </head>  

    <body>

      <nav class="navbar navbar-inverse navbar-fixed-top">
        <div class="container-fluid">
          <div class="navbar-header">
            <button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
              <span class="sr-only">Toggle navigation</span>
              <span class="icon-bar"></span>
              <span class="icon-bar"></span>
              <span class="icon-bar"></span>
            </button>
            <a class="navbar-brand" href="#" style="color:white;">beyond</a>
          </div>
          <div id="navbar" class="navbar-collapse collapse">
            <ul class="nav navbar-nav navbar-right">
              <li><a href="#">Dashboard</a></li>
              <li><a href="#">Settings</a></li>
              <li><a href="#">Profile</a></li>
              <li><a href="#">Help</a></li>
            </ul>
            <form class="navbar-form navbar-right">
              <input type="text" class="form-control" placeholder="Search...">
            </form>
          </div>
        </div>
      </nav>

      <div class="container-fluid">
        <div class="row">
          <div class="col-sm-3 col-md-2 sidebar">
            <ul class="nav nav-sidebar">
              <li class="active"><a href="#">Overview <span class="sr-only">(current)</span></a></li>
              <li><a href="#">Reports</a></li>
              <li><a href="#">Analytics</a></li>
              <li><a href="#">Export</a></li>
            </ul>
            <ul class="nav nav-sidebar">
              <li><a href="">Nav item</a></li>
              <li><a href="">Nav item again</a></li>
              <li><a href="">One more nav</a></li>
              <li><a href="">Another nav item</a></li>
              <li><a href="">More navigation</a></li>
            </ul>
            <ul class="nav nav-sidebar">
              <li><a href="">Nav item again</a></li>
              <li><a href="">One more nav</a></li>
              <li><a href="">Another nav item</a></li>
            </ul>
          </div>
          <div class="col-sm-9 col-sm-offset-3 col-md-10 col-md-offset-2 main">
            <h1 class="page-header" style="color:white;text-shadow:2px 2px 4px #000;letter-spacing:5px;" class="sgcontentcolor sgcenter">未闻花名<small><a href="/" style="color:white;">回到首页</a></small></h1>

            <div class="row placeholders">
              <!-- <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div>
              <div class="col-xs-6 col-sm-3 placeholder">
                <img src="" width="200" height="200" class="img-responsive" alt="Generic placeholder thumbnail">
                <h4>Label</h4>
                <span class="text-muted">Something else</span>
              </div> -->
            </div>

            <!-- <h2 class="sub-header">动漫神作列表 </h2> -->



            <form action="/update" method="post">
              <div class="form-group">
                <label for="in_input_name">芳名</label>
                <input type="text" class="form-control" id="in_input_name" placeholder="请输入妳的名字" name="girlName" value="{{ girl.girlName }}">
              </div>
              <div class="form-group">
                <label for="id_input_age">年龄</label>
                <input type="text" class="form-control" id="id_input_age" placeholder="请输入妳的芳龄" name="girlAge"  value="{{ girl.girlAge }}">
              </div>
              <!-- 隐藏表单 -->
              <input type="hidden" name="_id" value="{{ girl._id }}" />

              <div class="form-group">
                <label for="id_input_anime">动漫推荐</label>
                <input type="text" class="form-control" id="id_input_anime" placeholder="请输入妳主演过的动漫" name="girlDescription"  value="{{ girl.girlDescription }}">
              </div>
              <button type="submit" class="btn btn-default">Submit</button>
            </form>
          </div>
        </div>
        <footer id="copyright">
          <p style="font-size:14px;text-align:center;font-style:italic;">  
            Copyright © <a id="author">2018</a> Powered by <a id="author">beyond</a>  
          </p>        
        </footer>
      </body>
      </html>

# Node.js 第5天课堂笔记

## 知识点

- Express
- MongoDB
- 项目
  + 一天半的时间

## 反馈

-  新版sublime 怎么格式化 怎么一起选中长度不等的内容 怎么改颜色 有的写对了也没颜色 仍然是白色
  +  HTML-CSS-JS Prettify
-  callback是不是相当于函数自调用?
  +  很简单,函数也是一种数据类型,既可以当作参数进行传递,也可以当作方法的返回值
- 我们现在用的模块化是CMD吧 老师能不能给我们扩展一下AMD
  + PHP 中为什么就可以直接 `require`、`include` 因为 PHP 当初在设计的时候就加入了这个功能
  + PHP 这门语言天生就支持
  + 模块作用域
  + 可以使用 API 来进行文件与文件之间的依赖加载
  + 在 Node 这个环境中对 JavaScript 进行了特殊的模块化支持 CommonJS
  + JavaScript 天生不支持模块化
    * require
    * exports
    * Node.js 才有的
  + 在浏览器中也可以像在 Node 中的模块一样来进行编程
    * `<script>` 标签来引用加载,而且你还必须考虑加载的顺序问题
    * require.js 第三方库 AMD
    * sea.js     第三方库 CMD
  + 无论是 CommonJS、AMD、CMD、UMD、EcmaScript 6 Modules 官方规范
    * 都是为了解决 JavaScript 的模块化问题
    * CommonJS、AMD、CMD 都是民间搞出来的
    * EcmaScript 是官方规范定义
    * 官方看民间都在乱搞,开发人员为了在不同的环境使用不同的 JavaScript 模块化解决方案
    * 所以 EcmaScript 在 2015 年发布了 EcmaScript 2016 官方标准
    * 其中就包含了官方对 JavaScript 模块化的支持
    * 也就是说语言天生就支持了
    * 但是虽然标准已经发布了,但是很多 JavaScript 运行换将还不支持
    * Node 也是只在 8.5 版本之后才对 EcmaScript 6 module 进行了支持
    * 后面学 Vue 的时候会去学习
    * less 编译器 > css
    * EcmaScript 6 -> 编译器 -> EcmaScript 5
    * 目前的前端情况都是使用很多新技术,然后利用编译器工具打包可以在低版本浏览器运行。
    * 使用新技术的目的就是为了提高效率,增加可维护性
  +  本着达芬奇画鸡蛋的精神
  +  《使徒行者》三哥
  +  《反黑》陈小春
    *  卧底 8年卧底
    *  文职工作
    *  报了电脑版
    *  吃饭都在看书
    *  学习 -》吃饭也是看书
    *  边角余料
  + 这里涉及到一个中间件的概念
  + app.use 不仅仅是用来处理静态资源的
  + 还可以做很多工作
  + 配置 body-parse 也是通过 app.use 来配置的
  + 这叫中间件,其中有一套规则
- npm init --yes 生成一个package.json 文件 npm --save 文件名 又生成一个package-lock.json文件,又生成的文件和初始化生成的文件有区别吗?
  + 当你安装包的时候,新版的 npm 还会自动生成一个文件:package-lock.json
  + 中间件
  + EcmaScript 6 的 find 方法
- 在express框架中怎么判断访问页面不存在的情况?

## 复习

- 文件路径中的 `/` 和模块标识中的 `/`
- Express 中配置使用 art-template 模板引擎
- Express 中配置使用 body-parser
- Express 中配置处理静态资源
- CRUD 案例中单独提取路由模块

## 上午总结

- 回调函数
  + 异步编程
  + 如果需要得到一个函数内部异步操作的结果,这是时候必须通过回调函数来获取
  + 在调用的位置传递一个函数进来
  + 在封装的函数内部调用传递进来的函数
- find、findIndex、forEach
  + 数组的遍历方法,都是对函数作为参数一种运用
    + every
  + some
  + includes
  + map
  + reduce
- package-lock.json 文件的作用
  + 下载速度快了
  + 锁定版本
- JavaScript 模块化
  + Node 中的 CommonJS
  + 浏览器中的
    * AMD require.js
    * CMD sea.js
  + EcmaScript 官方在 EcmaScript 6 中增加了官方支持
  + EcmaScript 6
  + 后面我们会学,编译工具
- MongoDB 数据库
  + MongoDB 的数据存储结构
    * 数据库
    * 集合(表)
    * 文档(表记录)
- MongoDB 官方有一个 mongodb 的包可以用来操作 MongoDB 数据库
  + 这个确实和强大,但是比较原始,麻烦,所以咱们不使用它
- mongoose
  + 真正在公司进行开发,使用的是 mongoose 这个第三方包
  + 它是基于 MongoDB 官方的 mongodb 包进一步做了封装
  + 可以提高开发效率
  + 让你操作 MongoDB 数据库更方便
- 掌握使用 mongoose 对数据集合进行基本的 CRUD
- 把之前的 crud 案例改为了 MongoDB 数据库版本
- 使用 Node 操作 mysql 数据库

## 下午总结

## 目标

未完待续,下一章节,つづく