一,基础。

1,Node的包管理器:npm; 安装node环境时会自动安装。
本地模式获取一个包:npm install [package_name]
此时包被安装到当前木的node_modules子目录下。
全局模式获取一个包;npm install -g [package_name]
全局模式安装的包不能直接通过require使用,但通过npm link命令可以在当前目录创建一个指向全局包的链接。比如,如果已经通过 npm install -g express 安装了 express, 这时在工程的目录下运行命令:  
npm link express  
./node_modules/express -> /usr/local/lib/node_modules/express 
 
2,第一个程序。
创建helloworld.js文件,并写入:console.log("Hello World");
打开cmd,进入hellodworld.js所在目录,运行:node hellowrold.js
 
3,命令行工具node。
查看帮助信息:node --help
REPL模式:即时求值的环境,类似于浏览器的控制台,直接输入node。
 
4,建立Http服务器。
运行以下代码,然后浏览器打开 localhost:3000
//app.js
var http = require('http');
http.createServer(function(req, res) {
res.writeHead(200, {'Content-Type': 'text/html'});
res.write('<h1>Node.js</h1>');
res.end('<p>Hello World</p>');
}).listen(3000);
console.log("HTTP server is listening at port 3000.");

如果运行app.js后,再修改app.js并不会实时反映到浏览器端,必须退出重新运行。

如果想实时看到修改后的效果,可以采用supervisor。
使用方法:
    安装supervisor:  npm install -g supervisor
    使用supervisor运行app.js: supervisor app.js
 
5,读取文件。
5.1异步读取文件:
//readfile.js
var fs = require('fs');
fs.readFile('file.txt', 'utf-8', function(err, data) {
if (err) {
console.error(err);
} else {
console.log(data);
}
});
console.log('end.');

可以看到,是通过回调函数实现异步。调用时只是将I/O请求发送给了操作系统,然后继续执行后面的代码,执行完成后进入事件循环监听事件。当fs接受到I/O请求完成的事件时,事件循环会主动调用回调函数。

 
5.2同步读取文件:
//readfilesync.js
var fs = require('fs');
var data = fs.readFileSync('file.txt', 'utf-8');
console.log(data);
console.log('end.');

6,事件。

//event.js
var EventEmitter = require('events').EventEmitter;
var event = new EventEmitter(); event.on('some_event', function() {
console.log('some_event occured.');
}); setTimeout(function() {
event.emit('some_event');
}, 1000);
7,node.js的事件循环机制。
node.js程序本身就是一堆事件的回调函数。程序的入口是事件循环第一个事件的回调函数。事件的回调函数在执行的过程中,可能会有异步请求或直接emit事件,执行完毕后再返回事件循环,事件循环会检查事件队列中的未处理事件,直到程序结束。
 
8,模块和包(Module 、Package)。
8.1创建并使用模块。
//module.js
var name;
exports.setName = function(thyName) {
name = thyName;
};
exports.sayHello = function() {
console.log('Hello ' + name);
};

//getmodule.js

var myModule = require('./module');
myModule.setName('BYVoid');
myModule.sayHello();
注意:模块是单例的,即无论多少次require,获得的都是同一个实例。
 
8.2把一个对象封装到模块中
 
//hello.js
function Hello() {
var name;
this.setName = function(thyName) {
name = thyName;
};
this.sayHello = function() {
console.log('Hello ' + name);
};
};
module.exports = Hello;

//gethello.js

var Hello = require('./hello');
hello = new Hello();
hello.setName('BYVoid');
hello.sayHello();

原理:exports本身仅仅是一个普通的空对象,即{}。

 
8.3包
包就是C#中的类库。
node中一个文件夹加上接口(index.js)再加上一个配置文件就是一个包。可以通过 npm init 命令根据提示一步一步创建一个标准的package.json配置。

二、核心模块。

1,全局对象
node中的全局对象是global,所有的全局变量都是global的属性,包括console、process等
全局变量process描述当前node的进程状态,process.nextTick(callback)的功能是为时间循环设置一项任务,node会在下次事件循环时调用callback.
console.log() 
console.error() 
console.trace()
 
2, util模块
util.inherits(constructor, superConstructor) 实现对象间的原型继承,但构造函数内部创造的属性和函数都不会被继承
var util = require('util');
function Base() {
this.name = 'base';
this.base = 1991;
this.sayHello = function() {
console.log('Hello ' + this.name);
};
} Base.prototype.showName = function() {
console.log(this.name);
};
function Sub() {
this.name = 'sub';
}
util.inherits(Sub, Base);
var objBase = new Base();
objBase.showName();
objBase.sayHello();
console.log(objBase); var objSub = new Sub();
objSub.showName();
//objSub.sayHello();
console.log(objSub);

util.inspect(object,[showHidden],[depth],[colors])是一个将任意对象转换 为字符串的方法, 通常用于调试和错误输出。

util还提供了util.isArray()、 util.isRegExp()、 util.isDate()、util.isError() 四个类型测试工具,以及 util.format()、util.debug() 等工具
 
3,events模块。
监听和触发事件:
var events = require('events');
var emitter = new events.EventEmitter();
emitter.on('someEvent', function(arg1, arg2) {
console.log('listener1', arg1, arg2);
});
emitter.on('someEvent', function(arg1, arg2) {
console.log('listener2', arg1, arg2);
});
emitter.emit('someEvent', 'byvoid', 1991);

EventEmitter提供的函数:

EventEmitter.on(event, listener)
EventEmitter.emit(event, [arg1], [arg2], [...])
EventEmitter.once(event, listener)
EventEmitter.removeListener(event, listener)
EventEmitter.removeAllListeners([event]) 
 
4,文件操作 - fs模块
fs.readFile(filename,[encoding],[callback(err,data)])
fs.readFileSync(filename, [encoding])
fs.open(path, flags, [mode], [callback(err, fd)])
fs.read(fd, buffer, offset, length, position, [callback(err, bytesRead,
buffer)])
 
5,http模块
 
//app.js
var http = require('http');
http.createServer(function(req, res) {
res.writeHead(200, {'Content-Type': 'text/html'});
res.write('<h1>Node.js</h1>');
res.end('<p>Hello World</p>');
}).listen(3000);
console.log("HTTP server is listening at port 3000.");

http客户端:

http.request(options, callback) 发起 HTTP 请求,以下是发送post请求的代码:
var http = require('http');
var querystring = require('querystring');
var contents = querystring.stringify({
name: 'byvoid',
email: 'byvoid@byvoid.com',
address: 'Zijing 2#, Tsinghua University',
});
var options = {
host: 'www.byvoid.com',
path: '/application/node/post.php',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
'Content-Length' : contents.length
}
};
var req = http.request(options, function(res) {
res.setEncoding('utf8');
res.on('data', function (data) {
console.log(data);
});
});
req.write(contents);
req.end();

http.get(options, callback)更加简便的方法用于处 理GET请求

var http = require('http');
http.get({host: 'www.byvoid.com'}, function(res) {
res.setEncoding('utf8');
res.on('data', function (data) {
console.log(data);
});
});
http.request 或 http.get创建的对象是http.ClientRequest,请求完成返回的对象是http.ClientRequest

三、web开发

1,使用Express框架。
类似于asp.net ,express实现的功能:路由控制、模版解析支持、动态视图、缓存等功能。
express默认支持的模版引擎:jade、ejs
 
安装express: > npm install -g express
安装express命令行工具: >npm install -g express-generator
查看帮助信息: > express --help
建立web基本结构(使用jade模版引擎):> express  myWeb
建立web基本结构(使用ejs模版引擎): > express -e myWeb2
进入刚建立的目录并初始化:>cd myWeb
                                            :> npm install
启动:> npm start
此时浏览http://localhost:3000 可以看到第一个界面了。
关闭服务器 ctrl+c
 
创建的代码中,app.js是工程的入口。
 
2,路由控制。
app.js中,引用路由:var routes = require('./routes/index');
把Url“/”映射到路由:app.use('/', routes);
 
routes/index.js中路由的写法:
    var express = require('express');
    var router = express.Router();
    /* GET home page. */
    router.get('/', function(req, res) {
      res.render('index', { title: 'Express' });
    });
    module.exports = router;
 
路由匹配:
/user/[username]  :
app.get('/user/:username', function(req, res) {
  res.send('user: ' + req.params.username);
}); 
 
3,视图。
布局、局部视图跟asp.net mvc一样。
<ul><%- partial('listitem', items) %></ul> 
 
视图助手(asp.net mvc中的helper 方法)
var util = require('util');
app.helpers({
inspect: function(obj) {
return util.inspect(obj, true);
}
});
app.dynamicHelpers({
headers: function(req, res) {
return req.headers;
}
});
app.get('/helper', function(req, res) {
res.render('helper', {
title: 'Helpers'
});
});
<%=inspect(headers)%>
 
4,数据库访问(以MongoDB为例)。
4.1 MongoDB。
MongoDB以文档的形式存储数据,一下是一个文档的示例:
{ "_id" : ObjectId( "4f7fe8432b4a1077a7c551e8" ),
  "uid" : 2004,
  "username" : "byvoid",
  "net9" : { "nickname" : "BYVoid",
    "surname" : "Kuo",
    "givenname" : "Carbo",
    "fullname" : "Carbo Kuo",
    "emails" : [ "byvoid@byvoid.com", "byvoid.kcp@gmail.com" ],
    "website" : "http://www.byvoid.com",
    "address" : "Zijing 2#, Tsinghua University" }
 
4.2 连接数据库。
 
安装MongoDB.
 
package.json中添加依赖代码:
"dependencies": {
    "express": "~4.2.0",
    "static-favicon": "~1.0.0",
    "morgan": "~1.0.0",
    "cookie-parser": "~1.0.1",
    "body-parser": "~1.0.0",
    "debug": "~0.7.4",
    "jade": "~1.3.0",
    "mongodb":">=0.9.9"
  }
 
运行:  > cd myWeb
             > npm install
 
工程目录中创建settings.js文件,代码如下:
module.exports = {
  cookieSecret: 'microblogbyvoid',
  db: 'microblog',
  host: 'localhost',
};
注:db是数据库名称,host是数据库的地址,cookieSecret用于Cookie加密与数据库无关。
 
创建models子目录,并在其中添加db.js文件,内容是:
var settings = require('../settings');
var Db = require('mongodb').Db;
var Connection = require('mongodb').Connection;
var Server = require('mongodb').Server;
module.exports = new Db(settings.db, new Server(settings.host, Connection.DEFAULT_PORT, {}));

注:通过module.exports输出了创建的数据库链接。

 
使用:
var mongodb = require('./db');

function User(user) {
this.name = user.name;
this.password = user.password;
};
module.exports = User; User.prototype.save = function save(callback) {
// 存入 Mongodb 的文档
var user = {
name: this.name,
password: this.password,
};
mongodb.open(function(err, db) {
if (err) {
return callback(err);
}
// 读取 users 集合
db.collection('users', function(err, collection) {
if (err) {
mongodb.close();
return callback(err);
}
// 为 name 属性添加索引
collection.ensureIndex('name', {unique: true});
// 写入 user 文档
collection.insert(user, {safe: true}, function(err, user) {
mongodb.close();
callback(err, user);
});
});
});
}; User.get = function get(username, callback) {
mongodb.open(function(err, db) {
if (err) {
return callback(err);
}
// 读取 users 集合
db.collection('users', function(err, collection) {
if (err) {
mongodb.close();
return callback(err);
}
// 查找 name 属性为 username 的文档
collection.findOne({name: username}, function(err, doc) {
mongodb.close();
if (doc) {
// 封装文档为 User 对象
var user = new User(doc);
callback(err, user);
} else {
callback(err, null);
}
});
});
});
};

四,其他。

1,模块加载机制。
核心模块已经被编译成二进制代码,可以直接require获取,如 require('fs');
require文件模块的时候,如果指明了路径则按指定的加载,如果没有指定,则去node_modules目录加载。
即使多次require同一个模块, 加载到的仍是同一个对象,因为node根据实际文件名把加载过的文件模块缓存了。
 
2,日志功能。
启用日志功能,需要以产品模式运行express : > NODE-ENV=production node app.js
记录访问日志和错误日志:
var fs = require('fs');
var accessLogfile = fs.createWriteStream('access.log', {flags: 'a'});
var errorLogfile = fs.createWriteStream('error.log', {flags: 'a'});
至于错误日志,需要单独实现错误响应,修改如下:
app.configure('production', function(){
app.error(function (err, req, res, next) {
var meta = '[' + new Date() + '] ' + req.url + '\n';
errorLogfile.write(meta + err.stack + '\n');
next();
});
});
 
3,javascript的作用域。
JavaScript的作用域是由函数来决定的,if、for语句中的花括号不是独立的作用域。
if (true) {
  var somevar = 'value';
}
console.log(somevar); // 输出 value 
 
在一个函数中定义的变量只对这个函数内部可见。
 
4,call 和 apply 的用法。
var someuser = {
  name: 'byvoid',
  display: function(words) {
    console.log(this.name + ' says ' + words);
  }
};
 
var foo = {
  name: 'foobar'
};
 
someuser.display.call(foo, 'hello'); // 输出 foobar says hello 
 
5,bind的用法。
var someuser = {
name: 'byvoid',
func: function() {
console.log(this.name);
}
}; var foo = {
name: 'foobar'
}; foo.func = someuser.func;
foo.func(); // 输出 foobar foo.func1 = someuser.func.bind(someuser);
foo.func1(); // 输出 byvoid func = someuser.func.bind(foo);
func(); // 输出 foobar func2 = func;
func2(); // 输出 foobar bind绑定参数列表:
var person = {
name: 'byvoid',
says: function(act, obj) {
console.log(this.name + ' ' + act + ' ' + obj);
}
}; person.says('loves', 'diovyb'); // 输出 byvoid loves diovyb byvoidLoves = person.says.bind(person, 'loves');
byvoidLoves('you'); // 输出 byvoid loves you

6, === 和 == 的区别。

==包含隐式转换。
 
7,原型、原型链。
prototype 、 _proto_
Object 、Function
Object.prototype 是所有对象的祖先,Function.prototype 是所有函数的原型,包括构造函数。
 
参考:《Node.js从入门到精通》

鸟瞰Nodejs的更多相关文章

  1. 一文读懂NodeJS全栈开发利器:CabloyJS(万字长文)

    目录 0 修订 0.1 修订说明 0.2 修订历史 1 基本概念 1.1 CabloyJS是什么 1.2 CabloyJS核心解决什么问题 1.3 CabloyJS的开发历程 2 数据版本与开发流程 ...

  2. 浏览器事件循环 & nodejs事件循环

    第1篇:如何理解EventLoop——宏任务和微任务篇 宏任务(MacroTask)引入 在 JS 中,大部分的任务都是在主线程上执行,常见的任务有: 渲染事件 用户交互事件 js脚本执行 网络请求. ...

  3. NodeJs之OS

    OS Node.js提供了一些基本的底层操作系统的模块OS. API var os = require('os'); console.log('[arch] 操作系统CPU架构'+os.arch()) ...

  4. NodeJs之Path

    Path模块 NodeJs提供的Path模块,使得我们可以对文件路径进行简单的操作. API var path = require('path'); var path_str = '\\Users\\ ...

  5. NodeJs之调试

    关于调试 当我们只专注于前端的时候,我们习惯性F12,这会给我们带来安全与舒心的感觉. 但是当我们使用NodeJs来开发后台的时候,我想噩梦来了. 但是也别泰国担心,NodeJs的调试是很不方便!这是 ...

  6. NodeJs在Linux下使用的各种问题

    环境:ubuntu16.04 ubuntu中安装NodeJs 通过apt-get命令安装后发现只能使用nodejs,而没有node命令 如果想避免这种情况请看下面连接的这种安装方式: 拓展见:Linu ...

  7. NodeJs之child_process

    一.child_process child_process是NodeJs的重要模块.帮助我们创建多进程任务,更好的利用了计算机的多核性能. 当然也支持线程间的通信. 二.child_process的几 ...

  8. nodejs进阶(6)—连接MySQL数据库

    1. 建库连库 连接MySQL数据库需要安装支持 npm install mysql 我们需要提前安装按mysql sever端 建一个数据库mydb1 mysql> CREATE DATABA ...

  9. 图片访问实时处理的实现(nodejs和php)

    我在访问时光网.网易云音乐等网站时,发现将它们页面中的一些图片URL修改一下就可以得到不同尺寸的图片,于是思考了其实现方案,我的思路是:URL Rewrite + 实时处理 + 缓存,对用户请求的UR ...

随机推荐

  1. Egret 摇一摇功能

    一  什么是重力感应 二 什么是陀螺仪 三 Egret中的摇一摇是利用什么原理实现的 四 摇一摇代码 一 什么是重力感应 二 什么是陀螺仪 三 Egret中的摇一摇是利用什么原理实现的 大概是利用手机 ...

  2. 跨域文件 clientaccesspolicy.xml

    <?xml version="1.0" encoding="utf-8" ?> <access-policy> <cross-do ...

  3. 【转】awk、nawk、mawk、gawk的简答介绍

    来自http://blog.sina.com.cn/s/blog_3d2d79aa0100h47h.html awk 是一种编程语言,用于在linux/unix下对文本和数据进行处理.数据可以来自标准 ...

  4. java length size

    1 java中的length属性是针对数组说的,比如说你声明了一个数组,想知道这个数组的长度则用到了length这个属性. 2 java中的length()方法是针对字符串String说的,如果想看这 ...

  5. 手把手教你编写一个具有基本功能的shell(已开源)

    刚接触Linux时,对shell总有种神秘感:在对shell的工作原理有所了解之后,便尝试着动手写一个shell.下面是一个从最简单的情况开始,一步步完成一个模拟的shell(我命名之为wshell) ...

  6. VC++ CEdit

    CEDIT _1, //selection pEdit1->SetSel(0,strBuffer - m_strInput,0); pEdit1->SetFocus(); //the se ...

  7. [转]df命令

    linux中df命令参数功能:检查文件系统的磁盘空间占用情况.可以利用该命令来获取硬盘被占用了多少空间,目前还剩下多少空间等信息. 语法:df [选项] 该命令各个选项的含义如下: -a 显示所有文件 ...

  8. 新手入门Java需要注意的问题

    学习编程,虽然有老师教,但是更重要的事自学.这是很重要的. 现在互联网上面资源太多了,这也就有一个问题:怎么才能在一定时间内学习该知识,掌握该技能呢? 理论联系实践! 学以致用!! 网上的资源太多了, ...

  9. Java中XML格式的字符串4读取方式的简单比较

    Java中XML格式的字符串4读取方式的简单比较 1.java自带的DOM解析. import java.io.StringReader; import javax.xml.parsers.Docum ...

  10. Caffe(卷积神经网络框架)介绍

    Caffe(卷积神经网络框架)Caffe,全称Convolution Architecture For Feature Extraction caffe是一个清晰,可读性高,快速的深度学习框架.作者是 ...