Metaclasse

Em orientação a objetos, uma metaclasse é uma classe cujas instâncias também são classes e não objetos no sentido tradicional. Assim como classes definem o comportamento de certos objetos, metaclasses definem o comportamento de certas classes e suas instâncias.

Nem toda linguagem orientada a objeto suporta metaclasses. Entre as que suportam, a extensão de modificações que podem ser feitas nas classes varia. Cada linguagem possui seu próprio protocolo de metaobjeto, um conjunto de regras que definem como objetos, classes e metaclasses interagem.

Exemplo em Python

Em Python, a classe nativa type é uma metaclasse. Considere o exemplo abaixo:

class Carro(object):
    __slots__ = ['marca', 'modelo', 'ano', 'cor']

    def __init__(self, marca, modelo, ano, cor):
        self.marca = marca
        self.modelo = modelo
        self.ano = ano
        self.cor = cor

    @property 
    def descricao(self):
        """ Retorna uma descrição do carro. """
        return "%s %s %s %s" % (self.cor, self.ano, self.marca, self.modelo)

Em tempo de execução, Carro é um objeto type por herança. O código fonte da classe Carro mostrado acima não inclui detalhes como o tamanho em bytes dos objetos Carro, sua configuração na memória, como são alocados, que o método __init__ é invocado automaticamente cada vez que um objeto é criado, e assim por diante. Esses detalhes aparecem não somente quando um novo objeto Carro é criado, mas também cada vez que um atributo na classe é acessado. Em linguagens de programação sem metaclasses, esses detalhes são definidos pela especificação da linguagem e não podem ser modificados. Em Python, a metaclasse nativa type controla tais detalhes sobre o comportamento de Carro. Tais detalhes pode ser modificados usando-se outra metaclasse ao invés de type.

O exemplo acima contém código redundante relacionado aos quatro atributos marca, modelo, ano e cor. É possível eliminar parte da redundância usando uma metaclasse. Em Python, uma metaclasse é definida como uma subclasse de type.

 class TipoIniciaAtributo(type):
     def __call__(self, *args, **kwargs):
         """ Cria uma nova instância. """
 
         # Primeiro, cria um objeto da forma padrão.
         obj = type.__call__(self, *args)
 
         # Adicionalmente, configura-se atributos do novo objeto.
         for nome in kwargs:
             setattr(obj, nome, kwargs[nome])
 
         # Retorna o novo objeto.
         return obj

Essa metaclasse modifica a criação do objeto. Todos os outros aspectos do comportamento da classe e dos objetos ainda seguem o que foi definido por type.

Agora a classe Carro pode ser reescrita para usar essa metaclasse acima, o que é feito atribuindo a nova metaclasse em __metaclass__:

 class Carro(object):
     __metaclass__ = TipoIniciaAtributo
     __slots__ = ['marca', 'modelo', 'ano', 'cor']
 
     @property
     def descricao(self):
         """ Retorna uma descrição do carro. """
         return "%s %s %s %s" % (self.cor, self.ano, self.marca, self.modelo)

Objetos Carro podem então ser instanciados através de:

 carros = [
     Carro(marca='Toyota', modelo='Prius', ano=2005, cor='green'),
     Carro(marca='Ford', modelo='Prefect', ano=1979, cor='blue')]

Suporte em linguagens

As seguintes linguagens de programação suportam metaclasses:

Ver também

Strategi Solo vs Squad di Free Fire: Cara Menang Mudah!