zoukankan      html  css  js  c++  java
  • node学习笔记

    一、准备(github地址)

    什么是Javascript?

    • ...

    Javascript能做什么?

    • .....

    浏览器中的Javascript可以做什么?

    • 操作DOM(增删改查)
    • AJAX/跨域
    • BOM
    • ECMAScript

    浏览器中的Javascript不能做什么?

    • 文件操作
    • 没有办法操作系统信息
    • 由于运行环境
    编程语言的能力取决于?
    • 语言本身只是提供了定义变量、定义函数、定义类型、流程控制、循环结构之类的操作
    • 取决于运行该语言的平台(环境)
    • 对于js来说,经常说的JS实际是ES,大部分能力都是由浏览器的执行殷勤决定
    • BOM和DOM可以说是浏览器开放出来的接口
    • 比如:Cordova中提供JS调用摄像头,操作本地文件的API
    • Java既是语言也是平台
    • Java运行在Java虚拟机(跨操作系统)
    • PHP是语言也是平台
    • C#语言,平台:.net framework
    • C#可以运行在MONO这样的平台
    • 因为有人需要将C#运行在Linux平台,所以出现了MONO
    Javascript只可以运行在浏览器中吗?
    • 不是
    • 能运行在哪里,取决于这个环境有没有特定的平台

    node是什么?(Ryan Dahl)

    • Node是javascript运行于服务端的运行环境
    • 注意:是Node选择了Javascript,不是Javascript发展出来一个Node ###Node的诞生
    • 2008年左右,随着AJAX的普及,WEB走向复杂化、系统化
    • 2009年2月Ryan Dahl想要创建一个轻量级,适应现代WEB开的平台
    • 2009年5月Ryan Dahl在GitHub中开源了最初版本,同年11月的JSConf就安排了NODE讲座
    • 2010年底Joyent公司资助,Ryan Dahl也加入了改公司,专门负责NODE的开发
    • 2011年7月在微软的支持下登录Windows平台

    Node在web中的用途

    • node开发的Application处理用户的所有请求和给用户的响应
    • 分发数据(调用NODE服务器接口,再去调用传统服务器)请求,渲染HTML页面

    总结:Node是Javascript的运行环境(平台),不是一门语言,也不是Javascript的框架。

    环境配置

    • 下载安装包
    • 安装过程

    makdir .net->创建一个资源管理器不能创建的一些文件 node --use_strict->必须是严格模式

    匿名函数自执行的方式:
    • (function(){console.log(1)})()
    • +function(){console.log(1)}()
    • !function(){console.log(1)}()

    全局对象

    • global(等价于客户端javascript的window对象)
    • process.stdin->标准输入(采集用户的输入)
    • process.stdout.write()->标准输出类似console.log();
    模板字符串``中间可以随意换行
    • var msg='hello';
    • var a=1;
    • process.stdout.write(${msg} world);
    • process.stdout.write(${msg} world ${a});

    Debug

    • vs code编辑器里面可以调试
    • node debug XXX.js可以调试
    • node-inspector执行后打开127.0.0.1:8080页面调试(如:node-debug d.js此时会自动打开浏览器调试界面google)
    • devtool XXX.js也是调试有BUG

    练习

    • ctrl+c终止当前进程
    • 简单的人际交互,如何在控制台接收用户的输入,如:e.js

    异步操作

    • 现实
    • 程序>1、setTimeout();2、$.ajax()

    node中的异步操作

    • Node采用Chrome V8引擎处理Javascript脚本,V8最大的特点就是单线程运行,一次只执行一个任务。
    • Node大量采用异步操作(asynchronous operation),即任务不是马上执行,而是插在任务队列的尾部,等到前面的任务运行完成后再执行
    • 提高代码的影响能力

    什么是I/O

    • I/O【input/output】
    • 可以理解为从输入到输出之间的转化过程
    • 比如:敲键盘(输入),看到编辑器出现字符(输出)
    • 移动鼠标(输入),看到光标移动(输出)

    回调函数的设计

    • 对于一个函数如果需要定义一个回调函数:
      • 回调函数的一定作为参数的最后一个参数出现:1、function getFile(name,job,callback){}
      • 回调函数的第一个参数默认接收错误信息,第二个参数才是真正的回调数据(便于外界获取调用的错误情况);如:getFile('jason','coding',function(error,data){if(error) throw error;console.log(data);});
    • Node统一约定
      • 强调错误优先,因为之后的操作大部分都是异步方式,无法通过try catch捕获异常,所以错误优先的回调函数第一个参数为上一步的错误信息
    非阻塞I/O-->其实就是Node的核心特性

    事件驱动和非阻塞机制

    • Node平台将一个任务连同该任务的回调函数放到一个事件循环系统中
    • 事件循环高效的管理系统池同时高效执行每一个任务
    • 当任务执行完成过后自动咨询回调函数

    非阻塞的优势

    • 提高代码的响应效率
    • 充分利用单核CPU的优势
    • 改善I/O的不可预测带来的问题
    • 如何提高一个人的工作效率
    • 但是:目前大多数都是多核CUP

    异步回调的问题(相对于传统的代码)

    • 异步事件驱动的代码
    • 不易阅读
    • 不易调试
    • 不易维护

    进程和线程

    1、进程(XXX.exe)
    • 每一个正在运行的应用程序称之为进程
    • 每一个应用程序至少有一个进程
    • 进程是用来给应用程序提供一个运行环境
    • 进程是操作系统为应用程序分配资源的一个单位
    2、线程
    • 用来执行应用程序中的代码
    • 在一个进程内部,可以有多个线程
    • 在一个线程内部,同事只能做一件事
    • 而且传统的开发方式大部分都是I/O阻塞的
    • 所以需要多线程来更好的利用硬件资源
    • 给人一种错觉:线程越多越好

    二、模块化结构

    • Node实现的CommonJS规范,所以可以使用模块化的方式组织代码结构
    • Node采用的模块化结果是按照CommonJS规范
    • 模块与文件是一一对应关系,即加载一个模块,实际上就是加载对应的一个模块文件

    CommonJS规范概述

    CommonJS模块的特点

    • 所有代码都运行在模块作用域,不会污染全局作用域
    • 模块可以多次加载,但是只会在第一次加载时运行一次,然后运行结果就被缓存了,以后再加载,就直接读取缓存结果,要想让模块再次运行,必须清除缓存
    • 模块加载的顺序,按照其在代码中出现的顺序

    module对象

    • Node内部提供一个Module构建函数,所有模块都是Module的实例,属性如下:
      • module.id模块的识别符,通常是带有绝对路径的模块文件名
      • module.filename模块定义的文件的绝对路径
      • module.loaded返回一个布尔值,表示模块是否已经完成加载
      • module.parent返回一个对象,表示调用该模块的模块
      • module.children返回一个数组,表示该模块要用到的其他模块
      • module.exports表示模块对外输出的值!
    • 载入一个模块就是构建一个Module实例

    模块的定义

    • 一个新的JS文件就是一个模块
    • 一个合格的模块应该是有导出成员的,否则模块就失去了第一的意义
    • 模块内部是一个独立(封闭)的作用域,模块之间不会冲突
    • 模块之间必须通过导出导入的方式协同
    • 导出方式:
      • exports.name=value;
      • module.exports={name:value};
    • 此时就没必要在模块内部写自执行函数了!!
    • module.exports和exports
    • module.exports是用于为模块导出成员的接口
    • exports是指向module.exports的别名,相当于在模块开始的时候执行:var exports=module.exports;
    • 一旦为module.exports赋值,就会切断之前两者的相关性
    • 最终模块的导出成员以module.exports为准

    模块的分类

    • 文件模块:就是我们自己写的功能模块文件
    • 核心模块:Node平台自带的一套基本的功能模块,也有人称之为Node平台的API
    • 第三方模块:社区或第三方个人开发好的功能模块,可以直接拿来用

    模块化开发的流程

    • 创建模块:new some.js
    • 导出成员:module.exports={}
    • 载入模块:var some=require('./some.js');
    • 使用模块:some.add(1,2);

    模块内全局环境(伪)

    • 我们之后的文件操作中必须使用绝对路径
    • __dirname:用于获取当前文件所在目录的完整路径,在REPL环境无效
    • __filename:用来获取当前文件的完整路径,在REPL环境同样无效
    • module:模块对象
    • exports:映射到module.exports的别名
    • require():require.cache、require.extensions、require.main、require.resolve()

    练习:命令行下的计算器,c.js和e.js

    • c.js里是非模块化的
    • e.js是模块化的(CommonJS规范)
    • module.exports={a,b,c};ES6的'自动属性'

    模块的载入:require函数简介

    • node使用CommonJS模块规范,内置的require函数用于加载模块的文件
    • require的基本功能是:读取并执行一个javascript文件,然后返回该模块的exports对象
    • 如果没有发现指定的模块会报错!

    require扩展名:规则>描述的越详细越准确,类似CSS的选择器优先级

    • require加载文件时可以省略扩展名:require('./module1')
      • 此时会默认先找require('./module1.js');
      • 如果默认的.js文件不存在接下来会继续默认找require('./module1.json');

    加载文件规则

    • 通过"./"或"../"开头:则按照相对路径熊当前文件所在文件夹开始寻找模块:require('../xxx.js')>上级目录下找xxx.js文件;
    • 通过/开头:则以系统根目录开始寻找模块:require('/Users/xxx/Documents/sss.js')>以绝对路径的方式找,无意义!;
    • 如果参数字符串不以"./"或者"/"开始,则表示加载的是一个默认提供的核心模块(位于Node的系统安装目录中):require('fs')、require('http')、require('path')等等;
    • 或者从当前目录向上搜索node_modules目录中的文件:require('my_module')>各级node_modules夹中搜索my_nodule.js文件;
    • 如果require传入的是一个目录的路径,会自动查看该目录的package.json文件,然后加载main字段指定的入口文件
    • 如果package.json文件没有main字段,或者根本就没有package.json文件,则默认找目录下的index.js文件作为模块:require('./filesModule')>当前目录下找filesModule目录中的index.js文件;

    注意require('filename'):

    • 加载的模块名字如果跟系统的模块冲突,系统的优先级最高!
    • 就近原则去找node_modules文件夹下的filename文件

    require的实现机制:自定义的的$require,g.js和模块module2.js

    • 将传入的模块的ID通过加载规则找到对应的模块文件
    • 读取这个文件里面的代码
    • 通过拼接的方式为该段代码构建私有空间
    • 执行该代码
    • 拿到module.exports返回

    __dirname与__filename:如f.js

    • __dirname当前文件所在目录的完整路径
    • __filename当前文件的完整路径 ###思考与练习:
    • 1、如何定义和导入模块?
    • 2、如何使用模块化的方式组织代码结构?
    • 3、载入模块的规则
    • 4、了解模块的加载机制
    • 5、自定义一个$require类似require,如$require下面的:index.js

    模块的缓存:如h.js

    • 第一次加载某个模块时,Node会缓存该模块,以后再加载该模块就直接从缓存取出该模块的module.exports属性(不会再次执行该模块);
    • 如果需要多次执行模块中的代码,一般可以让模块暴露行为(函数)
    • 模块的缓存可以通过require.cache拿到,同样也可以删除

    缓存是如何实现(如:i.js)

    如何删除缓存(如:i.js),一般不会删除缓存

    • 缓存require.cache ##

    三、Node:核心模块、文件操作、文件流、网络操作

    1、核心模块

    • 核心模块的意义:

      • 如果只是在服务器运行javascript代码,意义不大,因为无法实现任何功能(读写文件,访问网络)
      • Node的用处在于它本身还提供的一系列功能模块,用于操作系统运动
      • 这些核心的功能模块在Node中的内置
    • 内置的有如下模块

      • path:处理文件路径
      • fs:操作文件系统
      • child_process:新建子进程
      • util:提供一些列实用的小工具
      • http:提供HTTP服务器功能
      • url:用于解析URL
      • querystring:解析URL中的查询字符串
      • crypto:提供加密和解密功能(MD5是不可逆,无法解密!!)
      • 其他

    Node Package

    由于Node是一套轻内核的平台,虽然提供了一系列的内置模块,但是不足以满足开发者的需求,于是出现了包(Package)的概念。与核心模块类似,就是讲一些预先设计好的功能或者API封装到一个文件夹,提供给开发者使用

    1):包的加载机制
    • 与内置模块相同,同样使用require方法,如:const express=require('express');
    • 加载机制也和内置模块加载机制相同
    • 加载注意事项:
      • 先在系统核心(优先级最高)的模块中找
      • 然后再到当前项目中的:node_mocules目录中去找
    2):如何管理好自己的包
    • 由于Node本身并没有太多的功能性API。所以市面上涌现出大量的第三方人员开发出来的Package
    • 包的生态圈一旦繁荣起来,就必须有工具去代替人脑或者文档的方式去管理
    • 此时NPM诞生了~

    NPM(Node Package Manager)

    • 随着时间的发展,NPM出现了两层概念:
      • 一层韩式是Node的开放式模块登记和管理系统,也可以说是一个生态圈,一个社区
      • 另一层含义是Node默认的模块管理器,是一个命令行下的软件,用来安装和管理Node模块
    • 官方地址:https://www.npmjs.com/
    • 国内加速镜像地址:https://npm.taobao.org/
    1):安装NPM
    • NPM不需要单独安装,默认在安装Node的时候会自动一起安装NPM
    • 但是Node附带的NPM可能不是最新版本,最好用下面的命令更新到最新版本:$ npm install npm -g
    • 默认安装到当前系统Node所在的目录下
    • 由于之前使用NVM的方式安装的Node所以需要重新配置NPM全局目录
    2):配置NPM的全局目录?
    • npm config set prefix [filepath]
    • 将NPM目录配置到其他目录时,必须将该目录放到环境变量中,否则无法在全局使用
    3):NPM常用命令->https://docs.npmjs.com/
    • npm config [ls|list|set|get] [name] [value]
    • npm init [--yes|-y]
    • npm search [name]
    • npm info [name]
    • npm install [--global|-g] [name]
    • npm uninstall [--global|-g] [name]
    • npm list [--global|-g]
    • npm outdated [--global|-g]
    • npm update [--global|-g] [name]
    • npm run [task]
    • npm cache [clean]

    2、文件操作(important)

    Node内核提供了很多与文件操作相关的模块,每个模块都提供了一些最基本的操作API,在NPM中也有社区提供的功能包

    fs:
    • 基础的文件操作API ######path:
    • 提供和路径相关的操作API ######readline:
    • 用于读取大文本文件,一行一行读 ######fs-extra:(第三方)
    • https://www.npmjs.com/package/fs-extra

    1):同步或者异步调用(如:day03/code/02.js)

    • fs模块对文件的几乎所有操作都有同步和异步两种形式
      • readFile()
      • readFileSync()
    • 区别:
      • 同步调用会阻塞代码的执行,异步则不会
      • 异步调用会将读取任务下达到任务队列,直到任务执行完成才会回调
      • 异常处理方面,同步必须使用 try catch 方式,异步可以通过回调函数的第一个参数
      • 同步比异步更加耗费时间

    2):路径模块

    在文件操作中,都必须使用物理路径(绝对路径),path模块提供了一些列与路径相关的API

    3):什么是缓冲区

    • 缓冲区就是内存中操作数据的容器
    • 只是数据的容器而已
    • 通过缓冲区可以很方便的操作二进制数据
    • 而且在大文件操作时必须有缓冲区
    如果读取文件没有指定编码的话,默认读取的是一个Buffer(缓冲区)
    • Buffer的作用,对咱们目前的情况来说只很少使用场景
      • 将图片转成BASE64编码(code/04.js)
      • 对文字进行BASE64编码

    4):为何要有缓冲区

    • JS是比较擅长处理字符串,但是早起的应用场景主要用于处理HTML文档,不会有太大批次的数据处理,也不会接触到二进制的数据
    • 而在Node中操作数据、网络通信是没办法完全以字符串的方式操作的
    • 所以在Node中引入了一个二进制的缓冲区的实现:Buffer
    • buf.toString([encoding[, start[, end]]])这里也可以设置编码:如code/03.js
  • 相关阅读:
    序号正确
    基于visual Studio2013解决C语言竞赛题之1071打印工资
    基于visual Studio2013解决C语言竞赛题之1070删除相同节点
    基于visual Studio2013解决C语言竞赛题之1069链表查找删除
    基于visual Studio2013解决C语言竞赛题之1068指针数组
    基于visual Studio2013解决C语言竞赛题之1067间隔排序
    基于visual Studio2013解决C语言竞赛题之1066进制转化
    基于visual Studio2013解决C语言竞赛题之1065二维排序
    基于visual Studio2013解决C语言竞赛题之1064互质数差1验证
    基于visual Studio2013解决C语言竞赛题之1063分橘子
  • 原文地址:https://www.cnblogs.com/-walker/p/6070472.html
Copyright © 2011-2022 走看看