mirror of
https://github.com/adambard/learnxinyminutes-docs.git
synced 2025-01-18 07:05:58 +00:00
581 lines
16 KiB
Markdown
581 lines
16 KiB
Markdown
---
|
||
contributors:
|
||
- ["Vitalii Elenhaupt", "http://veelenga.com"]
|
||
- ["Arnaud Fernandés", "https://github.com/TechMagister/"]
|
||
translators:
|
||
- ["Den Patin", "https://github.com/denpatin"]
|
||
---
|
||
|
||
```crystal
|
||
# — так начинается комментарий
|
||
|
||
|
||
# Всё является объектом
|
||
nil.class #=> Nil
|
||
100.class #=> Int32
|
||
true.class #=> Bool
|
||
|
||
# Возвращают false только nil, false и пустые указатели
|
||
!nil #=> true : Bool
|
||
!false #=> true : Bool
|
||
!0 #=> false : Bool
|
||
|
||
|
||
# Целые числа
|
||
|
||
1.class #=> Int32
|
||
|
||
# Четыре типа целых чисел со знаком
|
||
1_i8.class #=> Int8
|
||
1_i16.class #=> Int16
|
||
1_i32.class #=> Int32
|
||
1_i64.class #=> Int64
|
||
|
||
# Четыре типа целых чисел без знака
|
||
1_u8.class #=> UInt8
|
||
1_u16.class #=> UInt16
|
||
1_u32.class #=> UInt32
|
||
1_u64.class #=> UInt64
|
||
|
||
2147483648.class #=> Int64
|
||
9223372036854775808.class #=> UInt64
|
||
|
||
# Двоичные числа
|
||
0b1101 #=> 13 : Int32
|
||
|
||
# Восьмеричные числа
|
||
0o123 #=> 83 : Int32
|
||
|
||
# Шестнадцатеричные числа
|
||
0xFE012D #=> 16646445 : Int32
|
||
0xfe012d #=> 16646445 : Int32
|
||
|
||
# Числа с плавающей точкой
|
||
|
||
1.0.class #=> Float64
|
||
|
||
# Два типа чисел с плавающей запятой
|
||
1.0_f32.class #=> Float32
|
||
1_f32.class #=> Float32
|
||
|
||
1e10.class #=> Float64
|
||
1.5e10.class #=> Float64
|
||
1.5e-7.class #=> Float64
|
||
|
||
|
||
# Символьные литералы
|
||
|
||
'a'.class #=> Char
|
||
|
||
# Восьмеричный код символа
|
||
'\101' #=> 'A' : Char
|
||
|
||
# Код символа Unicode
|
||
'\u0041' #=> 'A' : Char
|
||
|
||
|
||
# Строки
|
||
|
||
"s".class #=> String
|
||
|
||
# Строки неизменяемы
|
||
s = "hello, " #=> "hello, " : String
|
||
s.object_id #=> 134667712 : UInt64
|
||
s += "Crystal" #=> "hello, Crystal" : String
|
||
s.object_id #=> 142528472 : UInt64
|
||
|
||
# Поддерживается интерполяция строк
|
||
"sum = #{1 + 2}" #=> "sum = 3" : String
|
||
|
||
# Поддерживается многострочность
|
||
"This is
|
||
multiline string"
|
||
|
||
# Строка с двойными кавычками
|
||
%(hello "world") #=> "hello \"world\""
|
||
|
||
|
||
# Символы — константы без значения, определяемые только именем. Часто
|
||
# используются вместо часто используемых строк для лучшей производительности.
|
||
# На внутреннем уровне они представлены как Int32.
|
||
|
||
:symbol.class #=> Symbol
|
||
|
||
sentence = :question? # :"question?" : Symbol
|
||
|
||
sentence == :question? #=> true : Bool
|
||
sentence == :exclamation! #=> false : Bool
|
||
sentence == "question?" #=> false : Bool
|
||
|
||
|
||
# Массивы
|
||
|
||
[1, 2, 3].class #=> Array(Int32)
|
||
[1, "hello", 'x'].class #=> Array(Int32 | String | Char)
|
||
|
||
# При объявлении пустого массива необходимо указать тип его элементов
|
||
[] # Syntax error: for empty arrays use '[] of ElementType'
|
||
[] of Int32 #=> [] : Array(Int32)
|
||
Array(Int32).new #=> [] : Array(Int32)
|
||
|
||
# Элементы внутри массива имеют свои индексы
|
||
array = [1, 2, 3, 4, 5] #=> [1, 2, 3, 4, 5] : Array(Int32)
|
||
array[0] #=> 1 : Int32
|
||
array[10] # raises IndexError
|
||
array[-6] # raises IndexError
|
||
array[10]? #=> nil : (Int32 | Nil)
|
||
array[-6]? #=> nil : (Int32 | Nil)
|
||
|
||
# Можно получать элементы по индексу с конца
|
||
array[-1] #=> 5
|
||
|
||
# С начала и с указанием размера итогового массива
|
||
array[2, 3] #=> [3, 4, 5]
|
||
|
||
# Или посредством указания диапазона
|
||
array[1..3] #=> [2, 3, 4]
|
||
|
||
# Добавление в массив
|
||
array << 6 #=> [1, 2, 3, 4, 5, 6]
|
||
|
||
# Удаление элемента из конца массива
|
||
array.pop #=> 6
|
||
array #=> [1, 2, 3, 4, 5]
|
||
|
||
# Удаление элемента из начала массива
|
||
array.shift #=> 1
|
||
array #=> [2, 3, 4, 5]
|
||
|
||
# Проверка на наличие элемента в массиве
|
||
array.includes? 3 #=> true
|
||
|
||
# Синтаксический сахар для массива строк и символов
|
||
%w(one two three) #=> ["one", "two", "three"] : Array(String)
|
||
%i(one two three) #=> [:one, :two, :three] : Array(Symbol)
|
||
|
||
# Массивоподобный синтаксис используется и для других типов, только если для
|
||
# них определены методы .new и #<<
|
||
set = Set{1, 2, 3} #=> [1, 2, 3]
|
||
set.class #=> Set(Int32)
|
||
|
||
# Вышеприведенное эквивалентно следующему
|
||
set = Set(typeof(1, 2, 3)).new
|
||
set << 1
|
||
set << 2
|
||
set << 3
|
||
|
||
|
||
# Хэши
|
||
|
||
{1 => 2, 3 => 4}.class #=> Hash(Int32, Int32)
|
||
{1 => 2, 'a' => 3}.class #=> Hash(Int32 | Char, Int32)
|
||
|
||
# При объявлении пустого хэша необходимо указать типы ключа и значения
|
||
{} # Syntax error
|
||
{} of Int32 => Int32 # {}
|
||
Hash(Int32, Int32).new # {}
|
||
|
||
# Значения в хэше легко найти по ключу
|
||
hash = {"color" => "green", "number" => 5}
|
||
hash["color"] #=> "green"
|
||
hash["no_such_key"] #=> Missing hash key: "no_such_key" (KeyError)
|
||
hash["no_such_key"]? #=> nil
|
||
|
||
# Проверка наличия ключа в хэше
|
||
hash.has_key? "color" #=> true
|
||
|
||
# Синтаксический сахар для символьных и строковых ключей
|
||
{key1: 'a', key2: 'b'} # {:key1 => 'a', :key2 => 'b'}
|
||
{"key1": 'a', "key2": 'b'} # {"key1" => 'a', "key2" => 'b'}
|
||
|
||
# Хэшеподобный синтаксис используется и для других типов, только если для них
|
||
# определены методы .new и #[]=
|
||
class MyType
|
||
def []=(key, value)
|
||
puts "do stuff"
|
||
end
|
||
end
|
||
|
||
MyType{"foo" => "bar"}
|
||
|
||
# Вышеприведенное эквивалентно следующему
|
||
tmp = MyType.new
|
||
tmp["foo"] = "bar"
|
||
tmp
|
||
|
||
|
||
# Диапазоны
|
||
|
||
1..10 #=> Range(Int32, Int32)
|
||
Range.new(1, 10).class #=> Range(Int32, Int32)
|
||
|
||
# Включающий и исключающий диапазоны
|
||
(3..5).to_a #=> [3, 4, 5]
|
||
(3...5).to_a #=> [3, 4]
|
||
|
||
# Проверка на вхождение в диапазон
|
||
(1..8).includes? 2 #=> true
|
||
|
||
|
||
# Кортежи
|
||
# Неизменяемые последовательности фиксированного размера, содержащие,
|
||
# как правило, элементы разных типов
|
||
|
||
{1, "hello", 'x'}.class #=> Tuple(Int32, String, Char)
|
||
|
||
# Доступ к элементам осуществляется по индексу
|
||
tuple = {:key1, :key2}
|
||
tuple[1] #=> :key2
|
||
tuple[2] #=> syntax error : Index out of bound
|
||
|
||
# Элементы кортежей можно попарно присвоить переменным
|
||
a, b, c = {:a, 'b', "c"}
|
||
a #=> :a
|
||
b #=> 'b'
|
||
c #=> "c"
|
||
|
||
|
||
# Процедуры
|
||
# Указатели на функцию с необязательным содержимым (замыкание).
|
||
# Обычно создаётся с помощью специального литерала ->
|
||
|
||
proc = ->(x : Int32) { x.to_s }
|
||
proc.class # Proc(Int32, String)
|
||
# Или посредством метода .new
|
||
Proc(Int32, String).new { |x| x.to_s }
|
||
|
||
# Вызываются посредством метода .call
|
||
proc.call 10 #=> "10"
|
||
|
||
|
||
# Управляющие операторы
|
||
|
||
if true
|
||
"if statement"
|
||
elsif false
|
||
"else-if, optional"
|
||
else
|
||
"else, also optional"
|
||
end
|
||
|
||
puts "if as a suffix" if true
|
||
|
||
# if как часть выражения
|
||
a = if 2 > 1
|
||
3
|
||
else
|
||
4
|
||
end
|
||
|
||
a #=> 3
|
||
|
||
# Тернарный if
|
||
a = 1 > 2 ? 3 : 4 #=> 4
|
||
|
||
# Оператор выбора
|
||
cmd = "move"
|
||
|
||
action = case cmd
|
||
when "create"
|
||
"Creating..."
|
||
when "copy"
|
||
"Copying..."
|
||
when "move"
|
||
"Moving..."
|
||
when "delete"
|
||
"Deleting..."
|
||
end
|
||
|
||
action #=> "Moving..."
|
||
|
||
|
||
# Циклы
|
||
|
||
index = 0
|
||
while index <= 3
|
||
puts "Index: #{index}"
|
||
index += 1
|
||
end
|
||
# Index: 0
|
||
# Index: 1
|
||
# Index: 2
|
||
# Index: 3
|
||
|
||
index = 0
|
||
until index > 3
|
||
puts "Index: #{index}"
|
||
index += 1
|
||
end
|
||
# Index: 0
|
||
# Index: 1
|
||
# Index: 2
|
||
# Index: 3
|
||
|
||
# Но лучше использовать each
|
||
(1..3).each do |index|
|
||
puts "Index: #{index}"
|
||
end
|
||
# Index: 1
|
||
# Index: 2
|
||
# Index: 3
|
||
|
||
# Тип переменной зависит от типа выражения
|
||
if a < 3
|
||
a = "hello"
|
||
else
|
||
a = true
|
||
end
|
||
typeof a #=> (Bool | String)
|
||
|
||
if a && b
|
||
# здесь гарантируется, что и a, и b — не nil
|
||
end
|
||
|
||
if a.is_a? String
|
||
a.class #=> String
|
||
end
|
||
|
||
|
||
# Методы
|
||
|
||
def double(x)
|
||
x * 2
|
||
end
|
||
|
||
# Методы (а также любые блоки) всегда возвращают значение последнего выражения
|
||
double(2) #=> 4
|
||
|
||
# Скобки можно опускать, если вызов метода не вносит двусмысленности
|
||
double 3 #=> 6
|
||
|
||
double double 3 #=> 12
|
||
|
||
def sum(x, y)
|
||
x + y
|
||
end
|
||
|
||
# Параметры методов перечисляются через запятую
|
||
sum 3, 4 #=> 7
|
||
|
||
sum sum(3, 4), 5 #=> 12
|
||
|
||
|
||
# yield
|
||
|
||
# У всех методов есть неявный необязательный параметр блока, который можно
|
||
# вызвать ключевым словом yield
|
||
|
||
def surround
|
||
puts '{'
|
||
yield
|
||
puts '}'
|
||
end
|
||
|
||
surround { puts "hello world" }
|
||
|
||
# {
|
||
# hello world
|
||
# }
|
||
|
||
# Методу можно передать блок
|
||
# & — ссылка на переданный блок
|
||
def guests(&block)
|
||
block.call "some_argument"
|
||
end
|
||
|
||
# Методу можно передать список параметров, доступ к ним будет как к массиву
|
||
# Для этого используется оператор *
|
||
def guests(*array)
|
||
array.each { |guest| puts guest }
|
||
end
|
||
|
||
# Если метод возвращает массив, можно попарно присвоить значение каждого из его
|
||
# элементов переменным
|
||
def foods
|
||
["pancake", "sandwich", "quesadilla"]
|
||
end
|
||
breakfast, lunch, dinner = foods
|
||
breakfast #=> "pancake"
|
||
dinner #=> "quesadilla"
|
||
|
||
# По соглашению название методов, возвращающих булево значение, должно
|
||
# оканчиваться вопросительным знаком
|
||
5.even? # false
|
||
5.odd? # true
|
||
|
||
# Если название метода оканчивается восклицательным знаком, по соглашению это
|
||
# означает, что метод делает что-то необратимое, например изменяет получателя.
|
||
# Некоторые методы имеют две версии: "опасную" версию с !, которая что-то
|
||
# меняет, и "безопасную", которая просто возвращает новое значение
|
||
company_name = "Dunder Mifflin"
|
||
company_name.gsub "Dunder", "Donald" #=> "Donald Mifflin"
|
||
company_name #=> "Dunder Mifflin"
|
||
company_name.gsub! "Dunder", "Donald"
|
||
company_name #=> "Donald Mifflin"
|
||
|
||
|
||
# Классы
|
||
# Определяются с помощью ключевого слова class
|
||
|
||
class Human
|
||
|
||
# Переменная класса является общей для всех экземпляров этого класса
|
||
@@species = "H. sapiens"
|
||
|
||
# Объявление типа переменной name экземпляра класса
|
||
@name : String
|
||
|
||
# Базовый конструктор
|
||
# Значением первого параметра инициализируем переменную @name.
|
||
# То же делаем и со вторым параметром — переменная @age. В случае, если мы
|
||
# не передаём второй параметр, для инициализации @age будет взято значение
|
||
# по умолчанию (в данном случае — 0)
|
||
def initialize(@name, @age = 0)
|
||
end
|
||
|
||
# Базовый метод установки значения переменной
|
||
def name=(name)
|
||
@name = name
|
||
end
|
||
|
||
# Базовый метод получения значения переменной
|
||
def name
|
||
@name
|
||
end
|
||
|
||
# Синтаксический сахар одновременно для двух методов выше
|
||
property :name
|
||
|
||
# А также по отдельности
|
||
getter :name
|
||
setter :name
|
||
|
||
# Метод класса определяется ключевым словом self, чтобы его можно было
|
||
# различить с методом экземпляра класса. Такой метод можно вызвать только
|
||
# на уровне класса, а не экземпляра.
|
||
def self.say(msg)
|
||
puts msg
|
||
end
|
||
|
||
def species
|
||
@@species
|
||
end
|
||
end
|
||
|
||
|
||
# Создание экземпляра класса
|
||
jim = Human.new("Jim Halpert")
|
||
|
||
dwight = Human.new("Dwight K. Schrute")
|
||
|
||
# Вызов методов экземпляра класса
|
||
jim.species #=> "H. sapiens"
|
||
jim.name #=> "Jim Halpert"
|
||
jim.name = "Jim Halpert II" #=> "Jim Halpert II"
|
||
jim.name #=> "Jim Halpert II"
|
||
dwight.species #=> "H. sapiens"
|
||
dwight.name #=> "Dwight K. Schrute"
|
||
|
||
# Вызов метода класса
|
||
Human.say("Hi") #=> выведет "Hi" и вернёт nil
|
||
|
||
# Переменные экземпляра класса (@) видно только в пределах экземпляра
|
||
class TestClass
|
||
@var = "I'm an instance var"
|
||
end
|
||
|
||
# Переменные класса (@) видны как в экземплярах класса, так и в самом классе
|
||
class TestClass
|
||
@@var = "I'm a class var"
|
||
end
|
||
|
||
# Переменные с большой буквы — это константы
|
||
Var = "I'm a constant"
|
||
Var = "can't be updated" # Error: already initialized constant Var
|
||
|
||
# Примеры
|
||
|
||
# Базовый класс
|
||
class Human
|
||
@@foo = 0
|
||
|
||
def self.foo
|
||
@@foo
|
||
end
|
||
|
||
def self.foo=(value)
|
||
@@foo = value
|
||
end
|
||
end
|
||
|
||
# Класс-потомок
|
||
class Worker < Human
|
||
end
|
||
|
||
Human.foo #=> 0
|
||
Worker.foo #=> 0
|
||
|
||
Human.foo = 2 #=> 2
|
||
Worker.foo #=> 0
|
||
|
||
Worker.foo = 3 #=> 3
|
||
Human.foo #=> 2
|
||
Worker.foo #=> 3
|
||
|
||
module ModuleExample
|
||
def foo
|
||
"foo"
|
||
end
|
||
end
|
||
|
||
# Подключение модуля в класс добавляет его методы в экземпляр класса
|
||
# Расширение модуля добавляет его методы в сам класс
|
||
|
||
class Person
|
||
include ModuleExample
|
||
end
|
||
|
||
class Book
|
||
extend ModuleExample
|
||
end
|
||
|
||
Person.foo # => undefined method 'foo' for Person:Class
|
||
Person.new.foo # => 'foo'
|
||
Book.foo # => 'foo'
|
||
Book.new.foo # => undefined method 'foo' for Book
|
||
|
||
|
||
# Обработка исключений
|
||
|
||
# Создание пользовательского типа исключения
|
||
class MyException < Exception
|
||
end
|
||
|
||
# Ещё одного
|
||
class MyAnotherException < Exception; end
|
||
|
||
ex = begin
|
||
raise MyException.new
|
||
rescue ex1 : IndexError
|
||
"ex1"
|
||
rescue ex2 : MyException | MyAnotherException
|
||
"ex2"
|
||
rescue ex3 : Exception
|
||
"ex3"
|
||
rescue ex4 # без указания конкретного типа исключения будут "отлавливаться" все
|
||
"ex4"
|
||
end
|
||
|
||
ex #=> "ex2"
|
||
```
|
||
|
||
## Дополнительная информация
|
||
|
||
### На русском
|
||
|
||
- [Официальная документация](http://ru.crystal-lang.org/docs/)
|
||
|
||
### На английском
|
||
|
||
- [Official Documentation](http://crystal-lang.org/)
|