开云足球直播-GEN绝杀IG,Bin制霸全场

  用Node.js写生成器是件非常简单的事儿,原因是

Node.js模块开发简单,js语法,而且二进制cli模块也极其简单

Npm发布包是所有开源的包管理器里最简单好用的

辅助模块多,将近40万个左右

  所以为了让大家零基础十分钟搞定生成器,这里精简一下,你只需要5步

初始化模块

cli二进制模块

模板引擎使用

解析cli参数和路径

npm发布

  这里假定你已经安装了Node.js,至于是什么版本,如何安装的并不重要。

  先要介绍一下,什么是Npm?

  https://www.npmjs.com/

  npm is the package manager for

browsers

java

nodejs

io.js

mobile

bower

docpad

test

  简单理解:NPM(node package manager),通常称为node包管理器。顾名思义,它的主要功能就是管理node包,包括:安装、卸载、更新、查看、搜索、发布等。只要安装了Node.js,它会默认安装的。

  它可不只是Node.js package manager,可见其定位是很广的,这从侧面也佐证了大前端和node全栈的机会。以前nodejs吹牛都是那异步说事儿,现在都是拿生态说事儿,这话不错,在09年谈异步,很多语言性能都很弱,但事情要以发展的眼光看,现在很多语言都支持了,而且性能还不错,所以才显得nodejs性能没那么突出。

  无论做哪方面工作都可以使用npm,所以使用Node.js来开发各种模块都是非常方便的。

  1)初始化模块

  确认模块名称

$ npm info xxxxxx

  如果没有找到对应的包,说明你可以使用这个名字,然后在github建立仓库,clone到本地即可

$ git clone xxx$ npm init

-

y

  生成package.json文件,此文件为模块的描述文件,非常重要。

{

GEN绝杀IG,Bin制霸全场

"name"

:

"a"

,

"version"

:

"1.0.0"

,

"deion"

:

""

,

"main"

:

"index.js"

,

"s"

:

{

"test"

:

"echo "Error: no test specified" && exit 1"

},

"keywords"

:

[],

"author"

:

""

,

"license"

:

"ISC"

  }

  说明

main是模块的入口文件,即普通代码对外提供调用的api入口。

s是npm s非常便利的,只要在package.json所在目录下,你执行npm test就会调用这里的test配置。如果是test,start等内置命令之外的,可以通过npm run xxx来定义

2)cli二进制模块

  Node.js分2种模块

普通模块,供代码调用

二进制模块,提供cli调用

  大家都知道,生成器是cli工具,所以我们应该使用cli二进制模块

  手动修改package.json文件

{

"name"

:

"a"

,

"version"

:

"1.0.0"

,

"deion"

:

""

,

"main"

:

"index.js"

,

"bin"

:

{

"gen"

:

"gen.js"

},

"s"

:

{

"test"

:

"echo "Error: no test specified" && exit 1"

},

"keywords"

:

[],

"author"

:

""

,

"license"

:

"ISC"

  }

  这里主要增加里一个bin的配置,bin里的gen为cli的具体命令,它的具体执行的文件gen.js,大家看到这是一个plain old object类型,所以可以配置多个命令的,各位可以按照自己的喜好来。

  既然gen的执行文件是gen.js,我们当然需要创建创建它

$ touch gen

.

js

  填写

#!/usr/bin/env node

var

argv

=

process

.

argv

;

var

filePath

=

__dirname

;

var

currentPath

=

process

.

cwd

();

console

.

log

(

argv

)

console

.

log

(

filePath

)

console

.

log

(

currentPath

)

  说明

argv是命令行的参数

filePath是当前文件的路径,也就是以后安装后文件的路径,用于存放模板文件非常好

currentPath是当前shell上下文路径,也就是生成器要生成文件的目标位置

  至此,二进制模块的代码就写完了,下面我们测一下

  (1)本地安装此模块

  在package.json文件路径下,执行

$ npm link

  /Users/sang/.nvm/versions/node/v4.4.5/bin/gen ->/Users/sang/.nvm/versions/node/v4.4.5/lib/node_modules/a/gen.js/Users/sang/.nvm/versions/node/v4.4.5/lib/node_modules/a ->/Users/sang/workspace/github/i5ting/a

  此时说明已经安装成功了。

  (2)执行gen测试

$ gen

  ['/Users/sang/.nvm/versions/node/v4.4.5/bin/node','/Users/sang/.nvm/versions/node/v4.4.5/bin/gen']/Users/sang/workspace/github/i5ting/a/Users/sang/workspace/github/i5ting/a

  可以换不同的目录来测试一下,看看结果的不同,来体会上面3个变量的妙用。

  3)模板引擎使用

  模板引擎是一种复用思想,通过定义模板,用的时候和数据一起编译,生成html,以便浏览器渲染。从这个定义里我们可以找出几个关键点

  编译(模板 + 数据) => html

  模板引擎有好多种,下面介绍2种典型的模板引擎

ejs:嵌入js语法的模板引擎(e = embed),类似于jsp,asp,erb的,在html里嵌入模板特性,如果熟悉html写起来就非常简单,只要区分哪些地方是可变,哪些地方是不变即可

jade:缩进式极简写法的模板引擎,发展历史 HAML -> Jade -> Slim -> Slm,最早是ruby里有的,目前以jade用的最多,这种写法虽好,,但需要大脑去转换,这其实是比较麻烦的,如果对html不是特别熟悉,这种思维转换是非常难受的。

  更多见 https://github.com/tj/consolidate.js#supported-template-engines

  这里我们选一个,目前Node.js里最火的应该也是最好的Nunjucks,我感觉它和ejs比较像,但跟jade一样强大,语法据说出自Python的某款模板引擎

$ npm install

--

save nunjucks

  然后我们修改模板引擎

#!/usr/bin/env node

// var argv = process.argv;

// var filePath = __dirname;

// var currentPath = process.cwd();

//

// console.log(argv)

// console.log(filePath)

// console.log(currentPath)

var

nunjucks

=

require

(

'nunjucks'

)

var

compiledData

=

nunjucks

.

renderString

(

'Hello {{ username }}'

,

{

username

:

'James'

});

console

.

log

(

compiledData

)

  注释一下前面说的3个变量,这里我们只看nunjucks代码。这是最简单的demo。

  (1)引入nunjucks模块

  (2)nunjucks.renderString方法是编译模板用的,它有2个参数

第一个是模板字符串

第二个是json数据

  (3)compiledData就是编译后的结果

  结合上面说的模板引擎原理,

  编译(模板 + 数据) => html

  再理解一下,效果会更好。

  但是这样看来对我们没啥用啊,生成器的内容总不能都写到字符串里吧?所以继续改造,把模板独立出去,然后通过文件读写来获取模板字符串。

  创建一个gen.tpl,内容为Hello {{ username }},下面我们看看如何修改gen.js来读取模板。

var

fs

=

require

(

'fs'

)

  varnunjucks =require('nunjucks')

  vartpl =fs.readFileSync('./gen.tpl').toString()

  varcompiledData =nunjucks.renderString(tpl,{username:'James'});

  console.log(compiledData)

  (1)引入fs模块,因为要读取文件

  (2)fs.readFileSync(’./gen.tpl’).toString(),使用了一个读取文件的同步方法,并把文件内容转成字符串,原来是buffer

  读文件还是挺简单吧。那么写文件呢?

fs

.

writeFileSync

(

'./gen.xxx'

,

compiledData

)

  至此,一个生成器的模型就出来

#!/usr/bin/env node

var

fs

=

require

(

'fs'

)

var

nunjucks

=

require

(

'nunjucks'

)

var

tpl

=

fs

.

readFileSync

(

'./gen.tpl'

).

toString

()

var

compiledData

=

nunjucks

.

renderString

(

tpl

,

{

username

:

'James'

});

console

.

log

(

compiledData

)

fs

.

writeFileSync

(

'./gen.xxx'

,

compiledData

)

  思考一下,可变得有哪些?

'./gen.tpl’是输入模板

{ username: ‘James’ } 要编译的数据

'./gen.xxx’是最终的输出

  那么,剩下的事儿就是围绕可变得内容来构造你想要的功能。

  4)解析cli参数和路径

  要说生成器,最经典的是rails的scaffold,曾经缔造了一个15分钟blog的神话

$ rails g book name

:

string

coordinates

:

string

  如果我们要实现它,怎么做呢?

rails g是固定的用于生成的命令

book是模型名称,俗称表名

而name和coordinates都是字段名称,string是表中的类型

  可变的只有表名和字段信息。所以只要解析到这些就够了,换成我们的gen命令,大概是这样

$ gen book name

:

string

coordinates

:

string

  修改gen.js代码

#!/usr/bin/env node

var

argv

=

process

.

argv

;

console

.

log

(

argv

)

  执行gen命令的结果是

$ gen book name

:

string

coordinates

:

string

  ['/Users/sang/.nvm/versions/node/v4.4.5/bin/node','/Users/sang/.nvm/versions/node/v4.4.5/bin/gen','book','name:string','coordinates:string']

  下面构造一个entity对象

var

argv

=

process

.

argv

;

argv

.

shift

()

  argv.shift()console.log(argv)

  vardata ={model:argv[0],attr:{}

  }

  for(vari =1;i <argv.length;i++){vararr =argv[i].split(':')vark =arr[0];varv =arr[1];data.attr[k]=v

  }

  console.dir(data)

  执行

$ gen book name

:

string

coordinates

:

string

  ['book','name:string','coordinates:string']data ={model:'book',attr:{name:'string',coordinates:'string'}

  }

  那这里的data可以做什么呢?想想模板引擎里的第二个参数~

// tpl compile

  varcompiledData =nunjucks.renderString(tpl,data)

  修改模板gen.tpl

module

.

exports

=

class

{{

model

}}

{

{%

for

k

,

v

in

attr

%}

{{

k

}}:

{{

v

}},

{%

else

%}

error

  {%endfor %}

  }

  结果gen.xxx为

module

.

exports

=

class

book

{

name

:

string

,

coordinates

:

string

,

}

  这里是只是示意,具体当按照你想要的结果为准。

#!/usr/bin/env node

var

fs

=

require

(

'fs'

)

var

nunjucks

=

require

(

'nunjucks'

)

var

argv

=

process

.

argv

;

// var filePath = __dirname;

// var currentPath = process.cwd();

//

// console.log(filePath)

// console.log(currentPath)

// cli parse

argv

.

shift

()

argv

.

shift

()

console

.

log

(

argv

)

var

data

=

{

model

:

argv

[

],

attr

:{

}

}

for

(

var

i

=

1

;

i

<

argv

.

length

;

i

++)

{

var

arr

=

argv

[

i

].

split

(

':'

)

var

k

=

arr

[

];

var

v

=

arr

[

1

];

data

.

attr

[

k

]

=

v

}

console

.

log

(

'data = '

)

console

.

dir

(

data

)

// read tpl

var

tpl

=

fs

.

readFileSync

(

'./gen.tpl'

).

toString

()

console

.

dir

(

data

)

// tpl compile

var

compiledData

=

nunjucks

.

renderString

(

tpl

,

data

)

console

.

log

(

compiledData

)

// write file

fs

.

writeFileSync

(

'./gen.xxx'

,

compiledData

)

  下面修改一下路径

tpl从__dirname走

而结果需要写到process.cwd()

  也就是我们前面说的那2个没有用到的变量filePath和currentPath。

#!/usr/bin/env node

var

fs

=

require

(

'fs'

)

var

nunjucks

=

require

(

'nunjucks'

)

var

argv

=

process

.

argv

;

var

filePath

=

__dirname

;

var

currentPath

=

process

.

cwd

();

//

// console.log(filePath)

// console.log(currentPath)

// cli parse

argv

.

shift

()

argv

.

shift

()

console

.

log

(

argv

)

var

data

=

{

model

:

argv

[

],

attr

:{

}

}

for

(

var

i

=

1

;

i

<

argv

.

length

;

i

++)

{

var

arr

=

argv

[

i

].

split

(

':'

)

var

k

=

arr

[

];

var

v

=

arr

[

1

];

data

.

attr

[

k

]

=

v

}

console

.

log

(

'data = '

)

console

.

dir

(

data

)

// read tpl

var

tpl

GEN绝杀IG,Bin制霸全场

=

fs

.

readFileSync

(

filePath

+

'/gen.tpl'

).

toString

()

console

.

dir

(

data

)

// tpl compile

var

compiledData

=

nunjucks

.

renderString

(

tpl

,

data

)

console

.

log

(

compiledData

)

// write file

fs

.

writeFileSync

(

currentPath

+

'/gen.xxx'

,

compiledData

)

  至此,完成了所有代码。此时你在任意目录输入

$ gen book name

:

string

coordinates

:

string

  你会发现当前目录下会有一个gen.xxx文件,和我们之前看到的结果一样。

  5)npm发布

  在package.json目录里执行

$ npm publish

.

  就可以了发布成功了。

  如果你想增加版本号,再次发布,你需要2步

$ npm version patch$ npm publish

.

  你可以自己测试一下

$ npm i

-

g xxxxxx

  share给别人吧

  更多

异常:各种可能考虑到并处理

测试:按照各位喜好 mocha, ava, jest

工具模块:比如使用debug模块处理调试信息,日志等

argv解析模块:commander 或者yargs

实用工具,比如各种大小写转换,驼峰式等 inflected

最后

  生成器理论是可以生成一切内容的,那么生成能够生成器模板代码么?自己想想吧

相关阅读

  • 开云体育平台APP-基米希一锤定音!莱比锡红牛巅峰对决马里,关键先生改写战局

    开云体育平台APP-基米希一锤定音!莱比锡红牛巅峰对决马里,关键先生改写战局

    在足球世界的聚光灯下,有些比赛注定被载入史册,当德甲劲旅莱比锡红牛在欧冠赛场遭遇法甲黑马里尔(注:根据常见译名,“马里”通常指非洲国家,但结合足球语境,此处应为“里尔”Lille的误译或简称,以下按“里尔”展开)时,一场技术与战术的巅峰对决...

    2026.02.27 20:15:16作者:开云体育
  • 开云体育入口-字母哥节奏完全掌控,骑士绝杀步行者

    开云体育入口-字母哥节奏完全掌控,骑士绝杀步行者

    字母哥的绝对掌控与骑士的绝杀艺术 在篮球的世界里,节奏是看不见的指挥官,它决定了比赛的呼吸与心跳,当字母哥以希腊神话般的统治力完全掌控赛场节奏,而骑士在最后时刻以冷血绝杀终结步行者——这两幕看似无关的场景,却共同揭示了现代篮球的深层逻辑:...

    2026.02.26 11:55:08作者:开云体育
  • 开云APP-世界排名争夺战之夜,文班亚马用实力证明自己

    开云APP-世界排名争夺战之夜,文班亚马用实力证明自己

    苍穹之下,新王加冕:文班亚马在世界排名争夺战之夜的封神之路 引子:风暴前的宁静 巴黎贝尔西体育馆内,空气仿佛凝固,世界男子篮球排名争夺战进入白热化阶段——法国对阵斯洛文尼亚,这不仅是两支劲旅的较量,更是新生代天才维克多·文班亚马与传奇球...

    2026.02.25 11:40:05作者:开云体育
  • 开云体育app-金字塔尖的较量,加维抢七封神,埃及巅峰对决奥地利

    开云体育app-金字塔尖的较量,加维抢七封神,埃及巅峰对决奥地利

    当终场哨声响起,电子记分牌上的数字凝固在 埃及 3-2 奥地利,这不是一场普通的小组赛,也不是寻常的淘汰赛——这是“文明巅峰杯”季后赛的抢七决战,而决定这场跨越时空文明对决的,是一位名叫加维的年轻人。 文明的对决:当尼罗河遇见多瑙河 “...

    2026.02.24 19:35:04作者:开云体育
  • 开云体育APP下载-英超争冠白热化,国米强势晋级映衬希腊神话

    开云体育APP下载-英超争冠白热化,国米强势晋级映衬希腊神话

    扩展思维后拟定): 《双城记:曼城阿森纳决战天王山,蓝黑军团欧冠晋级唤醒希腊足球荣光》 本周欧洲足坛上演了两场截然不同却又彼此映衬的焦点战役——一边是英超争冠路上决定性的“天王山之战”,另一边是国际米兰在欧冠赛场强势晋级,其对手雅典AE...

    2026.02.24 11:30:11作者:开云体育
  • 开云官网-西甲国家德比之夜,特奥完成里程碑,红蓝与纯白间的永恒传奇

    开云官网-西甲国家德比之夜,特奥完成里程碑,红蓝与纯白间的永恒传奇

    国家德比之夜,特奥里程碑:红蓝与纯白间的永恒传奇 马德里的夜空被伯纳乌的灯光点亮,空气中弥漫着一种只有西班牙国家德比才能制造的特殊电荷,2023年10月28日,当皇家马德里与巴塞罗那再次相遇,这场超越足球的对抗不仅关乎积分榜的领先位置,更...

    2026.02.23 03:15:19作者:开云体育
  • 开云下载-米切尔弑旧主!狂轰41+8+5,森林狼主场力克爵士,他证明了自己是真正的狼王

    开云下载-米切尔弑旧主!狂轰41+8+5,森林狼主场力克爵士,他证明了自己是真正的狼王

    明尼苏达的夜晚,标靶中心球馆内人声鼎沸,当终场哨声响起,比分定格在森林狼118:112爵士,全场球迷起立欢呼,而焦点只有一个——身披森林狼10号战袍的多诺万·米切尔。 就在几个月前,米切尔还是犹他爵士的当家球星,而今晚,他面对旧主,用一场...

    2026.02.22 19:15:06作者:开云体育
  • 开云下载-字母哥闪耀全场,火箭速胜魔术,一场天赋与战术的完美碰撞

    开云下载-字母哥闪耀全场,火箭速胜魔术,一场天赋与战术的完美碰撞

    扩展思维): 《字母哥统治攻防两端,火箭闪电战击溃魔术》 《从密尔沃基到休斯顿:字母哥的焦点之夜与火箭的速胜哲学》 《天赋聚焦:字母哥全能秀与火箭团队篮球的胜利方程式》 《焦点之战:字母哥个人英雄主义与火箭集体闪电战的对比启示》...

    2026.02.21 19:00:25作者:开云体育

添加新评论