系列目录:Node.js摸石头系列目录

一、一个错误引发的摸索

上回我们在获取 request 对象的 headers 属性的 'user-agent’  属性时,我使用了 request.headers.user-agent 这样的语法,谢谢网友artwl的提醒,这样写经实验是不行的。可是,为什么不行呢?这件事让我迷惑了。Js 中对象可以理解为属性的集合,属性的构成是键值对,Function 并不搞特殊化,和其他类型一视同仁。属性的值可以通过 object.key 或者 object[‘key’] 的方式来访问。问题出在哪里呢?上网一顿猛摸,无果。后来观察观察 headers 对象:

headers:   { host: 'localhost:888',connection: 'keep-alive','cache-control': 'max-age=0','user-agent': 'Mozilla/5.0 (Windows NT 6.1) AppleWebKit/535.1 (KHTML, like Gecko) Chrome/14.0.835.202 Safari/535.1 CoolNovoChromePlus/1.6.4.30',accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8','accept-encoding': 'gzip,deflate,sdch','accept-language': 'zh-CN,zh;q=0.8','accept-charset': 'GBK,utf-8;q=0.7,*;q=0.3' }

发现所有加了引号的键全部都有 '-' 啊,万恶的横杠啊,原来如此。

提醒:键名称如果有 '-' 、空格等貌似非法的字符,Js 是很宽容的,不过要求你用引号括起来。而访问时也无法再用圆点加键名的方式来获取值,需要用方括号加引号的方式。

两句话可以说清楚的事,罗嗦了半天。其实想说的是,小石头让咱卡壳是常态,摸他,排除他。

另外,如果您希望摸一摸某个对象,可以在 repl 环境下,把这个对象打印出来。比如,前面我们引入的 http 模块,可以 repl 提示符下键入:

D:> node

> require(‘http’)

您会得到下图:

你看,状态代码都不用查文档了吧?

二、读写文件

为了完成搭建静态服务器的任务,我们需要文件 I/O ,node.js 内置了 fs 模块,引入就可以读写文件了。请按下列方式组织目录:

MyHttpServer

|_____  app.js

|_____  webroot

|_____  index.htm

这里 webroot 文件夹就是我们静态页面的家了,我们希望以后把页面往里面一丢,就能从客户端访问。index.htm 文件里你随便贴几行字进去好了。一会我们要把他们从文件里读出来,显示在控制台并发送给浏览器。

在写代码之前,我们先用前面的方法查看 fs 模块:

fs 里方法真多啊。现在到了我们查阅文档的时候了。去官网文档,查到 fs.readFile 了吗?OK,就用他。测试该方法的代码就不单独写了,建议您自己先写一个,小步快跑,是学习的好方法,在编程上也适用,这样可以避免遇到问题时难以定位问题范围,可以不断地给自己小小地鼓励。直接上代码:

/* Read a file and put to user agent */
var http = require('http'),fs = require('fs');http.createServer(function(request, response){//读文件,读完后写入response,文件名系硬编码var path = __dirname + '/webroot/index.htm';fs.readFile( path,'utf-8', function(err,data) { //读出的内容在data里//在 console 打印//console.log(path);console.log(data);response.end(data);});
}).listen(888);
console.log('Server start at port 888.');

上面代码有个全局属性 __dirname ,说明下,就是当前运行代码所处的目录。查下文档赫然白底黑字地写着。嗯?怎么还有这么行字:__dirname isn't actually a global but rather local to each module. 乖乖个隆地洞,差点弄错了,原来这玩意是相对于每个模块的。另外,请注意,是两个下划线哦。_ _ d i r n a m e 。

三、分析请求路径

上节我们实现了读取文本文件,并送到客户端浏览器的工作。但是读取的文件名是硬编码在代码里的。现在,我们需要让用户来指定他需要什么文件,然后读出来给发过去。而用户指定的路径和文件名这个信息,正如我们前面所说的,是通过 request 传过来的,你一定还记得在系列三里,我们曾经将 request 在服务端后台打印出来过,而 request 的众多的属性里,有一个为 url 属性。是的,通常我们都是通过 url 来映射文件的路径的。不过,到了现在 MVC 和 REST 时代,情况开始变得有些复杂,暂且不提。下面我们要慢慢加快速度了。我会把一些解释逐渐移到代码的注释里面。所以,请看代码。呃,看代码之前,一条 url 一般可以分成主机地址、路径和键值对们,这个事你懂的。呃,有位园友希望讲细一点,好吧,如果你不觉得啰嗦的话,请做下图的试验:

我们尝试引入url模块,用这个工具来解析了一串示例,得到一个对象。其中 pathname 就是我们要的,不过现在我们需要将它映射为我们服务器端的绝对地址。好了,来试试:

/* Map the url path to serverpath */var http = require('http'),fs = require('fs'),urlutil = require('url'); //node.js 推荐变量名与模块同名,这里为了防止误解,暂时改下http.createServer(function(request, response){var path = urlutil.parse(request.url).pathname;console.log(path);//映射本地var absPath = __dirname + "/webroot" + path;console.log("Path at Server: " + absPath);
}).listen(888);
console.log('Server start in port 888.');

好任务完成。现在我们要把文件发给浏览器了。

var http = require('http'),fs = require('fs'),urlutil = require('url'),  path = require('path');http.createServer(function(request, response){//get path from request's urlvar urlpath = urlutil.parse(request.url).pathname;//map the path to server pathvar absPath = __dirname + "/webroot" + urlpath;//test whether the file is exists firstpath.exists(absPath, function(exists) {if(exists) {//if okfs.readFile(absPath,function(err, data) {//our work is hereif(err) throw err;console.log(data);response.write(data);response.end();});} else {//show 404response.end('404 File not found.');}});
}).listen(888);
console.log('Server start in port 888.');

嗯,代码很完美的实现了我们的任务。当然,还有点小问题是需要我们改进的。不过先休息下,找点形容词来赞美自己吧,对自己不要太吝啬了,反正也没人听见,是不是?

四、MIME

上面的代码还有一点不足,就是仅仅能够读出文本文件,而且控制台的提示也是乱乱的看不清楚。一个一个来。

首先把控制台的事搞定。只需要在读文件的时候指定编码就可以了,比如:readFile(absPath,’utf-8’,function(…  就可以了。

剩下的就是读写不同格式文件的问题了。为了告诉浏览器我们发给他的是什么类型的文件,需要给 response 写个头信息,然后发给浏览器,浏览器根据这个信息来确定发来的是什么类型的内容。这个信息仍然是个键值对,键是 Content-Type ,顾名思义,就是内容类型了。值是什么呢?大家知道,因为服务器和浏览器都是不同的开发者开发的,所以这个事需要沟通好,否则你说文件类型是文本,他理解成图片,那不是麻烦了?

而避免这个麻烦的东东就是MIME了。关于MIME请参考这里,不多说什么了,我们需要做的就是把这页的列表弄下来,放进自己的代码。为了清晰起见,丢进一个模块吧,模块名 mime 好了。很自然的,我们想到用一个对象来表示这个列表。

exports.types = {
'323':'text/h323',
acx:'application/internet-property-stream',
ai:'application/postscript',
aif:'audio/x-aiff',
aifc:'audio/x-aiff',
aiff:'audio/x-aiff',
asf:'video/x-ms-asf',
asr:'video/x-ms-asf',
asx:'video/x-ms-asf',
au:'audio/basic',
avi:'video/x-msvideo',
axs:'application/olescript',
bas:'text/plain',
bcpio:'application/x-bcpio',
bin:'application/octet-stream',
bmp:'image/bmp',
c:'text/plain',
cat:'application/vnd.ms-pkiseccat',
cdf:'application/x-cdf',
cer:'application/x-x509-ca-cert',
'class':'application/octet-stream',
clp:'application/x-msclip',
cmx:'image/x-cmx',
cod:'image/cis-cod',
cpio:'application/x-cpio',
crd:'application/x-mscardfile',
crl:'application/pkix-crl',
crt:'application/x-x509-ca-cert',
csh:'application/x-csh',
css:'text/css',
dcr:'application/x-director',
der:'application/x-x509-ca-cert',
dir:'application/x-director',
dll:'application/x-msdownload',
dms:'application/octet-stream',
doc:'application/msword',
dot:'application/msword',
dvi:'application/x-dvi',
dxr:'application/x-director',
eps:'application/postscript',
etx:'text/x-setext',
evy:'application/envoy',
exe:'application/octet-stream',
fif:'application/fractals',
flr:'x-world/x-vrml',
gif:'image/gif',
gtar:'application/x-gtar',
gz:'application/x-gzip',
h:'text/plain',
hdf:'application/x-hdf',
hlp:'application/winhlp',
hqx:'application/mac-binhex40',
hta:'application/hta',
htc:'text/x-component',
htm:'text/html',
html:'text/html',
htt:'text/webviewhtml',
ico:'image/x-icon',
ief:'image/ief',
iii:'application/x-iphone',
ins:'application/x-internet-signup',
isp:'application/x-internet-signup',
jfif:'image/pipeg',
jpe:'image/jpeg',
jpeg:'image/jpeg',
jpg:'image/jpeg',
js:'application/x-javascript',
latex:'application/x-latex',
lha:'application/octet-stream',
lsf:'video/x-la-asf',
lsx:'video/x-la-asf',
lzh:'application/octet-stream',
m13:'application/x-msmediaview',
m14:'application/x-msmediaview',
m3u:'audio/x-mpegurl',
man:'application/x-troff-man',
mdb:'application/x-msaccess',
me:'application/x-troff-me',
mht:'message/rfc822',
mhtml:'message/rfc822',
mid:'audio/mid',
mny:'application/x-msmoney',
mov:'video/quicktime',
movie:'video/x-sgi-movie',
mp2:'video/mpeg',
mp3:'audio/mpeg',
mpa:'video/mpeg',
mpe:'video/mpeg',
mpeg:'video/mpeg',
mpg:'video/mpeg',
mpp:'application/vnd.ms-project',
mpv2:'video/mpeg',
ms:'application/x-troff-ms',
mvb:'application/x-msmediaview',
nws:'message/rfc822',
oda:'application/oda',
p10:'application/pkcs10',
p12:'application/x-pkcs12',
p7b:'application/x-pkcs7-certificates',
p7c:'application/x-pkcs7-mime',
p7m:'application/x-pkcs7-mime',
p7r:'application/x-pkcs7-certreqresp',
p7s:'application/x-pkcs7-signature',
pbm:'image/x-portable-bitmap',
pdf:'application/pdf',
pfx:'application/x-pkcs12',
pgm:'image/x-portable-graymap',
pko:'application/ynd.ms-pkipko',
pma:'application/x-perfmon',
pmc:'application/x-perfmon',
pml:'application/x-perfmon',
pmr:'application/x-perfmon',
pmw:'application/x-perfmon',
pnm:'image/x-portable-anymap',
pot:'application/vnd.ms-powerpoint',
ppm:'image/x-portable-pixmap',
pps:'application/vnd.ms-powerpoint',
ppt:'application/vnd.ms-powerpoint',
prf:'application/pics-rules',
ps:'application/postscript',
pub:'application/x-mspublisher',
qt:'video/quicktime',
ra:'audio/x-pn-realaudio',
ram:'audio/x-pn-realaudio',
ras:'image/x-cmu-raster',
rgb:'image/x-rgb',
rmi:'audio/mid',
roff:'application/x-troff',
rtf:'application/rtf',
rtx:'text/richtext',
scd:'application/x-msschedule',
sct:'text/scriptlet',
setpay:'application/set-payment-initiation',
setreg:'application/set-registration-initiation',
sh:'application/x-sh',
shar:'application/x-shar',
sit:'application/x-stuffit',
snd:'audio/basic',
spc:'application/x-pkcs7-certificates',
spl:'application/futuresplash',
src:'application/x-wais-source',
sst:'application/vnd.ms-pkicertstore',
stl:'application/vnd.ms-pkistl',
stm:'text/html',
svg:'image/svg+xml',
sv4cpio:'application/x-sv4cpio',
sv4crc:'application/x-sv4crc',
swf:'application/x-shockwave-flash',
t:'application/x-troff',
tar:'application/x-tar',
tcl:'application/x-tcl',
tex:'application/x-tex',
texi:'application/x-texinfo',
texinfo:'application/x-texinfo',
tgz:'application/x-compressed',
tif:'image/tiff',
tiff:'image/tiff',
tr:'application/x-troff',
trm:'application/x-msterminal',
tsv:'text/tab-separated-values',
txt:'text/plain',
uls:'text/iuls',
ustar:'application/x-ustar',
vcf:'text/x-vcard',
vrml:'x-world/x-vrml',
wav:'audio/x-wav',
wcm:'application/vnd.ms-works',
wdb:'application/vnd.ms-works',
wks:'application/vnd.ms-works',
wmf:'application/x-msmetafile',
wps:'application/vnd.ms-works',
wri:'application/x-mswrite',
wrl:'x-world/x-vrml',
wrz:'x-world/x-vrml',
xaf:'x-world/x-vrml',
xbm:'image/x-xbitmap',
xla:'application/vnd.ms-excel',
xlc:'application/vnd.ms-excel',
xlm:'application/vnd.ms-excel',
xls:'application/vnd.ms-excel',
xlt:'application/vnd.ms-excel',
xlw:'application/vnd.ms-excel',
xof:'x-world/x-vrml',
xpm:'image/x-xpixmap',
xwd:'image/x-xwindowdump',
z:'application/x-compress',
zip:'application/zip'
}

类型比较多,所以很长,但结构很简单。注意我们在模块里,可以把需要暴露出去的东东链到 exports 下就可以了。把这个文件存为 mime.js ,后面我们就可以用

var mime = require(‘./mime’)

这样的语法来访问了。

万事具备,只欠东风了,离胜利只有一步之遥了。

五、完成

完成的代码:

/* Final Server */var http = require('http'),fs = require('fs'),urlutil = require('url'),  path = require('path'),mime = require('./mime');http.createServer(function(request, response){//get path from request's urlvar urlpath = urlutil.parse(request.url).pathname;//map the path to server pathvar absPath = __dirname + "/webroot" + urlpath;//test whether the file is exists firstpath.exists(absPath, function(exists) {if(exists) {//二进制方式读取文件fs.readFile(absPath,'binary',function(err, data) {//our work is hereif(err) throw err;//获取合适的 MIME 类型并写入 response 头信息var ext = path.extname(urlpath);ext = ext ? ext.slice(1) : 'unknown';console.log(ext);var contentType = mime.types[ext] || "text/plain";console.log(contentType);response.writeHead(200,{'Content-Type':contentType});//console.log(data);//使用二进制模式写response.write(data,'binary');response.end();});} else {//show 404response.end('404 File not found.');}});
}).listen(888);
console.log('Server start in port 888.');

猛然发现我们居然实现了Apache、IIS 的基本功能。好了,可以洗洗睡了。大家伙晚安。

差点忘了,下一回我们会进入聊天室的任务。

转载于:https://www.cnblogs.com/hsxixi/archive/2011/12/23/2298507.html

完成静态服务器——Node.js摸石头系列之四相关推荐

  1. Node.js摸石头系列目录

    1.Node.js 的安装和控制台命令--Node.js摸石头系列之一 2.架一个HTTP服务--Node.js摸石头系列之二 3.完成HelloWorld--Node.js摸石头系列之三 4.完成静 ...

  2. Node.js爬虫一站到底系列先导篇

    前言: 在web编程课上,老师布置了爬虫任务,而没有任何经验和相关方面知识的小白简直一头雾水,不知道该如何下手.一开始抱着一本厚厚的犀牛书啃了好几天,本以为对Javascipt语法有一定了解后便可以自 ...

  3. 开源 静态 文件 服务器,Node.js搭建静态服务器

    写在开头,本文是node.js最最初级的搭建静态服务器,比较适合新手入门,大神请绕道哦- Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境. Node.js 使用 ...

  4. 什么是node网站服务器,node.js

    Node.js发布于2009年5月,由Ryan Dahl开发,是一个基于Chrome V8引擎的JavaScript运行环境,使用了一个事件驱动.非阻塞式I/O模型,[1] 让JavaScript 运 ...

  5. nodejs 本地php服务器,node.js创建本地服务器详解

    本文主要和大家分享node.js创建本地服务器详解,简易上手node.js后,我们就可以在自己电脑上创建本地服务器了.希望能帮助到大家. 一.先上代码.//请求Node.js自带的http模块. va ...

  6. 【华为云技术分享】如何快速实现鲲鹏弹性云服务器Node.js部署和高可用性?

    "Node 开发者容易面临的前三大困惑分别是异步编程.事件驱动以及 Debug.同时,文档是大家最期待的资源,新人对视频教程和免费在线课程的呼声最高."这份<2020 年 N ...

  7. 单服务器node.js和php性能测试

    选手介绍: node.js,服务端javascript语言,以出色的事件驱动和I/O异步广受关注,它更像一辆性能出色的Mitsubishi Evlution X: php,耳熟能详的服务端语言,在互联 ...

  8. 突袭HTML5之WebSocket入门2 - 高效服务器Node.js

    JavaScript脱离浏览器的限制以后,隐隐有再次走向光明的迹象,看来还得重新审视一下它的前景.由于条件有限,这里所有的例子是运行在Windows平台下的.而且由于目前并没有太好的组织JS程序的ID ...

  9. node 邮箱服务器,Node.js 搭建邮件服务器

    Node.js 搭建邮件服务器 servervar smtp = require('smtp-protocol'); var server = smtp.createServer(function ( ...

最新文章

  1. 金山发布《2006年度信息安全报告》
  2. C# socket编程实践——支持广播的简单socket服务器
  3. enum2str做为queryValue时的问题
  4. 测试与CMMI质量体系
  5. linux 往文件写4k大小,[svc]为何linux ext4文件系统目录默认大小是4k?
  6. 微型计算机的电池,具有微型计算机芯片的电池蓄电模块、便携式计算机的制作方法...
  7. python dataframe 取一行_python – Pandas dataframe获取每个组的第一行
  8. java基于Springboot+vue的鲜花销售商城网站
  9. 【51单片机快速入门指南】5.2:SPI读取 12位ADC XPT2046 芯片
  10. 域名解析TTL是什么意思 TTL值设置为多少合适?
  11. 1 Pandas实例(一)-2012美国大选政治献金实例
  12. 目前最值得入手的蓝牙耳机有哪些?四款高性价比蓝牙耳机推荐
  13. Amy Palladino 加入 BCW,担任企业运作执行副总裁兼董事总经理
  14. 教你使用python在终端创建炫酷二维码!!!
  15. [CISCN2019 华东南赛区]Web4
  16. Proxmox VE 桌面虚拟化(windows 10)集群尝试
  17. 出现“未报告的异常错误,必须对其进行捕获或声明以便抛出”的解决
  18. GAME(A)性能测试过程模型
  19. windows下的host文件在哪里,有什么作用?
  20. matlab 小波的分解与重构

热门文章

  1. Android 闹钟app 课程设计
  2. 明日之后android和ios,明日之后互通区有哪些 明日之后ios和安卓互通区一览
  3. oracle dblink 验证,oracle通过dblink查询sqlserver报错
  4. 【苹果相册推】群发安装软件设备推送通知SSL允许证
  5. MTK_android11_WIFI(内网)和4G(外网)共存
  6. 车载测试常见关心问题解答
  7. 精彩的“利益均衡”,尤其是“四”
  8. MySql8.0windows无法启动服务解决方案can‘t creat test file
  9. 爬虫漫游指南:浏览器指纹
  10. Could not get unknown property ‘packageForR‘ for task ‘:app:processDebugResources‘ of type com.andro