Node.js诞生于2009年,Node.js采用C++语言编写而成,是一个JavaScript的运行环境,Node.js 是一个基于Chrome V8引擎的JavaScript的运行环境,让JavaScript的运行脱离浏览器端,可以使用JavaScript语言编写服务端代码.
安装nodejs
如果确定node已经安装,但未显示的话,可能出现是问题,node在mac中需要配置环境变量:
- touch ~/.bash_profile
- vim ~/.bash_profile
- export PATH = $PATH:/usr/local/bin
- 退出vim命令: :x
- 重启电脑或者scource ~/.bash_profile
学习资料:
(1) 菜鸟教程: http://www.runoob.com/nodejs/nodejs-tutorial.html
(2)nodejs参考手册: http://nodejs.cn/api/
nodejs模块化开发
为什么会有模块:
在JavaScript发展初期就是为了实现简单是页面交互逻辑,寥寥数语即可;如今CPU、浏览器性能得到了极大的提升,很多页面逻辑迁移到客户端(表单验证等),随着web2.0时代的到来,Ajax技术得到广泛的应用,jQuery等前端库层出不穷,前端代码日益膨胀,这时候JavaScript作为嵌入式的脚本语言的定位动摇了,JavaScript却没有为组织代码提供任何明显帮助,甚至没有类的概念,更不用说模块(module)了,JavaScript极其简单的代码组织规范不足以驾驭如此强大规模的代码。
模块的好处:
既然JavaScript不能处理如此大规模的代码,我们可以借鉴一下其他语言是怎么处理大规模程序设计的,在Java中又一个重要的概念——package,逻辑上相关的代码组织到同一个包内,包内是一个相对独立的王国,不用担心命名冲突什么的,那么如果外部使用呢?直接import对应package即可:import java.until.Arraylist.
遗憾的是JavaScript在设计时定位原因,没有提供类似的功能,开发者需要模拟出类似的功能,来隔离、阻止复杂的JavaScript代码,我们成为模块化。
一个模块就是实现特定功能的文件。有了模块,我们就可以更方便的使用别人的代码,想要什么功能,就加载什么模块。模块开发需要遵循一定的规范,各行其是就都乱套了。
commonjs规范
CommonJS就是为JS的表现来制定规范,因为js没有模块的功能所以CommonJs应运而生,它希望js可以在任何地方运行不只是浏览器中。
CommonJs能有一定的影响力,我觉得绝对离不开Node的人气,不过,Node、CommonJs、浏览器甚至W3C之间有什么关系呢
require()
Node.js的模块仓库https://www.npmjs.com/,已经存放了15万个模块,其中绝大部分都是CommonJS格式.这种格式的核心就是require语句,模块通过它加载.
require()的基本用法
当Node遇到 require(X) 时,按下面的顺序处理
(1) 如果 X 是内置模块 (比如 require("http") )
a. 返回该模块
b. 不在继续执行
(2) 如果 X 以"./" 或者 "/" 或者 "../"开头
a. 根据 X 所在的符模块,确定 X 的绝对路径
b. 将 X 当成文件,依次查找下面文件,只要其中有一个存在,就返回该文件,不再继续执行.
X
X.js
X.json
X.node
c. 将 X 当成目录,依次查找下面文件,只要其中有一个存在,就返回该文件,不再继续执行
X/package.json (main字段)
X/index.js
X/index.json
X/index.node
(3) 如果 X 不带路径
a. 根据 X 所在的父模块,确定 X 可能的安装目录
b. 依次在每个目录中,将 X 当成文件名或目录名加载
(4) 抛出 “not found”
举例:
当前脚本文件 /home/ry/projects/foo.js, 执行了require("bar"). 这属于上面的第三种情况.Node内部的运行过程如下:
首先确定 x 的绝对路径可能是下面这些位置,依次搜索每一个目录.
/home/ry/projects/node_modules/bar
/home/ry/node_modules/bar
/home/node_modules/bar
/node_modules/bar
搜索时,Node先将 bar 当成文件名,依次尝试加载下面这些文件,只要有一个成功就返回
bar
bar.js
bar.json
bar.node
如果都不成功,说明bar 可能是目录名,于是依次尝试加载下面这些文件
bar/package.json (main字段)
bar/index.js
bar/index.json
bar/index.node
如果在所有目录中,都无法找到bar对应的文件或目录,就抛出一个错误.
案例:引入url模块解析路径
var http = require("http"); //引入一个http的模块
var url = require("url"); //引入路径模块
var server = http.createServer(function(req,res){
//parse(路径,第二个参数为将请求参数解析为对象)
var urlObj = url.parse(req.url,true);
console.log(urlObj);
res.end();
});
server.listen(8080);
创建自己的模块
加载当前目录的模块
不加var的变量是全局变量,在模块内部最好加var
要导出一个属性或者方法使用:modules.exports
如果是模块下的方法和属性使用:modules.exports={xx:xx},如果是想让莫亏啊直接接受方法就导出函数
加载非当前目录的模块
首先会看当前目录是否有一个node_modules,如果有就进入里面查找是否有叫模块名称的文件或者是否有叫模块名称的文件夹,如果是模块名称的文件夹需要看一下文件夹内部是否有叫index.js或者package.json中是否有设置main属性,可以通过指令npm init 来进行创建
{
"name":"home",
"version" :"0.0.1",
"main": "home2.js"
}
如果当前目录没有node_modules,会向上查找.
加载原生模块
非node项目也可以使用模块化开发技术,用的比较多的模块加载器有require(CMD),seaJs(AMD)
nodejs的内置模块
Buffer , C/C++Addons , Child Processes , Cluster , Console , Crypto , Debugger , DNS , Domian , Errors , Events , File System , Globals , HTTP , HTTPS , Modules , Net , OS , Path , Process , Punycode , Query Strings , Readline , REPL , Stream , String Decoder , Timers , TLS/SSL , TTY , UDP/Datagram , URL , Utilities , V8 , VM , ZLIB ;
内置模块不需要安装,外置模块需要安装