associação pythonbrasil[11] django zope/plone planet Início Logado como (Entrar)

Revisão 5e 2009-05-10 05:20:37

Excluir mensagem

CarpintariaPython

(Esse texto é uma tradução de Intermediate and Advanced Software Carpentry in Python, de C Titus Brown. O original, segundo o autor, estão licenciados sob a Creative Commons, então a tradução também está.)

Os tópicos apresentados a partir de Pacotes úteis, subprocess, rpy, matplotlib, Python Idiomático 3: new-style classes, Atributos gerenciados, Descriptors, GUI Gossip, Python 3.0 já foram quase que completamente traduzidos por LeonardoAndrade; uma atualização da tradução será posta aqui nos próximos dias. Se você deseja ajudar na tradução desse artigo, escolha outros tópicos para evitar retrabalho. Também informe nessa páginas quais tópicos você já traduziu, para informar a outros colaboradores.

Carpintaria de Software Intermediária e Avançada em Python

Bem-vindo! Você acabou de cair nas apostilas de um curso que eu apresentei no Lawrence Livermore National Lab, entre 12 e 14 de junho de 2007.

Essas notas foram planejadas para acompanhar minha palestra, que era uma demostração de de várias características e pacotes "intermediários" de Python. Como a demonstração era interativa, essas notas não são notas completas do que foi apresentado no curso. (Perdoe-me por isso; eles atualizaram minhas apostilas para serem mais completas...)

Entretanto, todas as 70 páginas são livres para ver e imprimir; então, aproveite.

Todos os erros são, naturalmente, meus. Note que quase todos os exemplos começando com >>> são doctests, então você pode pegar o fonte e rodar doctest sobre ela para garantir que eu fui honesto. Só faça-me um favor e rode os doctests com Python 2.5 ;)

Note que o primeiro dia do curso foi até o final de "Testando seu software"; o segundo dia foi até o final de "Recursos online sobre Python"; e o terceiro dia terminou o curso.

Códigos de exemplo (a maior parte das seções sobre sobre extensões em C) estão disponíveis aqui; veja o README para mais informações.

Conteúdo

  1. Carpintaria de Software Intermediária e Avançada em Python
  2. Python Idiomático
    1. Alguns tipos de dados básicos
    2. List comprehentions
    3. Criando seus próprios tipos
    4. Iterators
    5. Generators
    6. assert
    7. Conclusões
  3. Estruturando, Testando e Mantendo Programas Python
    1. Programando para reusabilidade
    2. Módulos e scripts
    3. Pacotes
    4. Uma pequena digressão: nomes e formatação
    5. Outra pequena digressão: doctstrings
    6. Compartilhando dados entre código
    7. Escopo: uma digressão
    8. De volta ao compartilhamento de dados
    9. Como módulos são carregados (e quando código é executado)
    10. PYTHONPATH, e encontrando pacotes e módulos duranto o desenvolvimento
    11. setup.py e distutils: a maneira à antiga de instalar pacotes Python
    12. setup.py, eggs e easy_install: a nova maneira da moda de instalar pacotes Python
  4. Testando seu software
    1. Uma introdução a conceitos sobre teste
    2. O módulo doctest
    3. Tests unitários com unittest
    4. Testes com nose
    5. Análise de cobertura de código
    6. Adicionando testes para um projeto existente
    7. Pensamentos finais sobre testes automatizados
  5. Uma Introdução Estendida ao Framework de Teste Unitiário nose
    1. O que são testes unitários?
    2. Por que usar um framework? (e por que nose?)
    3. Alguns poucos exemplos simples
      1. Suporte a testes
      2. Exemplos são incluídos!
    4. Um guia um tanto mais completo para descoberta de testes e execução
      1. Rodando testes
      2. Depurando descoberta de testes
    5. A linha de comando do nose
      1. -w: Especificando o diretório corrente
      2. -s: Não capturar stdout
      3. -v: Saída de informação e debug
      4. Especificando uma lista de testes para rodar
    6. O plugin 'attrib' -- rodando seletivamente subconjuntos de testes
    7. Rodando nose programaticamente
    8. Escrevendo plugins -- um guia simples
    9. Ressavlas ao nose -- avise o comprador, ocasionamente
  6. Python Idiomático revisitado
    1. sets
    2. any e all
    3. Exceções e hierarquia de exceções
    4. Decorators de funções
    5. try/finally
    6. Argumentos de funções e wrapping functions
  7. Mensurando e Aumentando Performance
    1. Que profiler devo usar?
    2. Mensurando snippets de código com timeit
  8. Acelerando Python
    1. psyco
      1. Instalando psyco
      2. Usando psyco
    2. pyrex
  9. Ferramentas para Ajudar Você a Trabalhar
    1. IPython
    2. screen e VNC
    3. Trac
  10. Recursos Online sobre Python
  11. Embrulhando(?) C/C++ para Python
    1. Wrapping manual
    2. Embrulhando código Python com SWIG
    3. Embrulhando código C com pyrex
    4. ctypes
    5. SIP
    6. Recomendações
    7. Uma ou duas outras notas sobre wrapping
  12. Pacotes para Multiprocessamento
    1. threading
    2. parallelpython
    3. Rpyc
    4. pyMPI
    5. multitask
  13. Pacotes úteis
    1. subprocess
    2. rpy
    3. matplotlib
  14. Python Idiomático 3: new-style classes
    1. Atributos gerenciados
    2. Descriptors
  15. GUI Gossip
  16. Python 3.0

Python Idiomático

Alguns tipos de dados básicos

List comprehentions

Criando seus próprios tipos

Iterators

Iterators são outra característica de Python; diferente dos tipos lista e dicionário discutidos anteriormente, um iterator não é de fato um tipo, mas um protocolo. Isso significa que Python concorda em respeitar qualquer coisa que suporte um conjunto particular de métodos como se fosse um iterator. (Estes protocolos aparecem em todo lugar em Python; nós estivemos tirando vantagem do protocolo de mapeamento e de sequência anteriormente, quando definimos __getitem__ e __len__, respectivamente.

Iterators são versões mais gerais do protocolo de sequência; segue um exemplo:

>>> class SillyIter:
...   i = 0
...   n = 5
...   def __iter__(self):
...      return self
...   def next(self):
...      self.i += 1
...      if self.i > self.n:
...         raise StopIteration
...      return self.i

>>> si = SillyIter()
>>> for i in si:
...   print i
1
2
3
4
5

Aqui, __iter__ apenas retorna self, um objeto que possui a função next(), que (quando chamado) retorna um valor ou levanta uma excessão StopIteration.

Na verdade, nós já vimos diversos iterators disfarçados; em particular, enumerate é um iterator. Para fixar o conceito, segue uma reimplementação simples de enumerate:

>>> class my_enumerate:
...   def __init__(self, some_iter):
...      self.some_iter = iter(some_iter)
...      self.count = -1
...
...   def __iter__(self):
...      return self
...
...   def next(self):
...      val = self.some_iter.next()
...      self.count += 1
...      return self.count, val
>>> for n, val in my_enumerate(['a', 'b', 'c']):
...   print n, val
0 a
1 b
2 c

Você também pode iterar sobre um iterator à "moda antiga":

>>> some_iter = iter(['a', 'b', 'c'])
>>> while 1:
...   try:
...      print some_iter.next()
...   except StopIteration:
...      break
a
b
c

mas isso seria tolice na maior parte das situações! Eu uso isso apenas se quero pegar os primeiros valores de um iterator.

Com iterators, uma coisa para ficar atento é o retorno de self pela função __iter__. Você pode tão facilmente escrever um iterator que não é tão reutilizável quanto você pensa que é. Por exemplo, suponha que você tenha a seguinte classe:

>>> class MyTrickyIter:
...   def __init__(self, thelist):
...      self.thelist = thelist
...      self.index = -1
...
...   def __iter__(self):
...      return self
...
...   def next(self):
...      self.index += 1
...      if self.index < len(self.thelist):
...         return self.thelist[self.index]
...      raise StopIteration

Isso funciona como esperado contanto que você crie um novo objeto a cada vez:

>>> for i in MyTrickyIter(['a', 'b']):
...   for j in MyTrickyIter(['a', 'b']):
...      print i, j
a a
a b
b a
b b

mas vai quebrar se você criar o objeto apenas uma vez:

>>> mi = MyTrickyIter(['a', 'b'])
>>> for i in mi:
...   for j in mi:
...      print i, j
a b

porque self.index é incrementado em cada loop.

Generators

assert

Conclusões

Estruturando, Testando e Mantendo Programas Python

Programando para reusabilidade

Módulos e scripts

Pacotes

Uma pequena digressão: nomes e formatação

Outra pequena digressão: doctstrings

Compartilhando dados entre código

Escopo: uma digressão

De volta ao compartilhamento de dados

Como módulos são carregados (e quando código é executado)

PYTHONPATH, e encontrando pacotes e módulos duranto o desenvolvimento

setup.py e distutils: a maneira à antiga de instalar pacotes Python

setup.py, eggs e easy_install: a nova maneira da moda de instalar pacotes Python

Testando seu software

"Depurar é duas vezes mais difícil que escrever o código pela primeira vez. Portanto, se você escreve o código da forma mais inteligente possível, você não é, por definição, inteligente o suficiente para depurá-lo." -- Brian W. Kernighan.

Todo mundo testa seu software em algum nível, ainda que apenas executando e experimentando (tecnicamente conhecido como "teste de fumaça" ["smoke testing"]). A maioria dos programadores faz uma certa quantidade de testes exploratórios, o qual envolve a execução através de vários caminhos funcionais em seu código e vendo se eles funcionam.

Teste sistemático, no entanto, é um assunto diferente. Testes sistemáticos simplesmente não podem ser feitos de forma adequada sem uma certa (grande!) quantidade de automação, pois cada mudança no software significa que o mesmo precisa ser totalmente testado de novo.

Abaixo, vou apresentar você a alguns conceitos de teste automatizado de nível mais baixo e mostrar como usar as estruturas embutidas no Python para começar a escrever testes.

Uma introdução a conceitos sobre teste

Existem vários tipos de testes que são particularmente úteis para programadores de pesquisa. Testes unitários são testes para unidades de funcionalidade razoavelmente pequenas e específicas. Testes funcionais testam caminhos funcionais inteiros através do seu código. Testes de regressão asseguram que (dentro da resolução dos seus registros) a saída do seu programa não mudou.

Todos os três tipos de teste são necessários de formas diferentes.

Testes de regressão mostram quando ocorrem mudanças inesperadas de comportamento e podem confirmar que o processamento básico dos dados permanece funcionando. Para cientistas, isto é particularmente importante se você está tentando vincular resultados de pesquisas passadas a novos resultados: se você não consegue mais replicar seus resultados originais utilizando o código modificado, então vocês deve suspeitar do código, a menos que as mudanças sejam intencionais.

Em contraste, tanto testes unitários quanto funcionais tendem a ser baseados em expectativas. Quero dizer com isso que você usa os testes para estabelecer qual comportamento você espera do seu código e escreve seus testes de forma que eles assumam que tais expectativas sejam atendidas.

A diferença entre testes unitários e funcionais é nebulosa na maioria das implementações atuais. Testes unitários tendem a ser mais curtos e requerem menos preparação (setup) e encerramento (teardown), enquanto testes funcionais poderm ser bastante longos. Gosto da distinção de Kumar McMillan: testes funcionais mostram quando seu código está quebrado, enquanto que testes unitários mostram onde seu código está quebrado. Ou seja, por causa da granularidade mais fina dos testes unitários, um teste unitário quebrado pode identificar um trecho de código em particular como fonte de um erro, enquanto que os testes funcionais meramente mostram que uma funcionalidade está quebrada.

O módulo doctest

Tests unitários com unittest

Testes com nose

Análise de cobertura de código

Adicionando testes para um projeto existente

Pensamentos finais sobre testes automatizados

Uma Introdução Estendida ao Framework de Teste Unitiário nose

O que são testes unitários?

Por que usar um framework? (e por que nose?)

Alguns poucos exemplos simples

Suporte a testes

Exemplos são incluídos!

Um guia um tanto mais completo para descoberta de testes e execução

Rodando testes

Depurando descoberta de testes

A linha de comando do nose

-w: Especificando o diretório corrente

-s: Não capturar stdout

-v: Saída de informação e debug

Especificando uma lista de testes para rodar

== Rodando doctests no nose ===

O plugin 'attrib' -- rodando seletivamente subconjuntos de testes

Rodando nose programaticamente

Escrevendo plugins -- um guia simples

Ressavlas ao nose -- avise o comprador, ocasionamente

== Créditos==

Python Idiomático revisitado

sets

any e all

Exceções e hierarquia de exceções

Decorators de funções

try/finally

Argumentos de funções e wrapping functions

Mensurando e Aumentando Performance

Que profiler devo usar?

Mensurando snippets de código com timeit

Acelerando Python

psyco

Instalando psyco

Usando psyco

pyrex

Ferramentas para Ajudar Você a Trabalhar

IPython

screen e VNC

Trac

Recursos Online sobre Python

Embrulhando(?) C/C++ para Python

Wrapping manual

Embrulhando código Python com SWIG

Embrulhando código C com pyrex

ctypes

SIP

== Boost.Python==

Recomendações

Uma ou duas outras notas sobre wrapping

Pacotes para Multiprocessamento

threading

== Escrevendo (e indicando) extensões C threadsafe ==

parallelpython

Rpyc

pyMPI

multitask

Pacotes úteis

subprocess

rpy

matplotlib

Python Idiomático 3: new-style classes

Atributos gerenciados

Descriptors

GUI Gossip

Python 3.0