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
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
|
---
language: Nim
filename: learnNim-cn.nim
contributors:
- ["Jason J. Ayala P.", "http://JasonAyala.com"]
- ["Dennis Felsing", "https://dennis.felsing.org"]
translators:
- ["lzw-723", "https://github.com/lzw-723"]
lang: zh-cn
---
Nim(原名Nimrod)是一种静态类型的命令式编程语言,
它能在不影响运行时效率的情况下为程序员提供强大的功能。
Nim语言高效、有表现力、优雅。
```nim
# 单行注释以一个#开头
#[
这是多行注释
在Nim语言中,多行注释可以嵌套,以#[开头,以]#结尾
]#
discard """
这也可以作为多行注释使用。
或者用于无法解析、损坏的代码
"""
var # 声明(和赋值)变量
letter: char = 'n' # 带或不带类型批注
lang = "N" & "im"
nLength: int = len(lang)
boat: float
truth: bool = false
let # 使用let*一次性*声明和绑定变量。
legs = 400 # legs是不可改变的。
arms = 2_000 # _会被忽略,对long类型非常有用。
aboutPi = 3.15
const # 常量在编译时计算。这确保了
debug = true # 性能,在编译时表达式中很有用。
compileBadCode = false
when compileBadCode: # `when`是编译时的`if`
legs = legs + 1 # 这个错误永远不会被编译。
const input = readline(stdin) # const在编译时必须是已知的。
discard 1 > 2 # 注意:如果表达式的结果未使用,
# 编译器会发出警告。`discard`绕过了这一点。
#
# 数据结构
#
# 元组(Tuple)
var
child: tuple[name: string, age: int] # 元组有*字段名*
today: tuple[sun: string, temp: float] # 和*顺序*
child = (name: "Rudiger", age: 2) # 使用字面值()一次性赋值全部
today.sun = "Overcast" # 也可以单独赋值
today.temp = 70.1
# 序列(Sequence)
var
drinks: seq[string]
drinks = @["Water", "Juice", "Chocolate"] # @[V1,..,Vn] 是序列的字面值
drinks.add("Milk")
if "Milk" in drinks:
echo "We have Milk and ", drinks.len - 1, " other drinks"
let myDrink = drinks[2]
#
# 自定义类型
#
# 定义你自己的类型使得编译器为你工作。
# 这使得静态类型变得强大和有用。
type
Name = string # 类型别名为你提供一个新类型,
Age = int # 该类型可与旧类型互换,但更具描述性。
Person = tuple[name: Name, age: Age] # 也可以定义数据结构。
AnotherSyntax = tuple
fieldOne: string
secondField: int
var
john: Person = (name: "John B.", age: 17)
newage: int = 18 # 在这里使用Age比int要好
john.age = newage # 仍然有效,因为int和Age同义
type
Cash = distinct int # `distinct`使一个新类型与它的基本类型不兼容。
Desc = distinct string
var
money: Cash = 100.Cash # `.Cash`把int转换成我们的类型
description: Desc = "Interesting".Desc
when compileBadCode:
john.age = money # 错误!age是int类型、money是Cash类型
john.name = description # 编译器说:“没门!”
#
# 更多类型和数据结构
#
# 枚举类型只能具有有限数量的值之一
type
Color = enum cRed, cBlue, cGreen
Direction = enum # 可选格式
dNorth
dWest
dEast
dSouth
var
orient = dNorth # `orient`的类型是Direction,值是`dNorth`
pixel = cGreen # `pixel`的类型是Color,值是`cGreen`
discard dNorth > dEast # Enum通常是“序数”类型
# 子范围指定有限的有效范围
type
DieFaces = range[1..20] # 只有从1到20的int才是有效值
var
my_roll: DieFaces = 13
when compileBadCode:
my_roll = 23 # 错误!
# 数组(Array)
type
RollCounter = array[DieFaces, int] # 数组长度固定
DirNames = array[Direction, string] # 以任意有序类型索引
Truths = array[42..44, bool]
var
counter: RollCounter
directions: DirNames
possible: Truths
possible = [false, false, false] # 数组字面以[V1,..,Vn]表示
possible[42] = true
directions[dNorth] = "Ahh. The Great White North!"
directions[dWest] = "No, don't go there."
my_roll = 13
counter[my_roll] += 1
counter[my_roll] += 1
var anotherArray = ["Default index", "starts at", "0"]
# 可用的数据结构包括表、集合、列表、队列、压缩前缀树。
# http://nim-lang.org/docs/lib.html#collections-and-algorithms
#
# IO和控制流
#
# `case`, `readLine()`
echo "Read any good books lately?"
case readLine(stdin)
of "no", "No":
echo "Go to your local library."
of "yes", "Yes":
echo "Carry on, then."
else:
echo "That's great; I assume."
# `while`, `if`, `continue`, `break`
import strutils as str # http://nim-lang.org/docs/strutils.html
echo "I'm thinking of a number between 41 and 43. Guess which!"
let number: int = 42
var
raw_guess: string
guess: int
while guess != number:
raw_guess = readLine(stdin)
if raw_guess == "": continue # 跳出循环
guess = str.parseInt(raw_guess)
if guess == 1001:
echo("AAAAAAGGG!")
break
elif guess > number:
echo("Nope. Too high.")
elif guess < number:
echo(guess, " is too low")
else:
echo("Yeeeeeehaw!")
#
# 循环(Iteration)
#
for i, elem in ["Yes", "No", "Maybe so"]: # 也可以是`for elem in`
echo(elem, " is at index: ", i)
for k, v in items(@[(person: "You", power: 100), (person: "Me", power: 9000)]):
echo v
let myString = """
an <example>
`string` to
play with
""" # 多行字符串
for line in splitLines(myString):
echo(line)
for i, c in myString: # 索引和字符。或使用'for j in'只有字符
if i mod 2 == 0: continue # 紧凑的'if'形式
elif c == 'X': break
else: echo(c)
#
# 过程(Procedure)
#
type Answer = enum aYes, aNo
proc ask(question: string): Answer =
echo(question, " (y/n)")
while true:
case readLine(stdin)
of "y", "Y", "yes", "Yes":
return Answer.aYes # 枚举类型可以
of "n", "N", "no", "No":
return Answer.aNo
else: echo("Please be clear: yes or no")
proc addSugar(amount: int = 2) = # amount默认是2,不返回任何值
assert(amount > 0 and amount < 9000, "Crazy Sugar")
for a in 1..amount:
echo(a, " sugar...")
case ask("Would you like sugar in your tea?")
of aYes:
addSugar(3)
of aNo:
echo "Oh do take a little!"
addSugar()
# 这里不需要使用`else` 。只能是`yes`和`no`。
#
# 外部函数接口(FFI)
#
# 因为Nim可以编译为C,使用外部函数接口(FFI)很简单:
proc strcmp(a, b: cstring): cint {.importc: "strcmp", nodecl.}
let cmp = strcmp("C?", "Easy!")
```
除此以外,Nim通过元编程、性能和编译时特性将自己与其他同类分离开来。
## 进阶阅读
* [主页](http://nim-lang.org)
* [下载](http://nim-lang.org/download.html)
* [社区](http://nim-lang.org/community.html)
* [常见问题](http://nim-lang.org/question.html)
* [文档](http://nim-lang.org/documentation.html)
* [参考手册](http://nim-lang.org/docs/manual.html)
* [标准库](http://nim-lang.org/docs/lib.html)
* [Rosetta Code](http://rosettacode.org/wiki/Category:Nim)
|