Condividi questa pagina

Impara X in Y minuti

Dove X=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


Hai un suggerimento? Oppure una correzione? Apri un issue sul GitHub, o fa un pull request da solo!

In origine contribuita da David Underwood, e modificata da 15 contributore(i).