copy — Operações de cópia profunda e cópia rasa

Código-fonte: Lib/copy.py


As instruções de atribuição no Python não copiam objetos, elas criam ligações entre um destino e um objeto. Para coleções que são mutáveis ​ou contêm itens mutáveis, às vezes é necessária uma cópia para que seja possível alterar uma cópia sem alterar a outra. Este módulo fornece operações genéricas de cópia profunda e rasa (explicadas abaixo).

Resumo da interface:

copy.copy(obj)

Retorna uma cópia rasa de obj.

copy.deepcopy(obj[, memo])

Retorna uma cópia profunda de obj.

copy.replace(obj, /, **changes)

Cria um novo objeto do mesmo tipo que obj, substituindo campos por valores de changes.

Adicionado na versão 3.13.

exception copy.Error

Levantada para erros específicos do módulo.

A diferença entre cópia profunda e rasa é relevante apenas para objetos compostos (objetos que contêm outros objetos, como listas ou instâncias de classe):

  • Uma cópia rasa constrói um novo objeto composto e então (na medida do possível) insere nele referências aos objetos encontrados no original.

  • Uma cópia profunda constrói um novo objeto composto e então, recursivamente, insere nele cópias dos objetos encontrados no original.

Frequentemente, existem dois problemas com operações de cópia profunda que não existem com operações de cópia rasa:

  • Objetos recursivos (objetos compostos que, direta ou indiretamente, contêm uma referência a si mesmos) podem causar um laço recursivo.

  • Como a cópia profunda copia tudo, ela pode copiar muito, como dados que devem ser compartilhados entre as cópias.

A função deepcopy() evita esses problemas:

  • mantendo um dicionário memo de objetos já copiados durante a passagem de cópia atual; e

  • permitindo que as classes definidas pelo usuário substituam a operação de cópia ou o conjunto de componentes copiados.

Este módulo não copia tipos como módulo, método, stack trace (situação da pilha de execução), quadro de empilhamento, arquivo, soquete, janela ou outros tipos semelhantes. Ele “copia” funções e classes (rasas e profundamente), devolvendo o objeto original inalterado; isso é compatível com a maneira que estes itens são tratados pelo módulo pickle.

Cópias rasas de dicionários podem ser feitas usando dict.copy(), e de listas atribuindo uma fatia de toda a lista, por exemplo, lista_copiada = lista_original[:].

As classes podem usar as mesmas interfaces para controlar a cópia que usam para controlar a serialização com pickle. Veja a descrição do módulo pickle para informações sobre esses métodos. Na verdade, o módulo copy usa as funções pickle registradas do módulo copyreg.

Para que uma classe defina sua própria implementação de cópia, ela pode definir métodos especiais __copy__() e __deepcopy__().

object.__copy__(self)

Chamado para implementar a operação de cópia rasa; nenhum argumento adicional é passado.

object.__deepcopy__(self, memo)

Chamado para implementar a operação de cópia profunda; é passado um argumento, o dicionário memo. Se a implementação __deepcopy__ precisar fazer uma cópia profunda de um componente, ela deve chamar a função deepcopy() com o componente como primeiro argumento e o dicionário memo como segundo argumento. O dicionário memo deve ser tratado como um objeto opaco.

A função copy.replace() é mais limitada que copy() e deepcopy(), e tem suporte a apenas tuplas nomeadas criadas por namedtuple(), dataclasses e outras classes que definem o método __replace__().

object.__replace__(self, /, **changes)

Este método deve criar um novo objeto do mesmo tipo, substituindo campos por valores de changes.

Ver também

Módulo pickle

Discussão dos métodos especiais usados ​para dar suporte à recuperação e restauração do estado do objeto.