参考代码:https://github.com/michaelliao/learn-javascript
一、基本概念
1、node、npm的安装:不用说了
2、IDE:大家推荐Visual Studio Code,不过我自己还是更习惯用命令行。因为服务端开发,需要用的工具特别多。各种数据库各开一个,几个服务器再各开一个,开销就大了。使用批处理一次搞定,会比较方便。
3、模块:nodejs教程居然没有用class,而是使用模块。从‘女神快跑’项目看来,用class会使得代码干净的多。也许是因为老的工具库都是用模块的概念来实现面向对象的吧,如今我们自己写项目,可以考虑用class。
4、nodejs提供的一些全局变量:
(1)JavaScript有且仅有一个全局对象,在浏览器中,叫window对象。但在nodejs中,它叫global。
(2)process也是Node.js提供的一个对象,它代表当前Node.js进程。通过process对象可以拿到许多有用信息
5、基本工作方式:一个nodejs服务一般会创建一个轮询,不断接受客户端请求,然后对其作出响应。
二、一些基本模块:
1、fs模块:nodejs中读取文件的模块,它提供了同步和异步两种读取方式。
(1)异步代码:由于Node环境执行的JavaScript代码是服务器端代码,所以,绝大部分需要在服务器运行期反复执行业务逻辑的代码,必须使用异步代码,否则,同步代码在执行时期,服务器将停止响应,因为JavaScript只有一个执行线程。
(2)异步代码:服务器启动时如果需要读取配置文件,或者结束时需要写入到状态文件时,可以使用同步代码,因为这些代码只在启动和结束时执行一次,不影响服务器正常运行时的异步执行。
2、Stream模块
(1)流的概念:流的特点是数据是有序的,而且必须依次读取,或者依次写入,不能像Array那样随机定位。比如我们的标准输入流(stdin)和标准输出流(stdout)。不过用的最多的,应该还是文件的读取和写入流
(2)在Node.js中,流也是一个对象,我们只需要响应流的事件就可以了:data事件表示流的数据已经可以读取了,end事件表示这个流已经到末尾了,没有数据可以读取了,error事件表示出错了
var fs = require('fs');
// 打开一个流:
var rs = fs.createReadStream('sample.txt', 'utf-8');
rs.on('data', function (chunk) {
console.log('DATA:')
console.log(chunk);
});
rs.on('end', function () {
console.log('END');
});
rs.on('error', function (err) {
console.log('ERROR: ' + err);
});
(3)写入流
var fs = require('fs');
var ws1 = fs.createWriteStream('output1.txt', 'utf-8');
ws1.write('使用Stream写入文本数据...\n');
ws1.write('END.');
ws1.end();
var ws2 = fs.createWriteStream('output2.txt');
ws2.write(new Buffer('使用Stream写入二进制数据...\n', 'utf-8'));
ws2.write(new Buffer('END.', 'utf-8'));
ws2.end();
(4)pipe:就像可以把两个水管串成一个更长的水管一样,两个流也可以串起来。一个Readable流和一个Writable流串起来后,所有的数据自动从Readable流进入Writable流,这种操作叫pipe
var fs = require('fs');
var rs = fs.createReadStream('sample.txt');
var ws = fs.createWriteStream('copied.txt');
rs.pipe(ws);
3、http模块:
(1)要开发HTTP服务器程序,从头处理TCP连接,解析HTTP是不现实的。这些工作实际上已经由Node.js自带的http模块完成了。应用程序并不直接和HTTP协议打交道,而是操作http模块提供的request和response对象。
(2)用http模块实现一个最简单的Web程序
// 导入http模块:
var http = require('http');
// 创建http server,并传入回调函数:
var server = http.createServer(function (request, response) {
// 回调函数接收request和response对象,
// 获得HTTP请求的method和url:
console.log(request.method + ': ' + request.url);
// 将HTTP响应200写入response, 同时设置Content-Type: text/html:
response.writeHead(200, {'Content-Type': 'text/html'});
// 将HTTP响应的HTML内容写入response:
response.end('
Hello world!
');
});
// 让服务器监听8080端口:
server.listen(8080);
(3)实现文件服务器:url模块解析路径,path模块构造路径
var
fs = require('fs'),
url = require('url'),
path = require('path'),
http = require('http');
// 从命令行参数获取root目录,默认是当前目录:
var root = path.resolve(process.argv[2] || '.');
console.log('Static root dir: ' + root);
// 创建服务器:
var server = http.createServer(function (request, response) {
// 获得URL的path,类似 '/css/bootstrap.css':
var pathname = url.parse(request.url).pathname;
// 获得对应的本地文件路径,类似 '/srv/www/css/bootstrap.css':
var filepath = path.join(root, pathname);
// 获取文件状态:
fs.stat(filepath, function (err, stats) {
if (!err && stats.isFile()) {
// 没有出错并且文件存在:
console.log('200 ' + request.url);
// 发送200响应:
response.writeHead(200);
// 将文件流导向response:
fs.createReadStream(filepath).pipe(response);
} else {
// 出错了或者文件不存在:
console.log('404 ' + request.url);
// 发送404响应:
response.writeHead(404);
response.end('404 Not Found');
}
});
});
server.listen(8080);
4、crypto模块的目的是为了提供通用的加密和哈希算法,这部分先不管