CLI的全称是Command-line Interface(命令行界面),即在命令行接受用户的键盘输入并作出响应和执行的程序。

在Node.js中,全局安装的包一般都具有命令行界面的功能,例如我们用于启动web服务的http-server包,在全局安装后,可按下图所示在当前目录启动一个web服务:

如果我们也想编写这样的命令行程序,该如何做呢?很简单,只需先创建下面这样的目录结构:

在bin目录中,我们创建了一个my-cli.js的文件,文件内容如下:

#!/usr/bin/env node

console.log('hello my-cli');

注意:第一行非常重要,标记这是一个node环境可执行文件。

而图中的package.json的内容如下所示:

{
  "author": "Mr.L",
  "license": "MIT",
  "name": "my-cli",
  "version": "1.0.0",
  "description": "my-cli",
  "bin": {
    "my-cli": "bin/my-cli.js"
  },
  "preferGlobal": "true"
}

在package.json文件中,我们指定了bin属性,其中的key就是命令行指令名称,映射到bin/my-cli.js这个可执行文件,最后的`preferGlobal`设置为`true`,表示可在全局安装并运行。

最后,在命令行进入图中的my-cli目录,执行如下命令:

如图所示,我们先执行`npm install -g .`命令,全局安装当前包,在安装完成后,直接在命令行执行`my-cli`,即可运行相关的程序。

到这里我们就完成了自己的一个简单CLI程序,不过通常来说,CLI程序需要一些实际的交互功能,并且最好能够支持两种运行方式:即命令行和作为普通包引入。

我们稍微修改一下上面的目录结构:

如图所示,我们新创建了一个lib目录,用于放置主要的业务逻辑,代码如下:

let chalk = require('chalk');

let sayHello = name => {
    // 使用chalk 设置控制台字体样式
    return chalk.bold('Hello, ') + chalk.bold.bgRed(name);
};

// 导出
exports.sayHello = sayHello;

我们定义了一个`sayHello`方法,根据传递的`name`参数,生成一个问候语,需要注意的是,我们在该文件中引入了`chalk`库,该库用于设置命令行的文字样式。

下面我们再来看看bin/my-cli.js文件的内容:

#!/usr/bin/env node

let readline = require('readline');
let lib = require('../lib/index.js');

// 创建交互界面
let interface = readline.createInterface({
    input: process.stdin,
    output: process.stdout
});

// 开始界面交互
interface.question('What is your name? ', answer => {
    let greeting = lib.sayHello(answer);

    // 打印结果
    console.log(greeting);

    interface.close();
});

在上面的代码中,我们引入Node.js内置的`realline`模块,该模块在标准输入和标准输出的基础上做了封装,在命令行中给出提示,并接受用户输入;另外,我们也引入了lib目录中的index.js,在接受用户输入后,调用`sayHello`方法,生成结果并打印。

最后,package.json也要相应做些修改:

{
  "author": "Mr.L",
  "license": "MIT",
  "name": "my-cli",
  "version": "1.0.0",
  "description": "my-cli",
  "bin": {
    "my-cli": "bin/my-cli.js"
  },
  "main": "lib/index.js",
  "preferGlobal": "true",
  "dependencies": {
    "chalk": "^1.1.3"
  }
}

我们添加了一个`main`属性,指向lib/index.js文件,这样便于其他开发者在程序中引入,调用我们的`sayHello`方法,我们稍后会介绍。

现在我们再执行`npm install -g .`命令,全局安装当前包,然后打开新的命令行:

如图所示,该程序接受了命令行参数,并正确打印出了结果。

前面提到,我们的CLI也要作为一个普通包被开发者引入,现在我们就来演示一下。首先,创建一个测试目录,如下图所示:

这个目录很简单,只有一个test.js文件和一个node_modules目录,在node_modules目录中包含我们的my-cli目录:

我们简单看一下test.js文件的内容:

let cli = require('my-cli');

let greeting = cli.sayHello('Jack');

console.log(greeting);

然后,我们在命令行直接执行该文件,打印结果如下图所示:

最后,我们也可以将自己的库发布到npm仓库,按照官方文档指示即可完成。

参考资料:

https://bretkikehara.wordpress.com/2013/05/02/nodejs-creating-your-first-global-module

https://nodejs.org/api/readline.html

Node.js系列文章:编写自己的命令行界面程序(CLI)的更多相关文章

  1. Node.js系列文章:如何进行代码调试

    使用任何一门编程语言,都少不了代码调试这一功能.我们在使用JavaScript编写浏览器端代码时,Chrome提供了强大的调试工具Dev Tools,但是在编写Node.js代码时,大多数人最开始都使 ...

  2. Node.js系列文章:利用console输出日志文件

    通常我们在写Node.js程序时,都习惯使用console.log打印日志信息,但这也仅限于控制台输出,有时候我们需要将信息输出到日志文件中,实际上利用console也可以达到这个目的的,今天就来简单 ...

  3. node.js系列笔记之node.js初识《一》

    node.js系列笔记之node.js初识<一> 一:环境说明 1.1 Linux系统CentOS 5.8 1.2 nodejs v0.10.15 1.3 nodejs源码下载地址 htt ...

  4. Node.js系列——(4)优势及场景

    背景 之前几篇系列文章简单介绍了node.js的安装配置及基本操作: Node.js系列--(1)安装配置与基本使用 Node.js系列--(2)发起get/post请求 Node.js系列--(3) ...

  5. Ember.js系列文章

    JS前端框架之Ember.js系列文章 本文为文章索引,主要是罗列Ember.js的相关文章便于阅读. 相关演示代码:github for free. 基础篇 1. EmberJs之What|Why| ...

  6. 微信JS图片上传与下载功能--微信JS系列文章(三)

    概述 在前面的文章微信JS初始化-- 微信JS系列文章(一)中已经介绍了微信JS初始化的相关工作,接下来本文继续就微信JS的图片上传功能进行描述,供大家参考. 图片上传 $(function(){ v ...

  7. 微信JS分享功能--微信JS系列文章(二)

    概述 在上一篇文章微信JS初始化-- 微信JS系列文章(一)中已经介绍了微信JS初始化的相关工作,接下来本文继续就微信JS的分享功能进行描述,供大家参考. 代码 $(document).ready(f ...

  8. Node.js系列-http

    前言: 最近一直忙着公司项目的事,战友们的留言也没空回复,博客也有段时间没有更新了,年底了就是一个的忙啊~~~(ps:同感的也给个赞吧) 现在前端的就是一直地更新一直有新的东西出来,什么ES2015, ...

  9. Node.js系列-express(上)

    前言 Node.js系列的第一篇:http,大概描述了通过使用node.js内置的api创建一个服务并监听request实现简单的增删改查.现在,我们就通过通读express官网及使用express框 ...

随机推荐

  1. 洛谷U19464 山村游历(Wander)(LCT,Splay)

    洛谷题目传送门 LCT维护子树信息常见套路详见我的总结 闲话 题目摘自WC模拟试题(by Philipsweng),原题目名Wander,"山村游历"是自己搞出来的中文名. 数据自 ...

  2. 【洛谷2744 】【CJOJ1804】[USACO5.3]量取牛奶Milk Measuring

    题面 Description 农夫约翰要量取 Q(1 <= Q <= 20,000)夸脱(夸脱,quarts,容积单位--译者注) 他的最好的牛奶,并把它装入一个大瓶子中卖出.消费者要多少 ...

  3. [BZOJ4292] [PA2015] Równanie

    Description 对于一个正整数n,定义f(n)为它十进制下每一位数字的平方的和.现在给定三个正整数k,a,b,请求出满足a<=n<=b且k*f(n)=n的n的个数. Input 第 ...

  4. RHEL7 网卡绑定

     //bond(主要用于6系统) nmcli connection add type bond mode balance-rr con-name bond0 ifname bond0 ipv4.met ...

  5. CentOS配置本地yum源

    如果CentOS服务器处在内网环境中时,如果缺少依赖手动安装那么会非常麻烦,要花费很多时间来寻找rpm包,现在如果搭建本地的yum源,就非常方便了,使用yum源首先需要一个CentOS安装镜像,去官网 ...

  6. 设计模式——状态模式(C++实现)

    /////////context.cpp #include "context.h" void STContext::ChangeState(STState* pstState) { ...

  7. 《Linux命令行与shell脚本编程大全》- 读书笔记3 - 理解shell

    当用户登录终端的时候,通常会启动一个默认的交互式shell.系统究竟启动哪个shell,这取决于用户配置.一般这个shell都是/bin/shell.默认的系统shell(/bin/sh)用于系统sh ...

  8. Java技术总结

    1.在非空判断是一定把not null 判断写前边,否则如果为空先判断size为0会报错 String str = null; if(str !=null&&str.length()& ...

  9. MYSQL数据库学习十七 日志管理

    17.1 MySQL软件所支持的日志 MySQL所支持的日志文件里,除了二进制日志文件外,其他日志文件都是文本文件.默认情况下,MySQL只会启动错误日志文件,其他日志文件则需要手动启动. 二进制日志 ...

  10. oracle session数激增排查过程

    我们的生产系统使用的是oracle 11G RAC,昨天突然收到微信告警通知session数达到450个,平时的session数在200个左右. select username,status,mach ...