Scala suporta o conceito de classes case. Classes case são classes regulares que são:
- Imutáveis por padrão
- Decompostas por meio de correspondência de padrões
- Comparadas por igualdade estrutural ao invés de referência
- Sucintas para instanciar e operar
Aqui temos um exemplo de hierarquia de tipos para Notification que consiste em uma super classe abstrata Notification
e três tipos concretos de notificação implementados com classes case Email
, SMS
, e VoiceRecording
.
abstract class Notification
case class Email(sourceEmail: String, title: String, body: String) extends Notification
case class SMS(sourceNumber: String, message: String) extends Notification
case class VoiceRecording(contactName: String, link: String) extends Notification
Instânciar uma classe case é fácil: (Perceba que nós não precisamos da palavra-chave new
)
val emailDeJohn = Email("[email protected]", "Saudações do John!", "Olá Mundo")
Os parâmetros do construtor de uma classe case são tratados como valores públicos e podem ser acessados diretamente.
val titulo = emailDeJohn.title
println(titulo) // prints "Saudações do John!"
Com classes case, você não pode alterar seus campos diretamente. (ao menos que você declare var
antes de um campo, mas fazê-lo geralmente é desencorajado).
emailDeJohn.title = "Adeus do John!" // Erro the compilação. Não podemos atribuir outro valor para um campo que foi declarado como val, lembrando que todos os campos de classes case são val por padrão.
Ao invés disso, faça uma cópia utilizando o método copy
. Como descrito abaixo, então você poderá substituir alguns dos campos:
val emailEditado = emailDeJohn.copy(title = "Estou aprendendo Scala!", body = "É muito legal!")
println(emailDeJohn) // prints "Email([email protected],Saudações do John!,Hello World!)"
println(emailEditado) // prints "Email([email protected],Estou aprendendo Scala,É muito legal!)"
Para cada classe case em Scala o compilador gera um método equals
que implementa a igualdade estrutural e um método toString
. Por exemplo:
val primeiroSMS = SMS("12345", "Hello!")
val segundoSMS = SMS("12345", "Hello!")
if (primeiroSMS == segundoSMS) {
println("Somos iguais!")
}
println("SMS é: " + primeiroSMS)
Irá gerar como saída:
Somos iguais!
SMS é: SMS(12345, Hello!)
Com classes case, você pode utilizar correspondência de padrões para manipular seus dados. Aqui temos um exemplo de uma função que escreve como saída diferente mensagens dependendo do tipo de notificação recebida:
def mostrarNotificacao(notificacao: Notification): String = {
notificacao match {
case Email(email, title, _) =>
"Você recebeu um email de " + email + " com o título: " + title
case SMS(number, message) =>
"Você recebeu um SMS de" + number + "! Mensagem: " + message
case VoiceRecording(name, link) =>
"Você recebeu uma Mensagem de Voz de " + name + "! Clique no link para ouvir: " + link
}
}
val algumSMS = SMS("12345", "Você está aí?")
val algumaMsgVoz = VoiceRecording("Tom", "voicerecording.org/id/123")
println(mostrarNotificacao(algumSMS)) // Saída "Você recebeu um SMS de 12345! Mensagem: Você está aí?"
println(mostrarNotificacao(algumaMsgVoz)) // Saída "Você recebeu uma Mensagem de Voz de Tom! Clique no link para ouvir: voicerecording.org/id/123"
Aqui um exemplo mais elaborado utilizando a proteção if
. Com a proteção if
, o correspondência de padrão irá falhar se a condição de proteção retorna falso.
def mostrarNotificacaoEspecial(notificacao: Notification, emailEspecial: String, numeroEspecial: String): String = {
notificacao match {
case Email(email, _, _) if email == emailEspecial =>
"Você recebeu um email de alguém especial!"
case SMS(numero, _) if numero == numeroEspecial =>
"Você recebeu um SMS de alguém especial!"
case outro =>
mostrarNotificacao(outro) // Nada especial para mostrar, então delega para nossa função original mostrarNotificacao
}
}
val NumeroEspecial = "55555"
val EmailEspecial = "[email protected]"
val algumSMS = SMS("12345", "Você está aí?")
val algumaMsgVoz = VoiceRecording("Tom", "voicerecording.org/id/123")
val emailEspecial = Email("[email protected]", "Beber hoje a noite?", "Estou livre depois das 5!")
val smsEspecial = SMS("55555", "Estou aqui! Onde está você?")
println(mostrarNotificacaoEspecial(algumSMS, EmailEspecial, NumeroEspecial)) // Saída "Você recebeu um SMS de 12345! Mensagem: Você está aí?"
println(mostrarNotificacaoEspecial(algumaMsgVoz, EmailEspecial, NumeroEspecial)) // Saída "Você recebeu uma Mensagem de Voz de Tom! Clique no link para ouvir: voicerecording.org/id/123"
println(mostrarNotificacaoEspecial(smsEspecial, EmailEspecial, NumeroEspecial)) // Saída "Você recebeu um email de alguém especial!"
println(mostrarNotificacaoEspecial(smsEspecial, EmailEspecial, NumeroEspecial)) // Saída "Você recebeu um SMS de alguém especial!"
Ao programar em Scala, recomenda-se que você use classes case de forma pervasiva para modelar / agrupar dados, pois elas ajudam você a escrever código mais expressivo e passível de manutenção:
- Imutabilidade libera você de precisar acompanhar onde e quando as coisas são mutadas
- Comparação por valor permite comparar instâncias como se fossem valores primitivos - não há mais incerteza sobre se as instâncias de uma classe é comparada por valor ou referência
- Correspondência de padrões simplifica a lógica de ramificação, o que leva a menos bugs e códigos mais legíveis.
Contributors to this page:
Contents
- Introdução
- Basics
- Tipos Unificados
- Classes
- Parâmetro com Valor Padrão
- Parâmetros Nomeados
- Traits
- Tuplas
- Composição de Classes Mixin
- Funções de ordem superior
- Funções Aninhadas
- Currying
- Classes Case
- Correspondência de Padrões
- Objetos Singleton
- Padrões de Expressões Regulares
- Objetos Extratores
- For Comprehensions
- Classes Genéricas
- Variâncias
- Limitante Superior de Tipos
- Limitante Inferior de Tipos
- Classes Internas
- Tipos Abstratos
- Tipos Compostos
- Auto Referências Explicitamente Tipadas
- Parâmetros Implícitos
- Conversões Implícitas
- Métodos Polimórficos
- Inferência de Tipo Local
- Operadores
- By-name Parameters
- Anotações
- Packages and Imports
- Package Objects