XDL_NO.5 Node.js 中的I/O交互

回顾下上节课的知识点

  • 搭建一个简单的 Node.js 服务器
* 利用到 HTTP 模块(要使用HTTP服务器或客户端功能,需引用此模块)
  • 利用Node.js 搭建一个静态网站
* 利用到 HTTP 模块、URL模块、fs 文件操作模块

今天的主题是:Node.js 中的I/O交互


那么I/O是什么?

  • I input,O output 输入输出,交互,你有输入那我有输出(反馈)
  • I/O 操作包括读写操作、输入输出、请求响应、数据库...
  • 那了解到这,我就更需要知道node.js 中 是异步 I/O的,详细说明

交互,带给人们更好的体验!

readline 模块 逐行读取

要使用此模块,需要require('readline').Readline程序允许逐行读取一个流内容(例如process.stdin).
需要注意的是你一旦调用了这个模块,你的node程序将不会终止直到你关闭此接口。下面是如何让你的程序正常退出的方法:

rl.close();

通过readline 模块,实现一个简单的终端交互的例子

var readline = require('readline');    // 引入 readline 模块,实例化一个test.js
// 创建一个 readline 的接口实例
// input要监听的可读流 process.stdin
// output要写入的readline 的可写流 process.stdout
var rl = readline.createInterface({input:process.stdin,output: process.stdout});
rl.setPrompt("请输入:");               // 设置提示符
rl.prompt();                          // 为用户输入准备好readline,将现有的setPrompt选项放到新的一行,让用户有一个新的地方开始输入。
rl.on('line',(line)=>{                // 监听 line 事件,触发闭包函数
  var str = line.trim();              // 对用户输入的字符串进行两端去除空格
  console.log("你输入的是:"+str);      // 控制台显示用户输入的信息
  rl.prompt();                        // 然后再产生新的一行,给用户输入
});  

效果:

I/O终端交互

那这样就产生了一个简单的交互,但这个你只要写就会一直输出,不关闭窗口或者使用 Ctrl+c ,永远也不会停止 ,所以这里我要使用 readline 提供的一个让程序正常退出:

 rl.close();

先来一个,监听退出程序给予一个提示(使用Ctrl+c退出程序):

var readline = require('readline');    // 引入 readline 模块,实例化一个test.js
// 创建一个 readline 的接口实例
// input要监听的可读流 process.stdin
// output要写入的readline 的可写流 process.stdout
var rl = readline.createInterface({input:process.stdin,output: process.stdout});
rl.setPrompt("请输入:");               // 设置提示符
rl.prompt();                          // 为用户输入准备好readline,将现有的setPrompt选项放到新的一行,让用户有一个新的地方开始输入。
rl.on('line',(line)=>{                // 监听 line 事件,触发闭包函数
  var str = line.trim();              // 对用户输入的字符串进行两端去除空格
  console.log("你输入的是:"+str);      // 控制台显示用户输入的信息
  rl.prompt();                        // 然后再产生新的一行,给用户输入
}).on('close',()=>{                   // 监听退出程序时 ,调用闭包函数
  console.log("好好学习,天天向上!Coding、Coding!");
  // rl.close();
  process.exit();                     // 列队行进中 退出,好比我正在排队打饭,有事,退出队列
});

上面这个例子的退出是不是感觉并不友好,所以下面我们来个更加有好一点的:

终端交互

那接下来,我们来玩一个竞猜游戏,使用到readline 中的,rl.question(query, callback)
预先提示指定的query(询问),然后用户应答后触发指定的callback。 显示指定的query给用户后,当用户的应答被输入后,就触发了指定的callback

var readline = require("readline");                 // 引入 readline 模块
var rl       = readline.createInterface({           // 创建一个readline 接口
  input : process.stdin,                            // 要监听的可读流
  output: process.stdout                            // 要写人line的可写流u
});
rl.setPrompt("欢迎参加答题!请输入OK进入答题,输入NO退出答题!");      // 设置提示符
rl.prompt();                                                       // 为用户输入准备好readline,将现有的setPrompt选项放到新的一行,让用户有一个新的地方开始输入。
rl.on('line',(line)=>{                                             // 绑定事件,监听 line ,看用户输入什么
    var str = line.trim().toLowerCase();
    if(str == 'ok'){
      rl.question("你爱不爱我?",(answer)=>{
        var answer = answer.trim();
        if(answer == '爱'){
          console.log('你最好了!');
          rl.close();
        }else{
          console.log('你讨厌,不和你玩了!');
          rl.close();
        }
      });
    }else if(str == "no"){
      console.log("谢谢你的使用!");
      rl.close();
    }else {
      console.log("您输入的有误,请重新输入!");
      rl.prompt();
    }
});
竞猜

嘻嘻,自娱自乐中学习了 rl.question();

引入自定义模块

  exports.sk = function(exstr){
       console.log('这是一个上课模块'+ exstr);
  }
   
  文件:text.js 引用该模块文件
  var sk = require('./exports');
自定义模块
自定义模块

Query String

这个模块提供一些处理 query string(查询 字符串) 的工具。

querystring.stringify(obj, [sep], [eq])

序列化一个对象到一个 query string。可以选择是否覆盖默认的分割符('&')和分配符('=')。

 querystring.stringify({foo: 'bar', baz: 'qux'}, ';', ':')
  // 返回如下字串'foo:bar;baz:qux'
  // ; 作为分割符    : 分配符(替换分割)

querystring.parse(str, [sep], [eq], [options])

将一个 query string 反序列化为一个对象。可以选择是否覆盖默认的分割符('&')和分配符('=')。
options对象可能包含maxKeys属性(默认为1000),它可以用来限制处理过的键(key)的数量.设为0可以去除键(key)的数量限制.

querystring.parse('foo=bar&baz=qux&baz=quux&corge')
// returns{ foo: 'bar', baz: ['qux', 'quux'], corge: '' }

querystring.escape

供 querystring.stringify 使用的转意函数,在必要的时候可被重写。

querystring.unescape

供 querystring.parse 使用的反转意函数,在必要的时候可被重写。

知道了这个 模块对字符串的操作和处理,在结合前几天的所学,来做个小实验吧!Web 版的 简易计算器(node.js)

      // 引入http、url、fs、querystring 模块
var http = require("http");
var url  = require("url");
var fs   = require("fs");
var qs   = require("querystring");

// 监听的 端口、ip
var ip   = '192.168.1.9';
var port = 8080;
//创建一个新的WEB服务器对象
http.createServer(onRequest).listen(port,ip);
// 接收请求 的处理函数
function onRequest(req,res){
// 获取请求的url,地址
var pathname = url.parse(req.url).pathname; //parse() 把字符串转成json
// 读取文件
if(pathname == '/'){
  // 返回资源文件类型
  res.writeHead(200,{'Content-Type':'text/html;charset="UTF-8"'});
  // 读取文件内容,返回给客户端
  fs.readFile('./jsq.html',(err,content)=>{
    res.write(content);
    res.end();
  });
}else if(pathname == '/js'){   // 表单提交
    // 定义两个变量 接收数据 data 变量  和 计算结果值 result
    var info   = '';
    var result = '';
    // 添加一哥监听 事件, 监听 data 提交过来的数据 ;这里的addListener和on一样的
    req.addListener('data',function(postData){
      info += postData;        // 将提交过来的数据一一给予a 连接在一起
      var _info = qs.parse(info);  // 将info 字符串转换成json对象
      var num1  = parseInt(_info["num1"]); // 将想要的数据提取出来,转成整型
      var ysf   = parseInt(_info["ysf"]);
      var num2  = parseInt(_info["num2"]);
      console.log("参数1:"+num1);
      console.log("运算操作序号:"+ysf);
      console.log("参数2:"+num2);
      // 然后根据 运算符 的值  ,来做相应的运算操作
      switch (ysf) {
        case 1:
            result = num1+num2;
          break;
        case 2:
            result = num1-num2;
          break;
        case 3:
            result = num1*num2;
          break;
        case 4:
            result = num1/num2;
          break;
      }

      // 监听请求结束事件 将运算的结果返回给客户端
      req.on('end',function(){ // 证明下 on 和 addListener 无差别
        res.writeHead(200,{'Content-Type':'text/html;charset="UTF-8"'});
        fs.readFile("jsq.html",function(err,data){
          res.write(data);
          res.write("结果是:"+result);        // 返回运算结果,添加到页面当中去
          res.end();
        });
      });
    });
}else{
  res.writeHead(404,{'Content-Type':'text/html;charset="UTF-8"'});
  res.write("请求失败!");
  res.end();
}
}
简易计算器效果

这个是初略的显示下效果,简单的先完善功能!最后达到的效果如下:

最终版计算器
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容