2017-07-04 13:02:46 +00:00
|
|
|
|
---
|
|
|
|
|
contributors:
|
|
|
|
|
- ["Jason J. Ayala P.", "http://JasonAyala.com"]
|
2023-09-22 09:45:52 +00:00
|
|
|
|
- ["Dennis Felsing", "https://dennis.felsing.org"]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
translators:
|
|
|
|
|
- ["Nomadic", "https://github.com/n0madic"]
|
2019-12-13 17:58:03 +00:00
|
|
|
|
- ["dvska", "https://github.com/dvska"]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
Nim (ранее известный, как Nimrod) — язык программирования со статической
|
|
|
|
|
типизацией, поддерживающий процедурный, объектно-ориентированный,
|
|
|
|
|
функциональный и обобщённый стили программирования.
|
|
|
|
|
|
|
|
|
|
Nim эффективный, выразительный и элегантный.
|
|
|
|
|
|
|
|
|
|
```nim
|
2019-12-13 17:58:03 +00:00
|
|
|
|
var # Объявление (и присваивание) переменных,
|
|
|
|
|
буква: char = 'n' # с указанием типа или без
|
|
|
|
|
язык = "N" & "im"
|
|
|
|
|
nLength : int = len(язык)
|
2017-07-04 13:02:46 +00:00
|
|
|
|
boat: float
|
2019-12-13 17:58:03 +00:00
|
|
|
|
правда: bool = false
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
let # Используйте let *сразу* для объявления и связывания переменных.
|
|
|
|
|
ноги = 400 # ноги неизменяемый.
|
|
|
|
|
руки = 2_000 # Символ _ игнорируется и удобен для длинных чисел.
|
|
|
|
|
почтиПи = 3.15
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
const # Константы вычисляются во время компиляции. Это обеспечивает
|
|
|
|
|
debug = true # производительность и полезно в выражениях этапа компиляции.
|
2019-12-13 17:58:03 +00:00
|
|
|
|
компилироватьПлохойКод = false
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
when компилироватьПлохойКод: # `when` это `if` этапа компиляции.
|
|
|
|
|
ноги = ноги + 1 # Эта ошибка никогда не будет скомпилирована.
|
|
|
|
|
const ввод = readline(stdin) # Значения констант должны быть известны во
|
2017-07-04 13:02:46 +00:00
|
|
|
|
# время компиляции.
|
|
|
|
|
|
|
|
|
|
discard 1 > 2 # Примечание. Компилятор будет жаловаться, если результат
|
|
|
|
|
# выражения не используется. `discard` обходит это.
|
|
|
|
|
|
|
|
|
|
discard """
|
|
|
|
|
Это может использоваться как многострочный комментарий.
|
|
|
|
|
Или для не поддающегося синтаксическому анализу, сломанного кода
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# Структуры данных
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
# Кортежи
|
|
|
|
|
|
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
дитя: tuple[имя: string, возраст: int] # Кортежи определяют *как* имя поля
|
|
|
|
|
сегодня: tuple[солнце: string, температура: float] # так *и* порядок полей.
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
дитя = (имя: "Rudiger", возраст: 2) # Присвоить все сразу литералом ()
|
|
|
|
|
сегодня.солнце = "Пасмурно" # или отдельно по полям.
|
|
|
|
|
сегодня.температура = 20.1
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
# Последовательности
|
|
|
|
|
|
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
напитки: seq[string]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
напитки = @["Вода", "Сок", "Какао"] # @[V1,..,Vn] является литералом
|
|
|
|
|
# последовательности
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
напитки.add("Молоко")
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
if "Молоко" in напитки:
|
2020-08-21 14:34:01 +00:00
|
|
|
|
echo "У нас тут Молоко и ещё", напитки.len - 1, " напиток(ов)"
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
let мойНапиток = напитки[2]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# Определение типов
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
# Определение собственных типов позволяет компилятору работать на вас.
|
|
|
|
|
# Это то, что делает статическую типизацию мощной и полезной.
|
|
|
|
|
|
|
|
|
|
type
|
2019-12-13 17:58:03 +00:00
|
|
|
|
Имя = string # Псевдоним типа дает вам новый тип, который равнозначен
|
|
|
|
|
Возраст = int # старому типу, но более нагляден.
|
2020-08-21 14:34:01 +00:00
|
|
|
|
Человек = tuple[имя: Имя, возраст: Возраст] # Определение структур данных.
|
2019-12-13 17:58:03 +00:00
|
|
|
|
АльтернативныйСинтаксис = tuple
|
2017-07-04 13:02:46 +00:00
|
|
|
|
fieldOne: string
|
|
|
|
|
secondField: int
|
|
|
|
|
|
|
|
|
|
var
|
2020-08-21 14:34:01 +00:00
|
|
|
|
джон: Человек = (имя: "John B.", возраст: 17)
|
2019-12-13 17:58:03 +00:00
|
|
|
|
новыйВозраст: int = 18 # Было бы лучше использовать Возраст, чем int
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
джон.возраст = новыйВозраст # Но это все же работает, потому что int и Возраст синонимы.
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
type
|
2019-12-13 17:58:03 +00:00
|
|
|
|
Нал = distinct int # `distinct` делает новый тип несовместимым с его
|
|
|
|
|
Описание = distinct string # базовым типом.
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
money: Нал = 100.Нал # `.Нал` преобразует int в наш тип
|
|
|
|
|
описание: Описание = "Interesting".Описание
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
when компилироватьПлохойКод:
|
|
|
|
|
джон.возраст = money # Error! возраст is of type int and money is Нал
|
|
|
|
|
джон.имя = описание # Компилятор говорит: "Нельзя!"
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# Дополнительные типы и структуры данных
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
# Перечисления позволяют типу иметь одно из ограниченного числа значений
|
|
|
|
|
|
|
|
|
|
type
|
2019-12-13 17:58:03 +00:00
|
|
|
|
Цвет = enum цКрасный, цГолубой, цЗеленый
|
|
|
|
|
Направление = enum # Альтернативный формат
|
|
|
|
|
нСевер
|
|
|
|
|
нЗапад
|
|
|
|
|
нВосток
|
|
|
|
|
нЮг
|
2017-07-04 13:02:46 +00:00
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
напр = нСевер # `напр` имеет тип Направление, со значением `нСевер`
|
|
|
|
|
точка = цЗеленый # `точка` имеет тип Цвет, со значением `цЗеленый`
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
discard нСевер > нВосток # Перечисления обычно являются "порядковыми" типами
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
# Поддиапазоны определяют ограниченный допустимый диапазон
|
|
|
|
|
|
|
|
|
|
type
|
2019-12-13 17:58:03 +00:00
|
|
|
|
Кости = range[1..20] # 🎲 Допустимым значением являются только int от 1 до 20
|
2017-07-04 13:02:46 +00:00
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
мой_бросок: Кости = 13
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
when компилироватьПлохойКод:
|
|
|
|
|
мой_бросок = 23 # Error!
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
# Массивы
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
type
|
2019-12-13 17:58:03 +00:00
|
|
|
|
СчетчикБросков = array[Кости, int] # Массивы фиксированной длины и
|
|
|
|
|
ИменаНаправлений = array[Направление, string] # индексируются любым порядковым типом.
|
|
|
|
|
Истины = array[42..44, bool]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
счетчик: СчетчикБросков
|
|
|
|
|
направления: ИменаНаправлений
|
2020-08-21 14:34:01 +00:00
|
|
|
|
возможны: Истины
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2020-08-21 14:34:01 +00:00
|
|
|
|
возможны = [false, false, false] # Массивы создаются литералом [V1,..,Vn]
|
|
|
|
|
возможны[42] = true
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
направления[нСевер] = "ОО. Великий белый Север!"
|
|
|
|
|
направления[нЗапад] = "Нет, не иди туда."
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
мой_бросок = 13
|
|
|
|
|
счетчик[мой_бросок] += 1
|
|
|
|
|
счетчик[мой_бросок] += 1
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
var ещеМассив = ["Идекс по умолчанию", "начинается с", "0"]
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
# Доступны другие структуры данных, в том числе таблицы, множества,
|
|
|
|
|
# списки, очереди и crit-bit деревья.
|
|
|
|
|
# http://nim-lang.org/docs/lib.html#collections-and-algorithms (EN)
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# IO и поток управления выполнением
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
# `case`, `readLine()`
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
echo "Читали какие-нибудь хорошие книги в последнее время?"
|
|
|
|
|
|
2017-07-04 13:02:46 +00:00
|
|
|
|
case readLine(stdin)
|
2019-12-13 17:58:03 +00:00
|
|
|
|
of "нет", "Нет":
|
|
|
|
|
echo "Пойдите в свою местную библиотеку."
|
|
|
|
|
of "да", "Да":
|
|
|
|
|
echo "Тогда продолжим"
|
2017-07-04 13:02:46 +00:00
|
|
|
|
else:
|
2019-12-13 17:58:03 +00:00
|
|
|
|
echo "Здорово!"
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
# `while`, `if`, `continue`, `break`
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
import strutils as str # http://nim-lang.org/docs/strutils.html (EN)
|
|
|
|
|
echo "Я загадало число между 41 и 43. Отгадай!"
|
|
|
|
|
let число: int = 42
|
2017-07-04 13:02:46 +00:00
|
|
|
|
var
|
2019-12-13 17:58:03 +00:00
|
|
|
|
ввод_догадка: string
|
|
|
|
|
догадка: int
|
|
|
|
|
|
|
|
|
|
while догадка != число:
|
|
|
|
|
ввод_догадка = readLine(stdin)
|
|
|
|
|
|
|
|
|
|
if ввод_догадка == "": continue # Пропустить эту итерацию
|
|
|
|
|
|
|
|
|
|
догадка = str.parseInt(ввод_догадка)
|
|
|
|
|
|
|
|
|
|
if догадка == 1001:
|
2017-07-04 13:02:46 +00:00
|
|
|
|
echo("AAAAAAGGG!")
|
|
|
|
|
break
|
2019-12-13 17:58:03 +00:00
|
|
|
|
elif догадка > число:
|
|
|
|
|
echo("Неа. Слишком большое.")
|
|
|
|
|
elif догадка < число:
|
|
|
|
|
echo(догадка, " это слишком мало")
|
2017-07-04 13:02:46 +00:00
|
|
|
|
else:
|
2019-12-13 17:58:03 +00:00
|
|
|
|
echo("Точнооооо!")
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# Итерации (циклы)
|
|
|
|
|
#
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
for i, элем in ["Да", "Нет", "Может быть"]: # Или просто `for элем in`
|
|
|
|
|
echo(элем, " по индексу: ", i)
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2020-08-21 14:34:01 +00:00
|
|
|
|
for ключ, значение in items(@[(человек: "You", сила: 100), (человек: "Me", сила: 9000)]):
|
|
|
|
|
echo значение
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
let мояСтрока = """
|
|
|
|
|
<пример>
|
|
|
|
|
`строки` для
|
|
|
|
|
тренировки
|
2017-07-04 13:02:46 +00:00
|
|
|
|
""" # Многострочная "сырая" строка
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
for строка in splitLines(мояСтрока):
|
|
|
|
|
echo(строка)
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
for i, симв in мояСтрока: # Индекс и символ. Или `for j in` только для символов
|
|
|
|
|
if i mod 2 == 0: continue # Компактная форма `if`
|
|
|
|
|
elif симв == 'X': break
|
|
|
|
|
else: echo(симв)
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# Процедуры
|
|
|
|
|
#
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
type Ответ = enum оДа, оНет
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
proc спрос(вопрос: string): Ответ =
|
|
|
|
|
echo(вопрос, " (д/н)")
|
2017-07-04 13:02:46 +00:00
|
|
|
|
while true:
|
|
|
|
|
case readLine(stdin)
|
2019-12-13 17:58:03 +00:00
|
|
|
|
of "д", "Д", "да", "Да":
|
|
|
|
|
return Ответ.оДа # Перечисления могут быть квалифицированы
|
|
|
|
|
of "н", "Н", "нет", "Нет":
|
|
|
|
|
return Ответ.оНет
|
|
|
|
|
else: echo("Поточнее, да или нет")
|
|
|
|
|
|
2020-08-21 14:36:27 +00:00
|
|
|
|
proc добавьСахар(количество: int = 2) = # Значение по умолчанию 2, ничего не возвращает
|
|
|
|
|
assert(количество > 0 and количество < 9000, "Диабет ☠")
|
|
|
|
|
for a in 1..количество:
|
2019-12-13 17:58:03 +00:00
|
|
|
|
echo(a, " кубик...")
|
|
|
|
|
|
|
|
|
|
case спрос("Сахарку?")
|
|
|
|
|
of оДа:
|
|
|
|
|
добавьСахар(3)
|
|
|
|
|
of оНет:
|
|
|
|
|
echo "Ну немнооожко!"
|
|
|
|
|
добавьСахар()
|
|
|
|
|
# Здесь нет необходимости в `else`. Возможны только `да` и `нет`.
|
2017-07-04 13:02:46 +00:00
|
|
|
|
|
|
|
|
|
#
|
|
|
|
|
# FFI (интерфейс внешних функций)
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
# Так как Nim компилируется в C, то FFI делается очень просто:
|
|
|
|
|
|
|
|
|
|
proc strcmp(a, b: cstring): cint {.importc: "strcmp", nodecl.}
|
|
|
|
|
|
2019-12-13 17:58:03 +00:00
|
|
|
|
let cmp = strcmp("C?", "Легко!")
|
2017-07-04 13:02:46 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Кроме того, Nim выделяется среди себе подобных метапрограммированием,
|
|
|
|
|
производительностью, функциями этапа компиляции.
|
|
|
|
|
|
|
|
|
|
## Дальнейшее чтение (EN)
|
|
|
|
|
|
|
|
|
|
* [Домашняя страница](http://nim-lang.org)
|
|
|
|
|
* [Скачать](http://nim-lang.org/download.html)
|
|
|
|
|
* [Сообщество](http://nim-lang.org/community.html)
|
|
|
|
|
* [FAQ](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)
|