mirror of
https://github.com/adambard/learnxinyminutes-docs.git
synced 2025-01-22 00:55:58 +00:00
590 lines
14 KiB
Markdown
590 lines
14 KiB
Markdown
---
|
|
contributors:
|
|
- ["David Underwood", "http://theflyingdeveloper.com"]
|
|
- ["Joel Walden", "http://joelwalden.net"]
|
|
- ["Luke Holder", "http://twitter.com/lukeholder"]
|
|
- ["Tristan Hume", "http://thume.ca/"]
|
|
- ["Nick LaMuro", "https://github.com/NickLaMuro"]
|
|
- ["Marcos Brizeno", "http://www.about.me/marcosbrizeno"]
|
|
- ["Ariel Krakowski", "http://www.learneroo.com"]
|
|
- ["Dzianis Dashkevich", "https://github.com/dskecse"]
|
|
- ["Levi Bostian", "https://github.com/levibostian"]
|
|
- ["Rahil Momin", "https://github.com/iamrahil"]
|
|
- ["Gabriel Halley", "https://github.com/ghalley"]
|
|
- ["Persa Zula", "http://persazula.com"]
|
|
translators:
|
|
- ["Marcin Klocek", "https://github.com/mklocek"]
|
|
---
|
|
|
|
```ruby
|
|
# To jest komentarz
|
|
|
|
=begin
|
|
To jest wielolinijkowy komentarz
|
|
Nikt ich nie używa
|
|
Ty też nie powinieneś
|
|
=end
|
|
|
|
# Przede wszystkim: Wszystko jest obiektem.
|
|
|
|
# Liczby są obiektami
|
|
|
|
3.class #=> Fixnum
|
|
|
|
3.to_s #=> "3"
|
|
|
|
|
|
# Trochę podstawowej arytmetyki
|
|
1 + 1 #=> 2
|
|
8 - 1 #=> 7
|
|
10 * 2 #=> 20
|
|
35 / 5 #=> 7
|
|
2**5 #=> 32
|
|
5 % 3 #=> 2
|
|
5 ^ 6 #=> 3
|
|
|
|
# Arytmetyka jest zastąpeniem składni
|
|
# metod wywoływanych na obiektach
|
|
1.+(3) #=> 4
|
|
10.* 5 #=> 50
|
|
|
|
# Wartości specjalne są obiektami
|
|
nil # To na prawdę jest niczym
|
|
true # prawda
|
|
false # fałsz
|
|
|
|
nil.class #=> NilClass
|
|
true.class #=> TrueClass
|
|
false.class #=> FalseClass
|
|
|
|
# Równość
|
|
1 == 1 #=> true
|
|
2 == 1 #=> false
|
|
|
|
# Nierówność
|
|
1 != 1 #=> false
|
|
2 != 1 #=> true
|
|
|
|
# jedyną 'fałszywą' wartością poza false, jest nil
|
|
|
|
!nil #=> true
|
|
!false #=> true
|
|
!0 #=> false
|
|
|
|
# Więcej porównań
|
|
1 < 10 #=> true
|
|
1 > 10 #=> false
|
|
2 <= 2 #=> true
|
|
2 >= 2 #=> true
|
|
|
|
# Operatory logiczne
|
|
true && false #=> false
|
|
true || false #=> true
|
|
!true #=> false
|
|
|
|
# Istnieją alternatywne wersje operatorów logicznych ze znacznie mniejszym
|
|
# pierwszeństwem. Używane są by kontrolować wyrażenia w łańcuchach wyrażeń
|
|
# aż jedno z nich wróci true lub false.
|
|
|
|
# `zrob_cos_innego` wywołaj tylko wtedy gdy `zrob_cos` zakończy się sukcesem.
|
|
zrob_cos_innego() and zrob_cos()
|
|
# `log_error` wywołaj tylko wtedy gdy `zrob_cos` nie zakończy się sukcesem.
|
|
zrob_cos() or log_error()
|
|
|
|
|
|
# Stringi są obiektami
|
|
|
|
'Jestem stringiem.'.class #=> String
|
|
"Ja również jestem stringiem.".class #=> String
|
|
|
|
wypelnienie = 'użyć interpolacji stringa'
|
|
"Potrafię #{wypelnienie} używając podwójnych cudzysłowów."
|
|
#=> "Potrafię użyć interpolacji stringa używając podwójnych cudzysłowów."
|
|
|
|
# Staraj się zapisywać stringi za pomocą apostrof, zamiast cudzysłowów tam, gdzie to możliwe
|
|
# Cudzysłowy wykonują dodatkowe wewnętrzne operacje
|
|
|
|
|
|
# Łączenie stringów, ale nie liczb
|
|
'hej ' + 'świecie' #=> "hej świecie"
|
|
'hej ' + 3 #=> TypeError: can't convert Fixnum into String
|
|
'hej ' + 3.to_s #=> "hej 3"
|
|
|
|
# Łączenie stringów i operatorów
|
|
'hej ' * 3 #=> "hej hej hej "
|
|
|
|
# Dodawanie do stringa
|
|
'hej' << ' świecie' #=> "hej świecie"
|
|
|
|
# wydrukowanie wartości wraz z nową linią na końcu
|
|
puts "Drukuję!"
|
|
#=> Drukuję!
|
|
#=> nil
|
|
|
|
# wydrukowanie wartości bez nowej linii na końcu
|
|
print "Drukuję!"
|
|
#=> Drukuję! => nill
|
|
|
|
# Zmienne
|
|
x = 25 #=> 25
|
|
x #=> 25
|
|
|
|
# Zauważ, że przypisanie zwraca przypisywaną wartość
|
|
# To znaczy, że możesz wykonać wielokrotne przypisanie:
|
|
|
|
x = y = 10 #=> 10
|
|
x #=> 10
|
|
y #=> 10
|
|
|
|
# Zwyczajowo, używaj notacji nazwa_zmiennej dla nazw zmiennych
|
|
nazwa_zmiennej = true
|
|
|
|
# Używaj opisowych nazw zmiennych
|
|
sciezka_do_projektu = '/dobra/nazwa/'
|
|
sciezka = '/zla/nazwa/'
|
|
|
|
# Symbole (są obiektami)
|
|
# Symbole są niezmiennymi, wielokrotnie używanymi stałymi reprezentowanymi wewnętrznie jako
|
|
# liczby całkowite. Często używane są zamiast stringów w celu wydajniejszego przekazywania danych
|
|
|
|
:oczekujacy.class #=> Symbol
|
|
|
|
status = :oczekujacy
|
|
|
|
status == :oczekujacy #=> true
|
|
|
|
status == 'oczekujacy' #=> false
|
|
|
|
status == :zatwierdzony #=> false
|
|
|
|
# Tablice
|
|
|
|
# To jest tablica
|
|
array = [1, 2, 3, 4, 5] #=> [1, 2, 3, 4, 5]
|
|
|
|
# Tablice mogą zwierać różne typy danych
|
|
|
|
[1, 'hej', false] #=> [1, "hej", false]
|
|
|
|
# Tablice mogę być indeksowane
|
|
# Od początku
|
|
tablica[0] #=> 1
|
|
tablica.first #=> 1
|
|
tablica[12] #=> nil
|
|
|
|
# Podobnie jak przy arytmetyce, dostęp poprzez [zmienna]
|
|
# jest tylko czytelniejszą składnią
|
|
# dla wywoływania metody [] na obiekcie
|
|
tablica.[] 0 #=> 1
|
|
tablica.[] 12 #=> nil
|
|
|
|
# Od końca
|
|
tablica[-1] #=> 5
|
|
tablica.last #=> 5
|
|
|
|
# Z początkowym indeksem i długością
|
|
tablica[2, 3] #=> [3, 4, 5]
|
|
|
|
# Odwrotność tablicy
|
|
a=[1,2,3]
|
|
a.reverse! #=> [3,2,1]
|
|
|
|
# Lub zakres
|
|
array[1..3] #=> [2, 3, 4]
|
|
|
|
# Dodawanie do tablicy w taki sposób
|
|
tablica << 6 #=> [1, 2, 3, 4, 5, 6]
|
|
# Lub taki
|
|
tablica.push(6) #=> [1, 2, 3, 4, 5, 6]
|
|
|
|
# Sprawdzanie, czy tablica zawiera element
|
|
tablica.include?(1) #=> true
|
|
|
|
# Hasze są Ruby'owymi podstawowymi słownikami z parami klucz/wartość.
|
|
# Hasze są zapisywane za pomocą nawiasów klamrowych
|
|
hasz = { 'kolor' => 'zielony', 'numer' => 5 }
|
|
|
|
hasz.keys #=> ['kolor', 'numer']
|
|
|
|
# Można szybko sprawdzić zawartość hasza za pomocą kluczy:
|
|
hasz['kolor'] #=> 'zielony'
|
|
hasz['numer'] #=> 5
|
|
|
|
# Sprawdzenie wartośći dla nieistniejącego klucza zwraca nil:
|
|
hasz['nic tutaj nie ma'] #=> nil
|
|
|
|
# Od wersji 1.9, Ruby posiada specjalną składnię, gdy używamy symboli jako kluczy:
|
|
|
|
nowy_hasz = { stan: 3, akcja: true }
|
|
|
|
nowy_hasz.keys #=> [:stan, :akcja]
|
|
|
|
# Sprawdzenie istnienia kluczy i wartości w haszu
|
|
new_hash.has_key?(:defcon) #=> true
|
|
new_hash.has_value?(3) #=> true
|
|
|
|
# Wskazówka: Zarówno tablice, jak i hasze, są policzalne
|
|
# Współdzielą wiele metod takich jak each, map, count, i inne
|
|
|
|
# Instrukcje warunkowe
|
|
|
|
if true
|
|
'wyrażenie if'
|
|
elsif false
|
|
'wyrażenie if, opcjonalne'
|
|
else
|
|
'wyrażenie else, również opcjonalne'
|
|
end
|
|
|
|
for licznik in 1..5
|
|
puts "powtórzenie #{licznik}"
|
|
end
|
|
#=> powtórzenie 1
|
|
#=> powtórzenie 2
|
|
#=> powtórzenie 3
|
|
#=> powtórzenie 4
|
|
#=> powtórzenie 5
|
|
|
|
# JEDNAKŻE, Nikt nie używa pętli for.
|
|
# Zamiast tego, powinno się używać metody "each" i podawać jej blok.
|
|
# Blok jest kawałkiem kodu, który możesz podać metodzie podobnej do "each".
|
|
# Jest analogiczny do wyrażeń lambda, funkcji anonimowych lub zamknięć w innych
|
|
# językach programowania.
|
|
#
|
|
# Metoda "each" danego zakresu, wykonuje blok dla każdego elementu w zakresie.
|
|
# Do bloku zostaje przekazany licznik jako parametr.
|
|
# Wykonanie metody "each" z przekazaniem bloku wygląda następująco:
|
|
|
|
(1..5).each do |licznik|
|
|
puts "powtórzenie #{licznik}"
|
|
end
|
|
#=> powtórzenie 1
|
|
#=> powtórzenie 2
|
|
#=> powtórzenie 3
|
|
#=> powtórzenie 4
|
|
#=> powtórzenie 5
|
|
|
|
# Możesz również otoczyć blok nawiasami klamrowymi:
|
|
(1..5).each { |licznik| puts "powtórzenie #{licznik}" }
|
|
|
|
# Zawartość struktur danych również może być powtarzana używając each.
|
|
tablica.each do |element|
|
|
puts "#{element} jest częścią tablicy"
|
|
end
|
|
hasz.each do |klucz, wartosc|
|
|
puts "#{klucz} jest #{wartosc}"
|
|
end
|
|
|
|
# Jeśli nadal potrzebujesz indeksum, możesz użyć "each_with_index" i zdefiniować
|
|
# zmienną odpowiadającą indeksowi
|
|
tablica.each_with_index do |element, indeks|
|
|
puts "#{element} jest numerem #{indeks} w tablicy"
|
|
end
|
|
|
|
licznik = 1
|
|
while licznik <= 5 do
|
|
puts "powtórzenie #{licznik}"
|
|
licznik += 1
|
|
end
|
|
#=> powtórzenie 1
|
|
#=> powtórzenie 2
|
|
#=> powtórzenie 3
|
|
#=> powtórzenie 4
|
|
#=> powtórzenie 5
|
|
|
|
# W Ruby istnieje dużo pomocnych funkcji wykonujących pętle,
|
|
# na przykład "map", "reduce", "inject" i wiele innych. Map,
|
|
# w każdym wywołaniu, pobiera tablicę, na której wykonuję pętlę,
|
|
# wykonuje kod zapisany za pomocą bloku i zwraca całkowicie nową tablicę.
|
|
tablica = [1,2,3,4,5]
|
|
podwojone = tablica.map do |element|
|
|
element * 2
|
|
end
|
|
puts podwojona
|
|
#=> [2,4,6,8,10]
|
|
puts tablica
|
|
#=> [1,2,3,4,5]
|
|
|
|
ocena = 2
|
|
|
|
case ocena
|
|
when 1
|
|
puts 'Dobra robota, masz wolne'
|
|
when 2
|
|
puts 'Następnym razem będziesz miał więcej szczęścia'
|
|
when 3
|
|
puts 'Możesz to zrobić lepiej'
|
|
when 4
|
|
puts 'Przebrnąłeś'
|
|
when 5
|
|
puts 'Oblałeś!'
|
|
else
|
|
puts 'Inny system oceniania?'
|
|
end
|
|
#=> "Następnym razem będziesz miał więcej szczęścia"
|
|
|
|
# case może również użwać zakresów
|
|
ocena = 82
|
|
case ocena
|
|
when 90..100
|
|
puts 'Hurra!'
|
|
when 80...90
|
|
puts 'Dobra robota'
|
|
else
|
|
puts 'Oblałeś!'
|
|
end
|
|
#=> "Dobra robota"
|
|
|
|
# obsługa błędów:
|
|
begin
|
|
# kod, który może wywołać wyjątek
|
|
raise NoMemoryError, 'Zabrakło pamięci.'
|
|
rescue NoMemoryError => zmienna_wyjatku
|
|
puts 'Został wywołany NoMemoryError', zmienna_wyjatku
|
|
rescue RuntimeError => inna_zmienna_wyjatku
|
|
puts 'Teraz został wywołany RuntimeError'
|
|
else
|
|
puts 'To zostanie uruchomione, jeśli nie wystąpi żaden wyjątek'
|
|
ensure
|
|
puts 'Ten kod wykona się zawsze'
|
|
end
|
|
|
|
# Funkcje
|
|
|
|
def podwojenie(x)
|
|
x * 2
|
|
end
|
|
|
|
# Funkcje (i wszystkie bloki) zawsze zwracają wartość ostatniego wyrażenia
|
|
podwojenie(2) #=> 4
|
|
|
|
# Okrągłe nawiady są opcjonalne, gdy wynik jest jednoznaczny
|
|
podwojenie 3 #=> 6
|
|
|
|
podwojenie podwojenie 3 #=> 12
|
|
|
|
def suma(x, y)
|
|
x + y
|
|
end
|
|
|
|
# Argumenty metod są oddzielone przecinkami
|
|
suma 3, 4 #=> 7
|
|
|
|
suma suma(3, 4), 5 #=> 12
|
|
|
|
# yield
|
|
# Wszystkie metody mają ukryty, opcjonalny parametr bloku,
|
|
# który może być wykonany używając słowa kluczowego 'yield'
|
|
|
|
def otoczenie
|
|
puts '{'
|
|
yield
|
|
puts '}'
|
|
end
|
|
|
|
otoczenie { puts 'hej świecie' }
|
|
|
|
# {
|
|
# hej świecie
|
|
# }
|
|
|
|
|
|
# Możesz przekazać blok do funkcji
|
|
# "&" oznacza referencję to przekazanego bloku
|
|
def goscie(&blok)
|
|
blok.call 'jakis_argument'
|
|
end
|
|
|
|
# Możesz przekazać listę argumentów, które będę przekonwertowane na tablicę
|
|
# Do tego służy operator ("*")
|
|
def goscie(*tablica)
|
|
tablica.each { |gosc| puts gosc }
|
|
end
|
|
|
|
# Definiowanie klas używając słowa kluczowego class
|
|
class Czlowiek
|
|
|
|
# Zmienna klasowa. Jest współdzielona przez wszystkie instancje tej klasy.
|
|
@@gatunek = 'H. sapiens'
|
|
|
|
# Podstawowe inicjalizowanie
|
|
def initialize(imie, wiek = 0)
|
|
# Przypisanie argumentu do zmiennej danej instancji o nazwie "imie"
|
|
@imie = imie
|
|
# Jeśli nie podano wieku, zostanie użyta domyślna wartość z listy argumentów.
|
|
@wiek = wiek
|
|
end
|
|
|
|
# Podstawowa metoda przypisująca wartość
|
|
def imie=(imie)
|
|
@imie = imie
|
|
end
|
|
|
|
# Podstawowa metoda pobierająca wartość
|
|
def imie
|
|
@imie
|
|
end
|
|
|
|
# Powyższa funkcjonalność może być zastąpiona używając metody attr_accessor w taki sposób
|
|
attr_accessor :imie
|
|
|
|
# Metody przypisujące/pobierające mogą być stworzone indywidualnie
|
|
attr_reader :imie
|
|
attr_writer :imie
|
|
|
|
# Metody klasowe używają self aby odróżnić się od metody instancji.
|
|
# To może być wywołane na klasie, nie na instancji.
|
|
def self.powiedz(wiadomosc)
|
|
puts wiadomosc
|
|
end
|
|
|
|
def gatunek
|
|
@@gatunek
|
|
end
|
|
end
|
|
|
|
|
|
# Tworzenie instancji klasy
|
|
jim = Czlowiek.new('Jim Halpert')
|
|
|
|
dwight = Czlowiek.new('Dwight K. Schrute')
|
|
|
|
# Wywołajmy parę metod
|
|
jim.gatunek #=> "H. sapiens"
|
|
jim.imie #=> "Jim Halpert"
|
|
jim.imie = "Jim Halpert II" #=> "Jim Halpert II"
|
|
jim.imie #=> "Jim Halpert II"
|
|
dwight.gatunek #=> "H. sapiens"
|
|
dwight.imie #=> "Dwight K. Schrute"
|
|
|
|
# Wywołanie metody klasowej
|
|
Czlowiek.powiedz('Cześć') #=> "Cześć"
|
|
|
|
# Zasięg zmiennej jest definiowany poprzez jej nazwę.
|
|
# Zmienne, które zaczynają się na $ mają zasięg globalny
|
|
$zmienna = "Jestem zmienną globalną"
|
|
defined? $zmienna #=> "global-variable"
|
|
|
|
# Zmienne zczynające się na @ mają zasięg danej instancji
|
|
@zmienna = "Jestem zmienną instancji"
|
|
defined? @zmienna #=> "instance-variable"
|
|
|
|
# Zmienne, które zaczynają się na @@ mają zasięg danej klasy
|
|
@@zmienna = "Jestem zmienną klasową"
|
|
defined? @@zmienna #=> "class variable"
|
|
|
|
# Zmienne, które zaczynają się na dużą literę, są stałymi
|
|
Zmienna = "Jestem stałą"
|
|
defined? Zmienna #=> "constant"
|
|
|
|
# Klasa jest również obiektem w ruby. Może więc mieć zmienne instancji.
|
|
# Zmienna klasowa może być współdzielona między klasą i jej potomstwem.
|
|
|
|
# podstawowa klasa
|
|
class Czlowiek
|
|
@@cokolwiek = 0
|
|
|
|
def self.cokolwiek
|
|
@@cokolwiek
|
|
end
|
|
|
|
def self.cokolwiek=(wartosc)
|
|
@@cokolwiek = wartosc
|
|
end
|
|
end
|
|
|
|
# klasa pochodna
|
|
class Pracownik < Czlowiek
|
|
end
|
|
|
|
Czlowiek.cokolwiek # 0
|
|
Pracownik.cokolwiek # 0
|
|
|
|
Czlowiek.cokolwiek = 2 # 2
|
|
Pracownik.cokolwiek # 2
|
|
|
|
# Zmienna instancji danej klasy nie jest współdzielona przez jej potomstwo.
|
|
|
|
class Czlowiek
|
|
@cos = 0
|
|
|
|
def self.cos
|
|
@cos
|
|
end
|
|
|
|
def self.cos=(wartosc)
|
|
@cos = wartosc
|
|
end
|
|
end
|
|
|
|
class Doktor < Czlowiek
|
|
end
|
|
|
|
Czlowiek.cos # 0
|
|
Doktor.cos # nil
|
|
|
|
module PrzykladowyModul
|
|
def cokolwiek
|
|
'cokolwiek'
|
|
end
|
|
end
|
|
|
|
# Włączanie modułów łączy ich metody z metodami instancji klasy
|
|
# Rozszerzanie modułów łączy ich metody z metodami klasy
|
|
|
|
class Osoba
|
|
include PrzykladowyModul
|
|
end
|
|
|
|
class Ksiazka
|
|
extend PrzykladowyModul
|
|
end
|
|
|
|
Osoba.cokolwiek # => NoMethodError: undefined method `cokolwiek' for Osoba:Class
|
|
Osoba.new.cokolwiek # => 'cokolwiek'
|
|
Ksiazka.cokolwiek # => 'cokolwiek'
|
|
Ksiazka.new.cokolwiek # => NoMethodError: undefined method `cokolwiek'
|
|
|
|
# Gdy włączamy lub rozszerzamy muduły, wykonywane są tzw. wywołania zwrotne
|
|
|
|
module PrzykladowyModul
|
|
def self.included(baza)
|
|
baza.extend(MotodyKlasowe)
|
|
baza.send(:include, MetodyInstancji)
|
|
end
|
|
|
|
module MotodyKlasowe
|
|
def cos
|
|
'cos'
|
|
end
|
|
end
|
|
|
|
module MetodyInstancji
|
|
def xyz
|
|
'xyz'
|
|
end
|
|
end
|
|
end
|
|
|
|
class Cokolwiek
|
|
include PrzykladowyModul
|
|
end
|
|
|
|
Cokolwiek.cos # => 'cos'
|
|
Cokolwiek.xyz # => NoMethodError: undefined method `xyz'
|
|
Cokolwiek.new.cos # => NoMethodError: undefined method `cos'
|
|
Cokolwiek.new.xyz # => 'qux'
|
|
```
|
|
|
|
## Dodatkowe źródła
|
|
### Polskie
|
|
|
|
- [Dokumentacja](https://www.ruby-lang.org/pl/documentation/quickstart/)
|
|
|
|
### Angielskie
|
|
|
|
- [Official Documentation](http://www.ruby-doc.org/core-2.1.1/)
|
|
- [Ruby from other languages](https://www.ruby-lang.org/en/documentation/ruby-from-other-languages/)
|
|
- [Programming Ruby](http://www.amazon.com/Programming-Ruby-1-9-2-0-Programmers/dp/1937785491/) - An older [free edition](http://ruby-doc.com/docs/ProgrammingRuby/) is available online.
|
|
- [Ruby Style Guide](https://github.com/bbatsov/ruby-style-guide) - A community-driven Ruby coding style guide.
|