NodeJs(尚硅谷视频学习笔记)

内容来自尚硅谷Nodejs学习课件以及本身添加node

课前预热:CMD基本知识

1.命令行窗口(小黑屏)、CMD窗口、终端、shell

- 开始菜单 --> 运行 --> CMD --> 回车
- Win + R --> CMD --> 回车
- 经常使用的指令
	dir 			列出当前目录下的全部文件
	cd 目录名   		进入到指定的目录
	md 目录名		建立一个文件夹
	rd 目录名		删除一个文件夹
-目录
	. 表示当前目录
	..  表示上一级目录
-环境变量(window系统中的变量 )
	PATH 
		一个个的路径
		C:\Program Files\Microsoft SQL Server\130\Tools\Binn\
	- 当咱们在命令行窗口打开一个文件,或调用一个程序时,
		系统会首先在当前目录下寻找文件程序,若是找到了则直接打开
		若是没有找到则会依次到环境变量path的路径中寻找,直到找到为止
		若是没找到则报错
		相似于做用域链
	-因此咱们能够将一些常常须要访问的程序和文件的路径添加到path中,
		这样咱们就能够在任意位置来访问这些文件和程序了

2.进程和线程

进程
	- 进程负责为程序的运行提供必备的环境
	- 进程就至关于工厂中的车间

线程
	- 线程是计算机中最小的计算单位,线程负责执行进程中的程序
	- 线程就至关于工厂中的工人
单线程
	JS是单线程,在执行JS代码的时候网页是中止渲染的。
多线程
	主流的语言,如Java

Node.js简介

• Node.js是一个可以在服务器端运行JavaScript的开放源代码、跨平台JavaScript运行环境web

• Node采用Google开发的V8引擎运行js代码,使用事件驱动非阻塞异步I/O模型等技术来提升性能,可优化应用程序的传输量和规模。
在这里插入图片描述
• Node大部分基本模块都用JavaScript编写。在Node出现以前,JS一般做为客户端程序设计语言使用,以JS写出的程序常在用户的浏览器上运行。算法

• 目前,Node已被IBM、Microsoft、Yahoo!、Walmart、Groupon、SAP、 LinkedIn、Rakuten、PayPal、Voxer和GoDaddy等企业采用。shell

• Node主要用于编写像Web服务器同样的网络应用,这和PHP和Python是相似的。npm

• 可是Node与其余语言最大的不一样之处在于,PHP等语言是阻塞的而Node是非阻塞的。编程

• Node是事件驱动的。开发者能够在不使用线程的状况下开发出一个可以承载高并发的服务器。其余服务器端语言难以开发高并发应用,并且即便开发出来,性能也不尽人意。json

• Node正是在这个前提下被创造出来。后端

• Node把JS的易学易用和Unix网络编程的强大结合到了一块儿。跨域

• Node.js容许经过JS和一系列模块来编写服务器端应用和网络相关的应用。数组

• 核心模块包括文件系统I/O、网络(HTTP、TCP、UDP、DNS、TLS/SSL等)、二进制数据流、加密算法、数据流等等。Node模块的API形式简单,下降了编程的复杂度。

• 使用框架能够加速开发。经常使用的框架有Express.js、Socket.IO和Connect等。Node.js的程序能够在Microsoft Windows、Linux、Unix、Mac OS X等服务器上运行。

• Node.js也可使用CoffeeScript、TypeScript、Dart语言,以及其余可以编译成JavaScript的语言编程。

Node的用途

• Web服务API,好比REST
• 实时多人游戏
• 后端的Web服务,例如跨域、服务器端的请求
• 基于Web的应用
• 多客户端的通讯,如即时通讯

简介总结

I/O (Input/Output)
	- I/O操做指的是对磁盘的读写操做

Node
	- Node是对ES标准一个实现,Node也是一个JS引擎
	- 经过Node可使js代码在服务器端执行
	- Node仅仅对ES标准进行了实现,因此在Node中不包含DOM 和 BOM	
	- Node中可使用全部的内建对象
		String Number Boolean Math Date RegExp Function Object Array
		而BOM和DOM都不能使用
			可是可使用 console 也可使用定时器(setTimeout() setInterval())
			
	- Node能够在后台来编写服务器
		Node编写服务器都是单线程的服务器
		- 进程
			- 进程就是一个一个的工做计划(工厂中的车间)
		- 线程
			- 线程是计算机最小的运算单位(工厂中的工人)
				线程是干活的
				
	- 传统的服务器都是多线程的
		- 每进来一个请求,就建立一个线程去处理请求
		
	- Node的服务器单线程的
		- Node处理请求时是单线程,可是在后台拥有一个I/O线程池

在VScode中调出终端查看node.js代码执行

使用快捷键: ctrl + ·

COMMONJS规范

ECMAScript标准的缺陷

• 没有模块系统
• 标准库较少
• 没有标准接口
• 缺少管理系统

模块化

• 若是程序设计的规模达到了必定程度,则必须对其进行模块化。

• 模块化能够有多种形式,但至少应该提供可以将代码分割为多个源文件的机制。

• CommonJS 的模块功能能够帮咱们解决该问题。

CommonJS规范

• CommonJS规范的提出,主要是为了弥补当前JavaScript没有模块化标准的缺陷。

• CommonJS规范为JS指定了一个美好的愿景,但愿JS可以在任何地方运行。

• CommonJS对模块的定义十分简单:

– 模块引用
– 模块定义
– 模块标识

模块引用

• 在规范中,定义了require()方法,这个方法接手模块标识,以此将一个模块引入到当前运行环境中。

• 模块引用的示例代码:
– var math = require(‘math’);

模块定义

• 在运行环境中,提供了exports对象用于导出当前模块的方法或者变量,而且它是惟一的导出的出口。

• 在模块中还存在一个module对象,它表明模块自身,而exports是module的属性。

• 在Node中一个文件就是一个模块。

exports.xxx = function() {};
module.exports = {};

模块标识

• 模块标识其实就是模块的名字,也就是传递给require()方法的参数,它必须是符合驼峰命名法的字符串,或者是以.、…开头的相对路径、或者绝对路径。

• 模块的定义十分简单,接口也十分简洁。每一个模块具备独立的空间,它们互不干扰,在引用时也显得干净利落。

模块化总结

模块化
	- 在Node中,一个js文件就是一个模块
	- 在Node中,每个js文件中的js代码都是独立运行在一个函数中
		而不是全局做用域,因此一个模块的中的变量和函数在其余模块中没法访问

03module.js

//引入其余的模块
/*
在node中,经过require()函数来引入外部的模块
	require()能够传递一个文件的路径做为参数,node将会自动根据该路径来引入外部模块
	这里路径,若是使用相对路径,必须以.或..开头

使用require()引入模块之后,该函数会返回一个对象,这个对象表明的是引入的模块

咱们使用require()引入外部模块时,使用的就是模块标识,咱们能够经过模块标识来找到指定的模块
- 模块分红两大类
	核心模块
		- 由node引擎提供的模块
		- 核心模块的标识就是,模块的名字
		- 示例:var fs = require("fs");
	文件模块
		- 由用户本身建立的模块
		- 文件模块的标识就是文件的路径(绝对路径,相对路径)
			相对路径使用.或..开头

 */
//var md = require("./02.module");
var math = require("./math");
var fs = require("fs");

//console.log(md);
console.log(math.add(123,456));
//console.log(fs);
	
02module.js
/*
咱们能够经过 exports 来向外部暴露变量和方法
只须要将须要暴露给外部的变量或方法设置为exports的属性便可
*/
//向外部暴露属性或方法
exports.x = "我是02.module.js中的x";
exports.y = "我是y";
exports.fn = function () {

};

04module.js
/*
在node中有一个全局对象 global,它的做用和网页中window相似
	在全局中建立的变量都会做为global的属性保存
	在全局中建立的函数都会做为global的方法保存

当node在执行模块中的代码时,它会首先在代码的最顶部,添加以下代码
		function (exports, require, module, __filename, __dirname) {

在代码的最底部,添加以下代码
		}

实际上模块中的代码都是包装在一个函数中执行的,而且在函数执行时,同时传递进了5个实参
	 exports
	 	- 该对象用来将变量或函数暴露到外部

	 require
	 	- 函数,用来引入外部的模块

	 module
	 	- module表明的是当前模块自己
	 	- exports就是module的属性
	 	- 既可使用 exports 导出,也可使用module.exports导出

	 __filename
		C:\Users\lilichao\WebstormProjects\class0705\01.node\04.module.js
		- 当前模块的完整路径

  	 __dirname
		C:\Users\lilichao\WebstormProjects\class0705\01.node
		- 当前模块所在文件夹的完整路径

*/
//console.log(global.a);
/*
	arguments.callee
		- 这个属性保存的是当前执行的函数对象
*/
//console.log(arguments.callee + "");
//console.log(arguments.length);

//console.log(exports);
//console.log(module.exports == exports);

console.log(__dirname);

05module.js
/*
exports 和 module.exports
	- 经过exports只能使用.的方式来向外暴露内部变量
		exports.xxx = xxx

	- 而module.exports既能够经过.的形式,也能够直接赋值
		module.exports.xxx = xxxx
		module.exports = {}
 */

Node的模块实现

• Node中虽然使用的是CommonJS规范,可是其自身也对规范作了一些取舍。

• 在Node中引入模块,须要经历以下3个步骤:
– 路径分析
– 文件定位
– 编译执行

• 在Node中,模块分为三类:一类是底层由C++编写的内建模块,一类是Node提供的核心模块;还有一类是用户编写的模块,称为文件模块。

包 package

• CommonJS的包规范容许咱们将一组相关的模块组合到一块儿,造成一组完整的工具。

• CommonJS的包规范由包结构和包描述文件两个部分组成。

• 包结构
– 用于组织包中的各类文件

• 包描述文件
– 描述包的相关信息,以供外部读取分析

包结构

• 包实际上就是一个压缩文件,解压之后还原为目录。符合规范的目录,应该包含以下文件:

– package.json 描述文件
– bin 可执行二进制文件
– lib js代码
– doc 文档
– test 单元测试

包描述文件

• 包描述文件用于表达非代码相关的信息,它是一个JSON格式的文件 – package.json,位于包的根目录下,是包的重要组成部分。

• package.json中的字段

– name、description、version、keywords、maintainers、contributors、bugs、licenses、repositories、dependencies、homepage、os、cpu、engine、builtin、directories、implements、scripts、author、bin、main、devDependencies。

在JSON文件不能写注释

NPM(Node Package Manager)

• CommonJS包规范是理论,NPM是其中一种实践。

• 对于Node而言,NPM帮助其完成了第三方模块的发布、安装和依赖等。借助NPM,Node与第三方模块之间造成了很好的一个生态系统。

NPM命令

• npm –v
– 查看版本
• npm
– 帮助说明
• npm search 包名
– 搜索模块包
• npm install
–下载当前项目所依赖的包
• npm install 包名
– 在当前目录安装包
• npm install 包名 –g
– 全局模式安装包(全局安装的包通常都是一些工具)
• npm remove 包名
– 删除一个模块
• npm install 包名 --save
– 安装包并添加到依赖中
• npm install 文件路径
– 从本地安装
• npm install 包名 –registry=地址
– 从镜像源安装
• npm config set registry 地址
– 设置镜像源

NPM总结笔记

/*
	经过npm下载的包都放到node_modules文件夹中
		咱们经过npm下载的包,直接经过包名引入便可

	node在使用模块名字来引入模块时,它会首先在当前目录的node_modules中寻找是否含有该模块
		若是有则直接使用,若是没有则去上一级目录的node_modules中寻找
		若是有则直接使用,若是没有则再去上一级目录寻找,直到找到为止
		直到找到磁盘的根目录,若是依然没有,则报错

 */

Buffer(缓冲区)

• 从结构上看Buffer很是像一个数组,它的元素为16进制的两位数。
JS数组性能比其余语言的数组差

• 实际上一个元素就表示内存中的一个字节。

• 实际上Buffer中的内存不是经过JavaScript分配的,而是在底层经过C++申请的。

• 也就是咱们能够直接经过Buffer来建立内存中的空间。

Buffer的操做

• 使用Buffer保存字符串

let str = "你好 atguigu";
let buf = Buffer.from(str , "utf-8");

• 建立指定大小的Buffer对象

let buf3 = Buffer.alloc(1024*8)

Buffer的转换

Buffer与字符串间的转换

– 支持的编码:
• ASCII、UTF-八、UTF-16LE/UCS-二、Base6四、Binary、Hex

– 字符串转Buffer
• Buffer.from(str , [encoding]);

– Buffer转字符串
• buf.toString([encoding] , [start] , [end]);

写入操做

• 向缓冲区中写入字符串
buf.write(string[, offset[, length]][, encoding])

• 替换指定索引位置的数据
buf[index]

• 将指定值填入到缓冲区的指定位置
buf.fill(value[, offset[, end]][, encoding])

读取操做

• 将缓冲区中的内容,转换为一个字符串返回
buf.toString([encoding[, start[, end]]])

• 读取缓冲区指定索引的内容
buf[index]

其余操做

• 复制缓冲区
buf.copy(target[, targetStart[, sourceStart[, sourceEnd]]])

• 对缓冲区切片
buf.slice([start[, end]])

• 拼接缓冲区
Buffer.concat(list[, totalLength])

Buffer总结笔记

/*
	Buffer(缓冲区)
		- Buffer的结构和数组很像,操做的方法也和数组相似
		- 数组中不能存储二进制的文件,而buffer就是专门用来存储二进制数据
		- 使用buffer不须要引入模块,直接使用便可
		- 在buffer中存储的都是二进制数据,可是在显示时都是以16进制的形式显示
			buffer中每个元素的范围是从00 - ff   0 - 255
			00000000 - 11111111

			计算机 一个0 或一个1 咱们称为1位(bit)

			8bit = 1byte(字节)
			1024byte = 1kb
			1024kb = 1mb
			1024mb = 1gb
			1024gb = 1tb

			buffer中的一个元素,占用内存的一个字节

		- Buffer的大小一旦肯定,则不能修改,Buffer其实是对底层内存的直接操做

 */
 var str = "Hello 尚硅谷";

//将一个字符串保存到buffer中
var buf = Buffer.from(str);
//建立一个指定大小的buffer
//buffer构造函数都是不推荐使用的
//var buf2 = new Buffer(10);//10个字节的buffer
//console.log(buf2.length);

//建立一个10个字节的buffer
var buf2 = Buffer.alloc(10);
//经过索引,来操做buf中的元素
buf2[0] = 88;
buf2[1] = 255;
buf2[2] = 0xaa;
buf2[3] = 255;

//只要数字在控制台或页面中输出必定是10进制
//console.log(buf2[2].toString(16));

/*for(var i=0 ; i<buf2.length ; i++){
	console.log(buf2[i]);
}*/

//Buffer.allocUnsafe(size) 建立一个指定大小的buffer,可是buffer中可能含有敏感数据
/*var buf3 = Buffer.allocUnsafe(10);
console.log(buf3);*/

/*
	Buffer.from(str) 将一个字符串转换为buffer
	Buffer.alloc(size) 建立一个指定大小的Buffer
	Buffer.alloUnsafe(size) 建立一个指定大小的Buffer,可是可能包含敏感数据
 	buf.toString() 将缓冲区中的数据转换为字符串
 */

var buf4 = Buffer.from("我是一段文本数据");

console.log(buf4.toString());

fs(文件系统)

• 在Node中,与文件系统的交互是很是重要的,服务器的本质就将本地的文件发送给远程的客户端

• Node经过fs模块来和文件系统进行交互

• 该模块提供了一些标准文件访问API来打开、读取、写入文件,以及与其交互。

• 要使用fs模块,首先须要对其进行加载
const fs = require("fs");

同步和异步调用

• fs模块中全部的操做都有两种形式可供选择同步和异步。

• 同步文件系统会阻塞程序的执行,也就是除非操做完毕,不然不会向下执行代码。

• 异步文件系统不会阻塞程序的执行,而是在操做完成时,经过回调函数将结果返回。

打开和关闭文件

• 打开文件

– fs.open(path, flags[, mode], callback)
– fs.openSync(path, flags[, mode])

• 关闭文件

– fs.close(fd, callback)
– fs.closeSync(fd)

打开状态

在这里插入图片描述

写入文件

• fs中提供了四种不一样的方式将数据写入文件
– 简单文件写入
– 同步文件写入
– 异步文件写入
– 流式文件写入

简单文件写入

• fs.writeFile(file, data[, options], callback)
• fs.writeFileSync(file, data[, options])
• 参数:
– file 文件路径
– data 被写入的内容,能够是String或Buffer
– options 对象,包含属性(encoding、mode、flag)
– callback 回调函数

同步文件写入

• fs.writeSync(fd, buffer, offset, length[, position])
• fs.writeSync(fd, data[, position[, encoding]])
• 要完成同步写入文件,先须要经过openSync()打开文件来获取一个文件描述符,而后在经过writeSync()写入文件。
• 参数
– fd 文件描述符,经过openSync()获取
– data 要写入的数据(String 或 Buffer)
– offset buffer写入的偏移量
– length 写入的长度
– position 写入的起始位置
– encoding 写入编码

异步文件写入

• fs.write(fd, buffer, offset, length[, position], callback)
• fs.write(fd, data[, position[, encoding]], callback)
• 要使用异步写入文件,先须要经过open()打开文件,而后在回调函数中经过write()写入。
• 参数:
– fd 文件描述符
– data 要写入的数据(String 或 Buffer)
– offset buffer写入的偏移量
– length 写入的长度
– position 写入的起始位置
– encoding 写入编码

流式文件写入

• 往一个文件中写入大量数据时,最好的方法之一是使用流。
• 若要将数据异步传送到文件,首须要使用如下语法建立一个Writable对象:
– fs.createWriteStream(path[, options])
• path 文件路径
• options {encoding:"",mode:"",flag:""}
• 一旦你打开了Writable文件流,就可使用write()方法来写入它,写入完成后,在调用end()方法来关闭流。

读取文件

• fs中提供了四种读取文件的方式
– 简单文件读取
– 同步文件读取
– 异步文件读取
– 流式文件读取

简单文件读取

• fs.readFile(file[, options], callback)
• fs.readFileSync(file[, options])
– 参数:
• file 文件路径或文件描述符

• options <Object> | <String>
– encoding <String> | <Null> 默认 = null
– flag <String> 默认 = 'r'
• callback 回调函数,有两个参数err 、data

同步文件读取

• fs.readSync(fd, buffer, offset, length, position)
– 参数:
• fd 文件描述符
• buffer 读取文件的缓冲区
• offset buffer的开始写入的位置
• length 要读取的字节数
• position 开始读取文件的位置

异步文件读取

• fs.read(fd, buffer, offset, length, position, callback)
– 参数:
• fd 文件描述符
• buffer 读取文件的缓冲区
• offset buffer的开始写入的位置
• length 要读取的字节数
• position 开始读取文件的位置
• callback 回调函数 参数err , bytesRead , buffer

流式文件读取

• 从一个文件中读取大量的数据时,最好的方法之一就是流式读取,这样将把一个文件做为Readable流的形式打开。
• 要从异步从文件传输数据,首先须要经过如下语法建立一个Readable流对象:
– fs.createReadStream(path[, options])
• path 文件路径
• options {encoding:"",mode:"",flag:""}
• 当你打开Readable文件流之后,能够经过readable事件和read()请求,或经过data事件处理程序轻松地从它读出。

其余操做

• 验证路径是否存在
– fs.exists(path,callback)
– fs.existsSync(path)
• 获取文件信息
– fs.stat(path, callback)
– fs.statSync(path)
• 删除文件
– fs.unlink(path, callback)
– fs.unlinkSync(path)
• 列出文件
– fs.readdir(path[, options], callback)
– fs.readdirSync(path[, options])
• 截断文件
– fs.truncate(path, len, callback)
– fs.truncateSync(path, len)
• 创建目录
– fs.mkdir(path[, mode], callback)
– fs.mkdirSync(path[, mode])
• 删除目录
– fs.rmdir(path, callback)
– fs.rmdirSync(path)
• 重命名文件和目录
– fs.rename(oldPath, newPath, callback)
– fs.renameSync(oldPath, newPath)
• 监视文件更改写入
– fs.watchFile(filename[, options], listener)

fs总结笔记

/*
	文件系统(File System)
		- 文件系统简单来讲就是经过Node来操做系统中的文件
		- 使用文件系统,须要先引入fs模块,fs是核心模块,直接引入不须要下载

	同步文件的写入
		- 手动操做的步骤
			1.打开文件
 				fs.openSync(path, flags[, mode])
 					- path 要打开文件的路径
 					- flags 打开文件要作的操做的类型
 						r 只读的
 						w 可写的
 					- mode 设置文件的操做权限,通常不传
				 返回值:
				 - 该方法会返回一个文件的描述符做为结果,咱们能够经过该描述符来对文件进行各类操做

			2.向文件中写入内容
 				fs.writeSync(fd, string[, position[, encoding]])
 					- fd 文件的描述符,须要传递要写入的文件的描述符
 					- string 要写入的内容
 					- position 写入的起始位置
 					- encoding 写入的编码,默认utf-8

			3.保存并关闭文件
 				fs.closeSync(fd)
 					- fd 要关闭的文件的描述符



 */
var fs = require("fs");

//打开文件
var fd = fs.openSync("hello.txt" , "w");

//向文件中写入内容
fs.writeSync(fd , "今每天气真不错~~~", 2);

//关闭文件
fs.closeSync(fd);

console.log("程序向下执行~~~");
/**
	异步文件写入
 fs.open(path, flags[, mode], callback)
 	- 用来打开一个文件
    - 异步调用的方法,结果都是经过回调函数的参数返回的
 	- 回调函数两个参数:
 		err 错误对象,若是没有错误则为null
 		fd  文件的描述符
 fs.write(fd, string[, position[, encoding]], callback)
 	- 用来异步写入一个文件

 fs.close(fd, callback)
 	- 用来关闭文件

 */

//引入fs模块
var fs = require("fs");


//打开文件
fs.open("hello2.txt","w",function (err , fd) {
	//判断是否出错
	if(!err){
		//若是没有出错,则对文件进行写入操做
		fs.write(fd,"这是异步写入的内容",function (err) {
			if(!err){
				console.log("写入成功~~");
			}
			//关闭文件
			fs.close(fd , function (err) {
				if(!err){
					console.log("文件已关闭~~~");
				}
			});
		});
	}else{
		console.log(err);
	}
});

console.log("程序向下执行~~~");
/*
	简单文件写入
	 fs.writeFile(file, data[, options], callback)
	 fs.writeFileSync(file, data[, options])
		- file 要操做的文件的路径
		- data 要写入的数据
		- options 选项,能够对写入进行一些设置
		- callback 当写入完成之后执行的函数

		- flag
			r 只读
			w 可写
			a 追加
 */
//引入fs模块
var fs = require("fs");

/*fs.writeFile("hello3.txt","这是经过writeFile写入的内容",{flag:"r+"} , function (err) {
	if(!err){
		console.log("写入成功~~~");
	}else{
		console.log(err);
	}
});*/


//C:\Users\lilichao\Desktop\hello.txt
//C:\\Users\\lilichao\\Desktop\\hello.txt

fs.writeFile("C:/Users/lilichao/Desktop/hello.txt","这是经过writeFile写入的内容",{flag:"w"} , function (err) {
	if(!err){
		console.log("写入成功~~~");
	}else{
		console.log(err);
	}
});
/*
	同步、异步、简单文件的写入都不适合大文件的写入,性能较差,容易致使内存溢出
 */
var fs = require("fs");

//流式文件写入
//建立一个可写流
/*
	fs.createWriteStream(path[, options])
		- 能够用来建立一个可写流
		- path,文件路径
		- options 配置的参数
 */
var ws = fs.createWriteStream("hello3.txt");

//能够经过监听流的open和close事件来监听流的打开和关闭
/*
	on(事件字符串,回调函数)
		- 能够为对象绑定一个事件

	once(事件字符串,回调函数)
		- 能够为对象绑定一个一次性的事件,该事件将会在触发一次之后自动失效

* */
ws.once("open",function () {
	console.log("流打开了~~~");
});

ws.once("close",function () {
	console.log("流关闭了~~~");
});

//经过ws向文件中输出内容
ws.write("经过可写流写入文件的内容");
ws.write("今每天气真不错");
ws.write("锄禾日当午");
ws.write("红掌拨清清");
ws.write("清清真漂亮");

//关闭流,关闭写的那个文件的管口
ws.end();
//关闭流,关闭读的那个文件的管口
/*
	1.同步文件读取
	2.异步文件读取
	3.简单文件读取
	 fs.readFile(path[, options], callback)
	 fs.readFileSync(path[, options])
	 	- path 要读取的文件的路径
	 	- options 读取的选项
	 	- callback回调函数,经过回调函数将读取到内容返回(err , data)
	 		err 错误对象
	 		data 读取到的数据,会返回一个Buffer

	4.流式文件读取
 */

var fs = require("fs");

var path = "C:/Users/lilichao/Desktop/笔记.mp3";

fs.readFile("an.jpg" , function (err , data) {
	if(!err){
		//console.log(data);
		//将data写入到文件中
		fs.writeFile("C:/Users/lilichao/Desktop/hello.jpg",data,function(err){
			if(!err){
				console.log("文件写入成功");
			}
		} );
	}
});
/*
	流式文件读取也适用于一些比较大的文件,能够分屡次将文件读取到内存中
 */

var fs = require("fs");

//建立一个可读流
var rs = fs.createReadStream("C:/Users/lilichao/Desktop/笔记.mp3");
//建立一个可写流
var ws = fs.createWriteStream("a.mp3");

//监听流的开启和关闭
rs.once("open",function () {
	console.log("可读流打开了~~");
});

rs.once("close",function () {
	console.log("可读流关闭了~~");
	//数据读取完毕,关闭可写流

	ws.end();
});

ws.once("open",function () {
	console.log("可写流打开了~~");
});

ws.once("close",function () {
	console.log("可写流关闭了~~");
});

//若是要读取一个可读流中的数据,必需要为可读流绑定一个data事件,data事件绑定完毕,它会自动开始读取数据
rs.on("data", function (data) {
	//console.log(data);
	//将读取到的数据写入到可写流中
	ws.write(data);
});
/*
	流式文件读取也适用于一些比较大的文件,能够分屡次将文件读取到内存中
 */

var fs = require("fs");

//建立一个可读流
var rs = fs.createReadStream("C:/Users/lilichao/Desktop/笔记.mp3");
//建立一个可写流
var ws = fs.createWriteStream("b.mp3");

//pipe()能够将可读流中的内容,直接输出到可写流中
rs.pipe(ws);
node
	- 包(package)
		- 将多个模块组合为一个完整的功能,就是一个包
		- 包结构
			bin
				- 二进制的可执行文件,通常都是一些工具包中才有
			lib
				- js文件
			doc
				- 文档
			test
				- 测试代码
			package.json
				- 包的描述文件
				
		- package.json	
			- 它是一个json格式的文件,在它里面保存了包各类相关的信息
				name 包名
				version 版本
				dependencies 依赖
				main 包的主要的文件
				bin 可执行文件
				
	- npm(Node Package Manager node的包管理器)
		- 经过npm能够对node中的包进行上传、下载、搜索等操做
		- npm会在安装完node之后,自动安装
		- npm的经常使用指令
			npm -v 查看npm的版本
			npm version 查看全部模块的版本
			npm init 初始化项目(建立package.json)
			npm i/install 包名 安装指定的包
			npm i/install 包名 --save 安装指定的包并添加依赖
			npm i/install 包名 -g 全局安装(通常都是一些工具)
			npm i/install 安装当前项目所依赖的包
			npm s/search 包名 搜索包	
			npm r/remove 包名 删除一个包
			
文件系统(File System)
	- Buffer(缓冲区)
		- Buffer和数组的结构的很是相似,Buffer是用来存储二进制数据的
		- Buffer的方法
			- Buffer.from(字符串)
				- 将一个字符串中内容保存到一个buffer中
			- buf.toString()
				- 将buffer转换为一个字符串
			- Buffer.alloc(size)
				- 建立一个指定大小的buffer对象
			- Buffer.allocUnsafe(size)
				- 建立一个指定大小的buffer对象,能够包含敏感数据
				
				
	- fs模块
		- 在Node经过fs模块来对系统中的文件进行操做,fs模块是node中已经继承好了,不须要在使用npm下载,直接引入便可
		- 引入fs
			var fs = require("fs");
		- fs模块中的大部分操做都提供了两种方法,同步方法和异步方法
			同步方法带sync
			异步方法没有sync,都须要回调函数
			
		- 写入文件
			1.同步写入
			2.异步写入
			3.简单写入
			4.流式写入
			
		- 读取文件
			1.同步读取
			2.异步读取
			3.简单读取
			4.流式读取
			
		- 方法
			- 打开文件
				fs.open(path, flags[, mode], callback)
				fs.openSync(path, flags[, mode])
				
			- 读写文件
				fs.write(fd, string[, position[, encoding]], callback)
				fs.writeSync(fd, string[, position[, encoding]])
				
				fs.read(fd, buffer, offset, length, position, callback)
				fs.readSync(fd, buffer, offset, length, position)
				
			- 关闭文件
				fs.close(fd,callback)
				fs.closeSync(fd);
				
			- 简单文件读取和写入
				fs.writeFile(file, data[, options], callback)
				fs.writeFileSync(file, data[, options])
				
				fs.readFile(path[, options], callback)
				fs.readFileSync(path[, options])
				
				
			- 流式文件读取和写入
				- 流式读取和写入适用于一些比较大的文件
					fs.createWriteStream(path[, options])
					fs.createReadStream(path[, options])