一、一个错误引发的摸索

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

1
2
3
4
5
6
7
8
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,就用他。测试该方法的代码就不单独写了,建议您自己先写一个,小步快跑,是学习的好方法,在编程上也适用,这样可以避免遇到问题时难以定位问题范围,可以不断地给自己小小地鼓励。直接上代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/* 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 就是我们要的,不过现在我们需要将它映射为我们服务器端的绝对地址。好了,来试试:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
/* 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.');

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
var http = require('http'),
    fs = require('fs'),
    urlutil = require('url'),  
    path = require('path');
     
http.createServer(function(request, response){
    //get path from request's url
    var urlpath = urlutil.parse(request.url).pathname;
    //map the path to server path
    var absPath = __dirname + "/webroot" + urlpath;
    //test whether the file is exists first
    path.exists(absPath, function(exists) {
        if(exists) {
            //if ok
            fs.readFile(absPath,function(err, data) {
            //our work is here
            if(err) throw err;
            console.log(data);
            response.write(data);
            response.end();
    });
        } else {
            //show 404
            response.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 好了。很自然的,我们想到用一个对象来表示这个列表。

+ View Code

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

var mime = require(‘./mime’)

这样的语法来访问了。

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

五、完成

完成的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
/* 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 url
    var urlpath = urlutil.parse(request.url).pathname;
    //map the path to server path
    var absPath = __dirname + "/webroot" + urlpath;
     
    //test whether the file is exists first
    path.exists(absPath, function(exists) {
        if(exists) {
            //二进制方式读取文件
            fs.readFile(absPath,'binary',function(err, data) {
            //our work is here
            if(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 404
            response.end('404 File not found.');
        }
    });
}).listen(888);
console.log('Server start in port 888.');

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

摸石头——NOde.js(四)相关推荐

  1. 开场 Live,分享点干货——「深入了解 Node.js 包与模块机制」

    先放上 Live 地址: www.zhihu.com/lives/84274- 本次 Live 将深入剖析 Node.js 包与模块机制,包括且不限于解析 Node.js 源码.社区规范等.本人认为这 ...

  2. 【Node.js】初识Node.js

    系列文章目录 文章目录 系列文章目录 一.什么是 Node.js 二.下载和安装 Node.js 1.普通方式 2.使用 nvm 安装 三.Node.js 和 JavaScript 的区别 1.ECM ...

  3. Node.js (v19.1.0npm 8.19.3) vue.js安装配置教程(超详细)

    Node.js (v19.1.0npm 8.19.3) vue.js安装配置教程(超详细) 目录 一.安装Node(npm)需要的环境和版本发布信息 (1).Node版本和npm版本关系 (2).支持 ...

  4. 完成静态服务器——Node.js摸石头系列之四

    系列目录:Node.js摸石头系列目录 一.一个错误引发的摸索 上回我们在获取 request 对象的 headers 属性的 'user-agent'  属性时,我使用了 request.heade ...

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

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

  6. Hyperledger Fabric 1.0 实战开发系列 第四课 搭建node.js服务器

    接下来我要做的是用fabric sdk来做出应用程序,代替CLI与整个区块链网络交互.并且实现一个http API,向社区提供一个简单的接口,使社区轻松的与区块链交互. 官方虽然提供了Node.JS, ...

  7. Node.js「四」—— 路由 / EJS 模板引擎 / GET 和 POST

    本文为 Node.js 系列笔记第四篇.文章参考:nodejs 教程:<深入浅出 Node.js>:阮一峰 nodejs 博客: Node.js v16.13.0 文档 文章目录 一.路由 ...

  8. 详细记录基于vue+nodejs+mongodb构建的商城学习(四)基于项目的node.js开发后端的学习与梳理总结...

    前置: 本系列文章是一个本人边学习边梳理的学习笔记,俗话说好脑袋不如烂笔头,再好的记忆力时间长了也会有细节忘记,本项目选择的前端框架是vue,后端开发使用是node.js,数据库使用的是mongodb ...

  9. 四、Node.js - 数据库与身份认证

    文章目录 目标 一.数据库的基本概念 1.什么是数据库 2.常见的数据库及分类 3.传统型数据库的数据组织结构 (1)Excel 的数据组织结构 (2)传统型数据库的数据组织结构 (3)实际开发中库. ...

最新文章

  1. 2022-2028年中国AKD施胶剂行业市场研究及前瞻分析报告
  2. 使用元组输入进行计算和归约
  3. 如何找回丢失的Vista系统“休眠”菜单
  4. 海贼王热血航线正在连接服务器,《航海王热血航线》无法进入原因和解决方法 进不去如何解决...
  5. 前台传json ajax,如何在前台脚本通过json传递数据到后台(使用微软自带的ajax)
  6. 彻底解决_OBJC_CLASS_$_某文件名, referenced from:问题
  7. 使用.net资源文件时候发生的问题
  8. php swoole hyperf,【php】Hyperf为什么要关闭Swoole协程短名称
  9. 求生之路2正版服务器ip,求生之路2怎么看ip地址
  10. java.lang.SecurityException: Prohibited package name: java.xxx.xxxx
  11. DE25 Homogeneous Linear Systems with Constant Coefficients
  12. CSS3弹性盒模型flexbox布局基础版
  13. 软考初级-程序员了解
  14. 年度最流行英文字体20款
  15. 大一python选择题题库及答案_大学计算机python选择填空题库及答案
  16. 万兆交换机用什么网线_现在国内没有万兆宽带,那为什么有万兆网线呢?
  17. Minecraft 1.16.5模组开发(三十三) 自定义3D方块
  18. An unhandled exception occurred: listen EADDRNOTAVAIL: address not available
  19. 网络流24题(部分)
  20. excel怎么删除换行符

热门文章

  1. python menu_Python——Menu控件
  2. VMware Workstation 不可恢复错误: (vmx)Exception 0xc0000006 (disk error while paging) has occurred.
  3. 母版页已经有from 一页只能有一个服务器端 Form 标记
  4. VS中*.clw *.ncb *.opt *.aps这些文件是做什么用的?
  5. 哪款蓝牙耳机音质好?2022音质好的蓝牙耳机盘点
  6. erp是企业内部最重要的使用程序,它对日常运作至关重要
  7. OpenAI生成二次元美女【辣眼睛慎入】
  8. IOS6区别于IOS5的几个不明显的改变
  9. iMac恢复出厂设置及安装
  10. 给迷茫的计算机系大学生的一封信 JAVA