NodeJS用递归实现异步操作的链式调用,完成一个简易的命令行输入输出REPL交互接口

REPL —— Read-Eval-Print-Loop.

00.一门好的编程语言的必要条件

REPL并不是什么高大上的东西,简单的说就是一个从命令行程序,读取终端输入,处理,打印结果,如此循环。这是一门比较全面的编程语言的基础。

刚开始接触NodeJS,以为就是一个服务端Js,但学习了一段时间之后才感受到它的强大和魅力。如果说Java是编程游戏里一个喜欢循规蹈矩的古板选手,JavaScript就是一个天马行空的飘逸选手。刚开始转型异步编程非常不习惯,慢慢的懂得,异步编程,是一种思维。

01.依葫芦画不出瓢

一个命令行读取输入,处理的程序,Java可以写一个while循环来不停的用Scanner来readLine,但是如果模仿Java用JS就有点麻烦了,

var rl = require(‘readline‘);
while(true) {
    rl.on(‘line‘,function(data){
        //to deal data
    });
}

这样很显然是不行滴,不过习惯了同步编程的猿们说不定会这样干,JavaScript的异步意味着非阻塞式的设计,不会因为等用户输入阻塞下一次循环。所以要告诉JavaScript在这一次处理完之后,再进行下一次读取(回调),那么问题来了,臭名昭著的Callback hell就有可能发生了。

10.如何正确画瓢,还要看起来舒服

解决方案1:使用async,co,q, promise等等机制来同步

解决方案2:对于一个REPL程序,逻辑很简单,如果实现一个链式调用的接口就好了,像这样:

read("var1", function(data){}).read("var2",function(data){})...

写原生代码比使用开源库更加容易调试,而且杀鸡焉用牛刀,不麻烦co这样的模块了,q 这种then()的链式调用差强人意,想做什么就直接 .xx().xx()更直观。

11.银弹

用起来简单的东西,背后一定是用更复杂或更精巧的方式掩盖了其真实的复杂度。

repl.js

var readline = require(‘readline‘);    //非常好用的Node自带库

var rl = null;                                  //Singleton
var questions = [];                         //处理队列
var cursor = 0;                               //游标

function dealQuestions(_self) {       //递归处理队列中的读取请求
    if(cursor >= questions.length) {
        cursor = 0;questions = [];
        rl.close();
        rl = null;
        return _self;
    }

    rl.question(questions[cursor].question, function (data) {
        questions[cursor].deal(data);
        ++cursor;
        dealQuestions();                      //读取处理完毕回调
    });
}

var easyREPL = function(){
    if(rl == null) {                             //初始化模块
        rl = readline.createInterface({
            input: process.stdin,
            output: process.stdout
        });
    }
    return this;
};

/**
 * @description : a liner api to get console input
 * @param : question to ask , function to deal answer
 * */
easyREPL.prototype.question = function(question,dealAnswer){
    easyREPL();
    rl.prompt();
    var _self = this;                            //保存this指针
    questions.push({question:question,deal:dealAnswer});  //加入处理队列
    if(questions.length == 1) {
        dealQuestions(_self);
    }
    return this;                                   //实现链式调用的关键:返回this
};

exports.REPL_Mode = easyREPL;

测试代码:test.js

var easyREPL = require("./repl").REPL_Mode;

var test = new easyREPL();

/* comment following codes to test command mode */
test.question("Input name:" , function(data) {
    console.log("Get name:"+data);
}).question("Input password:",function(data){
    console.log("Get password:"+data);
}).question("Input Email:",function(data){
    console.log("Get email:"+data);
});

输出:

这是一个引子,NodeJS能做的事情太多了,另外还做了一个解析输入命令的程序,代码比这个多,但是不够精髓,还是不放了。

时间: 2024-10-11 06:25:48

NodeJS用递归实现异步操作的链式调用,完成一个简易的命令行输入输出REPL交互接口的相关文章

面向对象的链式调用

1. 对象的链式调用 function Chain(){ this.n=0;//属性不一定一开始的时候全部都要初始化 this.fn1=function(_obj){//this指向  new Chain()实例化的对象 alert(this.n++);//注意:alert(this.n++)与this.fn1中的this 不一定指向的对象是一样的 return this; } this.fn2=function(){//同上 alert(this.n++);//注意:alert(this.n+

进击的雨燕-------------可空链式调用

详情转自:http://wiki.jikexueyuan.com/project/swift/chapter2/07_Closures.html 可空链式调用(Optional Chaining)是一种可以请求和调用属性.方法及下标的过程,它的可空性体现于请求或调用的目标当前可能为空(nil).如果可空的目标有值,那么调用就会成功:如果选择的目标为空(nil),那么这种调用将返回空(nil).多个连续的调用可以被链接在一起形成一个调用链,如果其中任何一个节点为空(nil)将导致整个链调用失败.

在python中实现链式调用

用过jquery的一般都知道在jquery中可以链式调用,代码简洁优雅.比如$("a").addClass("test").show().html("foo");. 在redis-py中的pipeline中也可以链式调用,比如pipe.set('foo', 'bar').sadd('faz', 'baz').incr('auto_number').execute() . 那么究竟怎么实现的呢? 很简单,返回对象自己就行了,即return self

关于JavaScript中的setTimeout()链式调用和setInterval()探索

http://www.cnblogs.com/Wenwang/archive/2012/01/06/2314283.html http://www.cnblogs.com/yangjunhua/archive/2012/04/12/2444106.html 下面的参考:http://evantre.iteye.com/blog/1718777 1.选题缘起 在知乎上瞎逛的时候看到一个自问自答的问题: 知乎上,前端开发领域有哪些值得推荐的问答?,然后在有哪些经典的 Web 前端或者 JavaScr

JavaScript设计模式-8.链式调用

1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta charset="UTF-8"> 5 <title>javascript高级语法8-链式调用</title> 6 </head> 7 <body> 8 <div id="box"></div> 9 <script type="text/javas

链式调用

链式调用 不论是JavaScript还是C#程序,我们已经习惯了采用如下所示的"链式调用"的方式进行编程,这样确实会使我们的程序变得很精练. 1: new Foo() 2: .UseXxx() 3: .UseZzz() 4: .UseZzz() 5: .Invoke(); 采用这种链式调用方式的很多方法都是扩展方法,比如上面所示的UseXxx.UseYyy和UseXxx方法就是采用如下形式定义的扩展方法. 1: public interface IFoo 2: { 3: void In

javascript设计模式 第6章 链式调用

链式调用是一种语法招数.作用:能让你通过重用一个初始化操作来达到用少量代码表达复杂操作的目的.. 这种技术包含两个部分: 1.一个创建代表html元素的对象的工厂.以及一批对这个html元素执行某些操作的方法. 通过例子对比:之前和之后的代码,对链式调用的概念的初步认识.  之前:   addevent($('example'),"click",function(){    setstyle(this,"color",'green');    show(this);

如何学习Swift可空链式调用

今天我们一起来学习Swift可空链式调用.可空链式调用是一种可以请求和调用属性.方法以及下标的过程,它的可空体现在请求或调用的目标当前可能为nil.如果可空的目标有值,即调用就会成功:如果选择的目标为nil,即调用将返回nil.多个连续的调用可以被链接在一起形成一个调用链,如果其中任何一个节点为nil将导致整个链调用失败.    一.使用可空链式调用来强制展开    可空链的定义,即是在要调用非空的属性.方法.下标的可空值后面添加一个问号即可.特别的,可空链式调用的返回结果与原本的返回结果具有相

js简单实现链式调用

链式调用实现原理:对象中的方法执行后返回对象自身即可以实现链式操作.说白了就是每一次调用方法返回的是同一个对象才可以链式调用. js简单实现链式调用demo Object.prototype.show = function() {   console.log('show');   return this;  }  Object.prototype.hide = function() {   console.log('hide');   return this;  }  var div = doc