mirror of
https://github.com/adambard/learnxinyminutes-docs.git
synced 2025-01-15 05:35:59 +00:00
651 lines
16 KiB
Markdown
651 lines
16 KiB
Markdown
---
|
|
filename: learnruby-it.rb
|
|
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"]
|
|
- ["Jake Faris", "https://github.com/farisj"]
|
|
- ["Corey Ward", "https://github.com/coreyward"]
|
|
translators:
|
|
- ["abonte", "https://github.com/abonte"]
|
|
---
|
|
|
|
```ruby
|
|
# Questo è un commento
|
|
|
|
# In Ruby, (quasi) tutto è un oggetto.
|
|
# Questo include i numeri...
|
|
3.class #=> Integer
|
|
|
|
# ...stringhe...
|
|
"Hello".class #=> String
|
|
|
|
# ...e anche i metodi!
|
|
"Hello".method(:class).class #=> Method
|
|
|
|
# Qualche operazione aritmetica di base
|
|
1 + 1 #=> 2
|
|
8 - 1 #=> 7
|
|
10 * 2 #=> 20
|
|
35 / 5 #=> 7
|
|
2 ** 5 #=> 32
|
|
5 % 3 #=> 2
|
|
|
|
# Bitwise operators
|
|
3 & 5 #=> 1
|
|
3 | 5 #=> 7
|
|
3 ^ 5 #=> 6
|
|
|
|
# L'aritmetica è solo zucchero sintattico
|
|
# per chiamare il metodo di un oggetto
|
|
1.+(3) #=> 4
|
|
10.* 5 #=> 50
|
|
100.methods.include?(:/) #=> true
|
|
|
|
# I valori speciali sono oggetti
|
|
nil # equivalente a null in altri linguaggi
|
|
true # vero
|
|
false # falso
|
|
|
|
nil.class #=> NilClass
|
|
true.class #=> TrueClass
|
|
false.class #=> FalseClass
|
|
|
|
# Uguaglianza
|
|
1 == 1 #=> true
|
|
2 == 1 #=> false
|
|
|
|
# Disuguaglianza
|
|
1 != 1 #=> false
|
|
2 != 1 #=> true
|
|
|
|
# nil è l'unico valore, oltre a false, che è considerato 'falso'
|
|
!!nil #=> false
|
|
!!false #=> false
|
|
!!0 #=> true
|
|
!!"" #=> true
|
|
|
|
# Altri confronti
|
|
1 < 10 #=> true
|
|
1 > 10 #=> false
|
|
2 <= 2 #=> true
|
|
2 >= 2 #=> true
|
|
|
|
# Operatori di confronto combinati (ritorna '1' quando il primo argomento è più
|
|
# grande, '-1' quando il secondo argomento è più grande, altrimenti '0')
|
|
1 <=> 10 #=> -1
|
|
10 <=> 1 #=> 1
|
|
1 <=> 1 #=> 0
|
|
|
|
# Operatori logici
|
|
true && false #=> false
|
|
true || false #=> true
|
|
|
|
# Ci sono versioni alternative degli operatori logici con meno precedenza.
|
|
# Sono usati come costrutti per il controllo di flusso per concatenare
|
|
# insieme statement finché uno di essi ritorna true o false.
|
|
|
|
# `do_something_else` chiamato solo se `do_something` ha successo.
|
|
do_something() and do_something_else()
|
|
# `log_error` è chiamato solo se `do_something` fallisce.
|
|
do_something() or log_error()
|
|
|
|
# Interpolazione di stringhe
|
|
|
|
placeholder = 'usare l\'interpolazione di stringhe'
|
|
"Per #{placeholder} si usano stringhe con i doppi apici"
|
|
#=> "Per usare l'interpolazione di stringhe si usano stringhe con i doppi apici"
|
|
|
|
# E' possibile combinare le stringhe usando `+`, ma non con gli altri tipi
|
|
'hello ' + 'world' #=> "hello world"
|
|
'hello ' + 3 #=> TypeError: can't convert Fixnum into String
|
|
'hello ' + 3.to_s #=> "hello 3"
|
|
"hello #{3}" #=> "hello 3"
|
|
|
|
# ...oppure combinare stringhe e operatori
|
|
'ciao ' * 3 #=> "ciao ciao ciao "
|
|
|
|
# ...oppure aggiungere alla stringa
|
|
'ciao' << ' mondo' #=> "ciao mondo"
|
|
|
|
# Per stampare a schermo e andare a capo
|
|
puts "Sto stampando!"
|
|
#=> Sto stampando!
|
|
#=> nil
|
|
|
|
# Per stampare a schermo senza andare a capo
|
|
print "Sto stampando!"
|
|
#=> Sto stampando! => nil
|
|
|
|
# Variabili
|
|
x = 25 #=> 25
|
|
x #=> 25
|
|
|
|
# Notare che l'assegnamento ritorna il valore assegnato.
|
|
# Questo significa che è possibile effettuare assegnamenti multipli:
|
|
x = y = 10 #=> 10
|
|
x #=> 10
|
|
y #=> 10
|
|
|
|
# Per convenzione si usa lo snake_case per i nomi delle variabili
|
|
snake_case = true
|
|
|
|
# Usare nomi delle variabili descrittivi
|
|
path_to_project_root = '/buon/nome/'
|
|
m = '/nome/scadente/'
|
|
|
|
# I simboli sono immutabili, costanti riusabili rappresentati internamente da
|
|
# un valore intero. Sono spesso usati al posto delle stringhe per comunicare
|
|
# specifici e significativi valori.
|
|
|
|
:pendente.class #=> Symbol
|
|
|
|
stato = :pendente
|
|
|
|
stato == :pendente #=> true
|
|
|
|
stato == 'pendente' #=> false
|
|
|
|
stato == :approvato #=> false
|
|
|
|
# Le stringhe possono essere convertite in simboli e viceversa:
|
|
status.to_s #=> "pendente"
|
|
"argon".to_sym #=> :argon
|
|
|
|
# Arrays
|
|
|
|
# Questo è un array
|
|
array = [1, 2, 3, 4, 5] #=> [1, 2, 3, 4, 5]
|
|
|
|
# Gli array possono contenere diversi tipi di elementi
|
|
[1, 'hello', false] #=> [1, "hello", false]
|
|
|
|
# Gli array possono essere indicizzati
|
|
# Dall'inizio...
|
|
array[0] #=> 1
|
|
array.first #=> 1
|
|
array[12] #=> nil
|
|
|
|
|
|
# ...o dalla fine...
|
|
array[-1] #=> 5
|
|
array.last #=> 5
|
|
|
|
# With a start index and length
|
|
# ...o con un indice di inzio e la lunghezza...
|
|
array[2, 3] #=> [3, 4, 5]
|
|
|
|
# ...oppure con un intervallo.
|
|
array[1..3] #=> [2, 3, 4]
|
|
|
|
# Invertire l'ordine degli elementi di un array
|
|
a = [1,2,3]
|
|
a.reverse! #=> [3,2,1]
|
|
|
|
# Come per l'aritmetica, l'accesso tramite [var]
|
|
# è solo zucchero sintattico
|
|
# per chiamare il metodo '[]'' di un oggetto
|
|
array.[] 0 #=> 1
|
|
array.[] 12 #=> nil
|
|
|
|
# Si può aggiungere un elemento all'array così
|
|
array << 6 #=> [1, 2, 3, 4, 5, 6]
|
|
# oppure così
|
|
array.push(6) #=> [1, 2, 3, 4, 5, 6]
|
|
|
|
# Controllare se un elemento esiste in un array
|
|
array.include?(1) #=> true
|
|
|
|
# Hash è un dizionario con coppie di chiave e valore
|
|
# Un hash è denotato da parentesi graffe:
|
|
hash = { 'colore' => 'verde', 'numero' => 5 }
|
|
|
|
hash.keys #=> ['colore', 'numero']
|
|
|
|
# E' possibile accedere all'hash tramite chiave:
|
|
hash['colore'] #=> 'verde'
|
|
hash['numero'] #=> 5
|
|
|
|
# Accedere all'hash con una chiave che non esiste ritorna nil:
|
|
hash['nothing here'] #=> nil
|
|
|
|
# Quando si usano simboli come chiavi di un hash, si possono utilizzare
|
|
# queste sintassi:
|
|
|
|
hash = { :defcon => 3, :action => true }
|
|
hash.keys #=> [:defcon, :action]
|
|
# oppure
|
|
hash = { defcon: 3, action: true }
|
|
hash.keys #=> [:defcon, :action]
|
|
|
|
# Controllare l'esistenza di una chiave o di un valore in un hash
|
|
new_hash.key?(:defcon) #=> true
|
|
new_hash.value?(3) #=> true
|
|
|
|
# Suggerimento: sia gli array che gli hash sono enumerabili!
|
|
# Entrambi possiedono metodi utili come each, map, count e altri.
|
|
|
|
# Strutture di controllo
|
|
|
|
#Condizionali
|
|
if true
|
|
'if statement'
|
|
elsif false
|
|
'else if, opzionale'
|
|
else
|
|
'else, opzionale'
|
|
end
|
|
|
|
#Cicli
|
|
# In Ruby, i tradizionali cicli `for` non sono molto comuni. Questi semplici
|
|
# cicli, invece, sono implementati con un enumerable, usando `each`:
|
|
(1..5).each do |contatore|
|
|
puts "iterazione #{contatore}"
|
|
end
|
|
|
|
# Esso è equivalente a questo ciclo, il quale è inusuale da vedere in Ruby:
|
|
for contatore in 1..5
|
|
puts "iterazione #{contatore}"
|
|
end
|
|
|
|
# Il costrutto `do |variable| ... end` è chiamato 'blocco'. I blocchi
|
|
# sono simili alle lambda, funzioni anonime o closure che si trovano in altri
|
|
# linguaggi di programmazione. Essi possono essere passati come oggetti,
|
|
# chiamati o allegati come metodi.
|
|
#
|
|
# Il metodo 'each' di un intervallo (range) esegue il blocco una volta
|
|
# per ogni elemento dell'intervallo.
|
|
# Al blocco è passato un contatore come parametro.
|
|
|
|
# E' possibile inglobare il blocco fra le parentesi graffe
|
|
(1..5).each { |contatore| puts "iterazione #{contatore}" }
|
|
|
|
# Il contenuto delle strutture dati può essere iterato usando "each".
|
|
array.each do |elemento|
|
|
puts "#{elemento} è parte dell'array"
|
|
end
|
|
hash.each do |chiave, valore|
|
|
puts "#{chiave} è #{valore}"
|
|
end
|
|
|
|
# If you still need an index you can use 'each_with_index' and define an index
|
|
# variable
|
|
# Se comunque si vuole un indice, si può usare "each_with_index" e definire
|
|
# una variabile che contiene l'indice
|
|
array.each_with_index do |elemento, indice|
|
|
puts "#{elemento} è il numero #{index} nell'array"
|
|
end
|
|
|
|
contatore = 1
|
|
while contatore <= 5 do
|
|
puts "iterazione #{contatore}"
|
|
contatore += 1
|
|
end
|
|
#=> iterazione 1
|
|
#=> iterazione 2
|
|
#=> iterazione 3
|
|
#=> iterazione 4
|
|
#=> iterazione 5
|
|
|
|
# Esistono in Ruby ulteriori funzioni per fare i cicli,
|
|
# come per esempio 'map', 'reduce', 'inject' e altri.
|
|
# Nel caso di 'map', esso prende l'array sul quale si sta iterando, esegue
|
|
# le istruzioni definite nel blocco, e ritorna un array completamente nuovo.
|
|
array = [1,2,3,4,5]
|
|
doubled = array.map do |elemento|
|
|
elemento * 2
|
|
end
|
|
puts doubled
|
|
#=> [2,4,6,8,10]
|
|
puts array
|
|
#=> [1,2,3,4,5]
|
|
|
|
# Costrutto "case"
|
|
grade = 'B'
|
|
|
|
case grade
|
|
when 'A'
|
|
puts 'Way to go kiddo'
|
|
when 'B'
|
|
puts 'Better luck next time'
|
|
when 'C'
|
|
puts 'You can do better'
|
|
when 'D'
|
|
puts 'Scraping through'
|
|
when 'F'
|
|
puts 'You failed!'
|
|
else
|
|
puts 'Alternative grading system, eh?'
|
|
end
|
|
#=> "Better luck next time"
|
|
|
|
# 'case' può usare anche gli intervalli
|
|
grade = 82
|
|
case grade
|
|
when 90..100
|
|
puts 'Hooray!'
|
|
when 80...90
|
|
puts 'OK job'
|
|
else
|
|
puts 'You failed!'
|
|
end
|
|
#=> "OK job"
|
|
|
|
# Gestione delle eccezioni
|
|
begin
|
|
# codice che può sollevare un eccezione
|
|
raise NoMemoryError, 'Esaurita la memoria.'
|
|
rescue NoMemoryError => exception_variable
|
|
puts 'NoMemoryError è stato sollevato.', exception_variable
|
|
rescue RuntimeError => other_exception_variable
|
|
puts 'RuntimeError è stato sollvato.'
|
|
else
|
|
puts 'Questo viene eseguito se nessuna eccezione è stata sollevata.'
|
|
ensure
|
|
puts 'Questo codice viene sempre eseguito a prescindere.'
|
|
end
|
|
|
|
# Metodi
|
|
|
|
def double(x)
|
|
x * 2
|
|
end
|
|
|
|
# Metodi (e blocchi) ritornano implicitamente il valore dell'ultima istruzione
|
|
double(2) #=> 4
|
|
|
|
# Le parentesi sono opzionali dove l'interpolazione è inequivocabile
|
|
double 3 #=> 6
|
|
|
|
double double 3 #=> 12
|
|
|
|
def sum(x, y)
|
|
x + y
|
|
end
|
|
|
|
# Gli argomenit dei metodi sono separati dalla virgola
|
|
sum 3, 4 #=> 7
|
|
|
|
sum sum(3, 4), 5 #=> 12
|
|
|
|
# yield
|
|
# Tutti i metodi hanno un implicito e opzionale parametro del blocco.
|
|
# Esso può essere chiamato con la parola chiave 'yield'.
|
|
|
|
def surround
|
|
puts '{'
|
|
yield
|
|
puts '}'
|
|
end
|
|
|
|
surround { puts 'hello world' }
|
|
|
|
# {
|
|
# hello world
|
|
# }
|
|
|
|
# I blocchi possono essere convertiti in 'proc', il quale racchiude il blocco
|
|
# e gli permette di essere passato ad un altro metodo, legato ad uno scope
|
|
# differente o modificato. Questo è molto comune nella lista parametri del
|
|
# metodo, dove è frequente vedere il parametro '&block' in coda. Esso accetta
|
|
# il blocco, se ne è stato passato uno, e lo converte in un 'Proc'.
|
|
# Qui la denominazione è una convenzione; funzionerebbe anche con '&ananas'.
|
|
def guests(&block)
|
|
block.class #=> Proc
|
|
block.call(4)
|
|
end
|
|
|
|
# Il metodo 'call' del Proc è simile allo 'yield' quando è presente un blocco.
|
|
# Gli argomenti passati a 'call' sono inoltrati al blocco come argomenti:
|
|
|
|
guests { |n| "You have #{n} guests." }
|
|
# => "You have 4 guests."
|
|
|
|
# L'operatore splat ("*") converte una lista di argomenti in un array
|
|
def guests(*array)
|
|
array.each { |guest| puts guest }
|
|
end
|
|
|
|
# Destrutturazione
|
|
|
|
# Ruby destruttura automaticamente gli array in assegnamento
|
|
# a variabili multiple:
|
|
a, b, c = [1, 2, 3]
|
|
a #=> 1
|
|
b #=> 2
|
|
c #=> 3
|
|
|
|
# In alcuni casi si usa l'operatore splat ("*") per destrutturare
|
|
# un array in una lista.
|
|
classifica_concorrenti = ["John", "Sally", "Dingus", "Moe", "Marcy"]
|
|
|
|
def migliore(primo, secondo, terzo)
|
|
puts "I vincitori sono #{primo}, #{secondo}, e #{terzo}."
|
|
end
|
|
|
|
migliore *classifica_concorrenti.first(3)
|
|
#=> I vincitori sono John, Sally, e Dingus.
|
|
|
|
# The splat operator can also be used in parameters:
|
|
def migliore(primo, secondo, terzo, *altri)
|
|
puts "I vincitori sono #{primo}, #{secondo}, e #{terzo}."
|
|
puts "C'erano altri #{altri.count} partecipanti."
|
|
end
|
|
|
|
migliore *classifica_concorrenti
|
|
#=> I vincitori sono John, Sally, e Dingus.
|
|
#=> C'erano altri 2 partecipanti.
|
|
|
|
# Per convenzione, tutti i metodi che ritornano un booleano terminano
|
|
# con un punto interrogativo
|
|
5.even? #=> false
|
|
5.odd? #=> true
|
|
|
|
# Per convenzione, se il nome di un metodo termina con un punto esclamativo,
|
|
# esso esegue qualcosa di distruttivo. Molti metodi hanno una versione con '!'
|
|
# per effettuare una modifiche, e una versione senza '!' che ritorna
|
|
# una versione modificata.
|
|
nome_azienda = "Dunder Mifflin"
|
|
nome_azienda.upcase #=> "DUNDER MIFFLIN"
|
|
nome_azienda #=> "Dunder Mifflin"
|
|
# Questa volta modifichiamo nome_azienda
|
|
nome_azienda.upcase! #=> "DUNDER MIFFLIN"
|
|
nome_azienda #=> "DUNDER MIFFLIN"
|
|
|
|
# Classi
|
|
|
|
# Definire una classe con la parola chiave class
|
|
class Umano
|
|
|
|
# Una variabile di classe. E' condivisa da tutte le istance di questa classe.
|
|
@@specie = 'H. sapiens'
|
|
|
|
# Inizializzatore di base
|
|
def initialize(nome, eta = 0)
|
|
# Assegna il valore dell'argomento alla variabile dell'istanza "nome"
|
|
@nome = nome
|
|
# Se l'età non è fornita, verrà assegnato il valore di default indicato
|
|
# nella lista degli argomenti
|
|
@eta = eta
|
|
end
|
|
|
|
# Metodo setter di base
|
|
def nome=(nome)
|
|
@nome = nome
|
|
end
|
|
|
|
# Metodo getter di base
|
|
def nome
|
|
@nome
|
|
end
|
|
|
|
# Le funzionalità di cui sopra posso essere incapsulate usando
|
|
# il metodo attr_accessor come segue
|
|
attr_accessor :nome
|
|
|
|
# Getter/setter possono anche essere creati individualmente
|
|
attr_reader :nome
|
|
attr_writer :nome
|
|
|
|
# Un metodo della classe usa 'self' per distinguersi dai metodi dell'istanza.
|
|
# Può essere richimato solo dalla classe, non dall'istanza.
|
|
def self.say(msg)
|
|
puts msg
|
|
end
|
|
|
|
def specie
|
|
@@specie
|
|
end
|
|
end
|
|
|
|
|
|
# Instanziare una classe
|
|
jim = Umano.new('Jim Halpert')
|
|
|
|
dwight = Umano.new('Dwight K. Schrute')
|
|
|
|
# Chiamiamo qualche metodo
|
|
jim.specie #=> "H. sapiens"
|
|
jim.nome #=> "Jim Halpert"
|
|
jim.nome = "Jim Halpert II" #=> "Jim Halpert II"
|
|
jim.nome #=> "Jim Halpert II"
|
|
dwight.specie #=> "H. sapiens"
|
|
dwight.nome #=> "Dwight K. Schrute"
|
|
|
|
# Chiamare un metodo della classe
|
|
Umano.say('Ciao') #=> "Ciao"
|
|
|
|
# La visibilità della variabile (variable's scope) è determinata dal modo
|
|
# in cui le viene assegnato il nome.
|
|
# Variabili che iniziano con $ hanno uno scope globale
|
|
$var = "Sono una variabile globale"
|
|
defined? $var #=> "global-variable"
|
|
|
|
# Variabili che inziano con @ hanno a livello dell'istanza
|
|
@var = "Sono una variabile dell'istanza"
|
|
defined? @var #=> "instance-variable"
|
|
|
|
# Variabili che iniziano con @@ hanno una visibilità a livello della classe
|
|
@@var = "Sono una variabile della classe"
|
|
defined? @@var #=> "class variable"
|
|
|
|
# Variabili che iniziano con una lettera maiuscola sono costanti
|
|
Var = "Sono una costante"
|
|
defined? Var #=> "constant"
|
|
|
|
# Anche una classe è un oggetto in ruby. Quindi la classe può avere
|
|
# una variabile dell'istanza. Le variabili della classe sono condivise
|
|
# fra la classe e tutti i suoi discendenti.
|
|
|
|
# Classe base
|
|
class Umano
|
|
@@foo = 0
|
|
|
|
def self.foo
|
|
@@foo
|
|
end
|
|
|
|
def self.foo=(value)
|
|
@@foo = value
|
|
end
|
|
end
|
|
|
|
# Classe derivata
|
|
class Lavoratore < Umano
|
|
end
|
|
|
|
Umano.foo #=> 0
|
|
Lavoratore.foo #=> 0
|
|
|
|
Umano.foo = 2 #=> 2
|
|
Lavoratore.foo #=> 2
|
|
|
|
# La variabile dell'istanza della classe non è condivisa dai discendenti.
|
|
|
|
class Umano
|
|
@bar = 0
|
|
|
|
def self.bar
|
|
@bar
|
|
end
|
|
|
|
def self.bar=(value)
|
|
@bar = value
|
|
end
|
|
end
|
|
|
|
class Dottore < Umano
|
|
end
|
|
|
|
Umano.bar #=> 0
|
|
Dottore.bar #=> nil
|
|
|
|
module EsempioModulo
|
|
def foo
|
|
'foo'
|
|
end
|
|
end
|
|
|
|
# Includere moduli vincola i suoi metodi all'istanza della classe.
|
|
# Estendere moduli vincola i suoi metodi alla classe stessa.
|
|
class Persona
|
|
include EsempioModulo
|
|
end
|
|
|
|
class Libro
|
|
extend EsempioModulo
|
|
end
|
|
|
|
Persona.foo #=> NoMethodError: undefined method `foo' for Person:Class
|
|
Persona.new.foo #=> 'foo'
|
|
Libro.foo #=> 'foo'
|
|
Libro.new.foo #=> NoMethodError: undefined method `foo'
|
|
|
|
# Callbacks sono eseguiti quand si include o estende un modulo
|
|
module ConcernExample
|
|
def self.included(base)
|
|
base.extend(ClassMethods)
|
|
base.send(:include, InstanceMethods)
|
|
end
|
|
|
|
module ClassMethods
|
|
def bar
|
|
'bar'
|
|
end
|
|
end
|
|
|
|
module InstanceMethods
|
|
def qux
|
|
'qux'
|
|
end
|
|
end
|
|
end
|
|
|
|
class Something
|
|
include ConcernExample
|
|
end
|
|
|
|
Something.bar #=> 'bar'
|
|
Something.qux #=> NoMethodError: undefined method `qux'
|
|
Something.new.bar #=> NoMethodError: undefined method `bar'
|
|
Something.new.qux #=> 'qux'
|
|
```
|
|
|
|
## Ulteriori risorse
|
|
|
|
- [An Interactive Tutorial for Ruby](https://rubymonk.com/) - Imparare Ruby attraverso una serie di tutorial interattivi.
|
|
- [Official Documentation](http://ruby-doc.org/core)
|
|
- [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/) - Una passata [edizione libera](http://ruby-doc.com/docs/ProgrammingRuby/) è disponibile online.
|
|
- [Ruby Style Guide](https://github.com/bbatsov/ruby-style-guide) - A community-driven Ruby coding style guide.
|
|
- [Try Ruby](https://try.ruby-lang.org/) - Imparare le basi del linguaggio di programmazion Ruby, interattivamente nel browser.
|