learnxinyminutes-docs/ru/crystal.md
2024-12-08 20:29:09 -07:00

16 KiB
Raw Blame History

language filename contributors translators lang
Crystal learncrystal-ru.cr
Vitalii Elenhaupt
http://veelenga.com
Arnaud Fernandés
https://github.com/TechMagister/
Den Patin
https://github.com/denpatin
ru-ru
# — так начинается комментарий


# Всё является объектом
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"

Дополнительная информация

На русском

На английском