Introdução A Python Modulo A PDF
Introdução A Python Modulo A PDF
Introdução A Python Modulo A PDF
Apresentao
Este material foi desenvolvido para apoiar os cursos da srie Introduo a Python ministrados pelo Grupo Python para nossos mais diversos tipos de audincia. O guia inteiro composto por trs volumes que correspondem aos mdulos dados nos nossos cursos: Mdulo A Bem-vindo a Python!, Mdulo B Python Orientado a Objetos e Mdulo C Tkinter. Todos eles podem ser encontrados na internet, nas pginas do Grupo Python (http://grupopython.cjb.net), na pgina do autor (http://labaki.tk) e na pgina da comunidade Python no Brasil (http://www.python-brasil.com.br). Desenvolvemos este guia pensando tambm nos autodidatas que no participaram dos nossos cursos e baixaram este material da internet. Se voc est nesta situao, pode tirar dvidas por e-mail. Lembramos que o nosso objetivo no ensinar programao, e sim guiar voc nos passos bsicos em Python. Se sua inteno aprender a programar, prefira o excelente material do prof. Luciano Ramalho em (http://www.magnet.com.br). Recomendamos que voc acompanhe e implemente os exemplos, tente entender os erros que podem ocorrer e tente resolver as questes que eventualmente aparecem ao longo deste material. Ao final, h alguns exemplos e exerccios de fixao. Mande suas dvidas, sugestes, crticas ou comentrios por e-mail! Sua opinio sobre o guia muito importante para ns.
Josu Labaki
Grupo Python Departamento de Engenharia Mecnica UNESP Campus de Ilha Solteira
labaki@feis.unesp.br
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 2
ndice
Parte I Bem-vindo a Python! 5 1.Por que estudar Python? 5 2.Usando o interpretador de Python 6 3.Usando Python como uma calculadora 7 4.Variveis nada a declarar 8 Parte II Tipos de Objetos 10 1.Strings 10 2.Listas 15 3.Tuplas 19 4.Dicionrios 21 Parte III Formatando Objetos 23 1.Formatando strings 23 2.Recursos de manipulao de listas 26 Parte IV Testes, Loops, Varreduras e Erros 29 1.Enquanto... 29 2.Entrada de dados 31 3.If... Elif... Else... 33 4.For 35 5.Break 40 6.Tratamento de erros 42 Parte V Nosso primeiro programa completo 46 Parte VI Funes e Mdulos 51 1.Definindo funes 51 2.Funes sobre seqncias 57 3.Return versus side effects 62 JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 3
INTRODUO A PYTHON MDULO A BEM-VINDO A PYTHON! 4.Mdulos 66 5.Um pouco de teoria: Escopo 73 Parte VII Alguns truques... 79 1.Manipulao de arquivos 79 2.Mdulo OS mexendo no Sistema Operacional 83 3.Time e Random 86 4.Python e AutoCAD: bons amigos 89 5.Msica, maestro! 92 6.Calendar 93 Parte VIII Exemplos 95 1.numanalise.py Trabalhando com funes 95 2.catch_mail.py Trabalhando com strings 98 3.matrizes.py Cadeias de for e listas 100 4.morse_code.py Dicionrios e sons 102 5.newton-raphson.py Escrevendo em arquivos 104 6.memoria.py Um joguinho simples de memria 109 Parte IX - Exerccios 111 Vem a... 113 JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 4
Porque Python uma linguagem simples e elegante. Porque Python fcil de aprender. Porque Python pode ser usado para resolver uma grande variedade de problemas. Porque Python incentiva voc a escrever seus programas da maneira correta, sem que isso se torne um empecilho produtividade. Python tem uma curva de aprendizado bastante interessante, permitindo que novos programadores, mesmo os que nunca tenham programado antes, sejam imediatamente produtivos escrevendo scripts procedurais. O programador pode rodar o interpretador como um shell, vendo imediatamente o resultado da sada de cada comando e explorando os recursos da linguagem interativamente. Para construir aplicaes mais complexas, Python possibilita a fcil migrao para a programao orientada a objetos. Um programa pode evoluir naturalmente para esse paradigma medida que se torna mais complexo. A facilidade inicial do Python no barateia a linguagem, como comum em linguagens que tm por objetivo expresso serem fceis de aprender. Python simples de aprender porque uma linguagem bem planejada.
O sinal usado por Python para potncias algbricas **: >>> 2**3 # o mesmo que 23 (dois ao cubo). 8 >>> 2**(3+6) # Dois elevado a 9 512 H um ltimo operador que retorna o resto da diviso entre dois nmeros: >>> 7 % 2 # O resto da diviso entre 7 e 2 1
>>> valor1=2 >>> valor1 2 >>> type(valor1) <type 'int'> Perceba que no precisamos declarar a varivel valor1 antes de utiliza-la. Se agora voc quiser atribuir o valor Boa tarde! ao nome valor1, pode fazer isso naturalmente. Desta vez, o objeto valor1 ir incorporar o tipo de Boa tarde! (que uma string o que talvez voc conhea de FORTRAN como character): >>> valor1='Boa tarde!' >>> valor1 'Boa tarde!' >>> type(valor1) <type 'str'> Os principais tipos de objetos em Python so inteiros, floats (nmeros reais, que o computador representa da melhor maneira possvel por meio de variaes na posio do separador decimal), strings (texto), listas, tuplas e dicionrios. Pode-se transformar o tipo original de um objeto para inteiro, float ou string por meio das funes int, float e str, respectivamente, como veremos melhor adiante.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 9
>>> palavra[2] 'r' >>> 2*palavra[0] 'tt' Voc tambm pode solicitar um intervalo de uma seqncia. Por exemplo, para solicitar os valores da seqncia palavra que esto entre os endereos 9 e 12, fazemos >>> palavra[9:12] 'mic' Para entender melhor o que aconteceu aqui, lembre-se de que os endereos em Python se referem aos intervalos entre os itens, e no ao item em si. Isto , o endereamento usado no objeto palavra o seguinte: Assim fica fcil entender porque palavra[9:12] retorna os itens m, i e c! Podemos omitir o endereo da direita, se ele for o ltimo da seqncia: >>> palavra[9:] 'mica'
INTRODUO A PYTHON MDULO A BEM-VINDO A PYTHON! JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 11
Da mesma forma, os valores entre o primeiro endereo (0) e o 9 so >>> palavra[0:9] 'termodin\xe2' Ou, como estamos falando do primeiro endereo, >>> palavra[:9] 'termodin\xe2' Perceba a diferena: >>> print palavra[:9] 'termodin'
O que obteremos de print palavra[:]? Se voc no deseja recolher todos os valores da seqncia contidos no intervalo, pode determinar um incremento: >>> palavra[1:8] 'ermodin'
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 12
>>> palavra[1:8:2] 'emdn' Assim so retornados os valores de palavra entre os endereos 1 e 8, com incremento 2 (ou seja, os elementos dos endereos 1, 3, 5 e 7). Veja um exemplo com incremento 3: >>> palavra[1:8:3] 'eon' Ou seja, os elementos dos endereos 1, 4 e 7. Com incremento negativo, possvel varrer uma seqncia de trs para frente, assim: >>> palavra[7:0:-1] 'nidomre' Na ltima sentena conseguimos capturar os valores do endereo 7 at 0, exceto o prximo zero, vasculhando a seqncia palavra de trs para frente. Como estamos falando do primeiro endereo (zero), podemos omitir esta informao: >>> palavra[7::-1] 'nidomret' Igualmente se a varredura comea do ltimo endereo:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 13
>>> palavra[:4:-2] 'ai\xe2i' (Voc j est acostumado a enxergar \xe2 como , no ?). Ora, se podemos omitir o ltimo e o primeiro endereos, ento fica fcil inverter uma seqncia qualquer: >>> palavra[::-1] 'acim\xe2nidomret' Talvez voc j tenha pensado como isso deve ser interessante para verificar se uma palavra um palndromo1... >>> palindromo='socorram me subi no onibus em marrocos' >>> palindromo[::-1] 'socorram me subino on ibus em marrocos' A varredura de strings ao contrrio no to usada, mas este conceito importante e ser bem mais til quando aplicado a outros tipos de seqncias que veremos adiante. Quando executado sobre strings, o operador + significa concatenao, isto , adio ao final da string anterior. 1 Palndromos so palavras ou sentenas que so iguais tanto se lidas de trs para frente quanto no sentido normal.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 14
>>> palavra=palavra+' aplicada' >>> print palavra 'termodinmica aplicada' >>> palavra[14:] 'aplicada' Outra funo que podemos executar sobre qualquer seqncia em Python len(x), que retorna o tamanho da seqncia x. >>> len(palavra) 22
2. Listas
Um dos tipos de objeto mais teis em Python a lista. A lista tambm uma seqncia, e sua sintaxe : >>> lista=[1,2,3] >>> lista [1, 2, 3] Neste exemplo, temos uma lista de inteiros, mas uma lista pode conter floats, strings, outras listas ou outros objetos quaisquer.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 15
Como qualquer seqncia, a lista segue o mesmo sistema de endereamento que j vimos em strings: >>> lista[0] 1 >>> lista[0]+lista[2] 4 O operador + tambm representa concatenao em listas. A seguir estamos concatenando a lista [1,2,3] com a lista [4] (que uma lista de nico elemento). Fazer lista+4 d erro, porque estamos tentando concatenar o inteiro 4 lista [1,2,3]. >>> lista=lista+[4] >>> lista [1, 2, 3, 4] >>> lista+4 Traceback (most recent call last): File "<pyshell#1>", line 1, in -toplevellista+4 TypeError: can only concatenate list (not "int") to list >>> lista=lista+[0,0,0] >>> lista [1, 2, 3, 4, 0, 0, 0]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 16
Tambm podemos solicitar um valor atribudo aos endereos de uma seqncia de trs para frente. O ltimo endereo tambm chamado de [-1], o penltimo de [-2], e assim por diante: >>> lista[-1] 0 >>> lista[2:-2] [3, 4, 0] Aqui, questionamos sobre o tamanho da seqncia lista: >>> len(lista) 7 Uma caracterstica importante das listas que elas so seqncias mutveis, ao contrario das strings. Isso quer dizer que podemos alterar o valor atribudo a um determinado endereo de uma lista, mas no podemos fazer isso com strings. >>> lista[0] = 'zero' >>> lista ['zero', 2, 3, 4, 0, 0, 0] >>> lista[1] = lista[1]+lista[2] >>> lista ['zero', 5, 3, 4, 0, 0, 0] >>> lista[2] = lista[0]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 17
>>> lista ['zero', 5, 'zero', 4, 0, 0, 0] Mas com strings... >>> a='Boa tarde!' >>> a[0]='N' Traceback (most recent call last): File "<pyshell#3>", line 1, in -toplevela[0]='N' TypeError: object doesn't support item assignment No tenha medo destas mensagens de erro! Elas nos ajudam a entender o que est havendo de errado. A linha fundamental a ltima, que diz que tipo de erro ocorreu. Neste caso, a mensagem object doesn't support item assignment (o objeto no suporta atribuio aos itens), confirma o que dissemos sobre as strings serem imutveis. Observe uma aplicao interessante para listas de listas: >>> linha1=[1,2,3] >>> linha2=[0,-1,1] >>> linha3=[3,3,3] >>> matriz=[linha1,linha2,linha3] >>> matriz
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 18
[[1, 2, 3], [0, -1, 1], [3, 3, 3]] >>>matriz[1] [0, -1, 1]
Por que matriz[1] retorna esta lista em vez do inteiro 2? Chamando um elemento dessa pseudo-matriz: >>> matriz[1][2] 1
3. Tuplas
Tuplas so objetos como as listas, com a diferena de que tuplas so imutveis como strings. Uma vez criadas, no podem ser modificadas. A sintaxe de tuplas : >>> tupl=(1,2,3) >>> tupl (1, 2, 3) >>> tupl[0]=0 Traceback (most recent call last): File "<pyshell#10>", line 1, in -toplevelJOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 19
tupl[0]=0 TypeError: object doesn't support item assignment As tuplas podem tambm conter quaisquer objetos; inteiros, floats, listas, outras tuplas, etc. Com as tuplas podemos fazer algo bem comum nos programas em Python, chamado packing-unpacking: >>> a,b = 0,'Deu certo?' >>> a 0 >>> b 'Deu certo?' como se estivssemos dizendo a e b so iguais a 0 e Deu certo?, respectivamente. Mais fantstico ainda, podemos trocar o valor de dois objetos facilmente: >>> a,b=b,a >>> a 'Deu certo?' >>> b 0
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 20
4. Dicionrios
Dicionrios tambm so contineres, mas no so seqncias porque no so indexados seqencialmente da forma que strings, listas e tuplas so. Seu sistema de endereamento por chaves. Cada chave tem um valor atribudo. Se voc quer saber um valor de um dicionrio, deve perguntar pelas chaves, no pelos endereos. A sintaxe de dicionrios : >>> aurelio={'denomiao':'ilha solteira','populao':23000,'renda':1500} >>> aurelio {'popula\xe7\xe3o': 23000, 'denomia\xe7\xe3o': 'ilha solteira', 'renda':1500} Neste exemplo, temos o inteiro 23000 atribudo chave populao. Para adicionar novos valores a outras chaves, fazemos >>> aurelio['vocao']='turismo' atribuindo assim a string turismo nova chave vocao. >>> aurelio {'popula\xe7\xe3o': 23000, 'voca\xe7\xe3o': 'turismo', 'denomia\xe7\xe3o': 'ilha solteira', 'renda': 1500} E para requisitar o valor atribudo a alguma chave,
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 21
>>> aurelio['renda'] 1500 Dicionrios so mutveis. fcil alterar o valor contido em uma chave: >>> aurelio['renda']=aurelio['renda']+200 >>> aurelio['renda'] 1700 Podemos verificar quais chaves um dicionrio possui ou perguntar se ele contm alguma chave em especial: >>> aurelio.keys() ['popula\xe7\xe3o', 'voca\xe7\xe3o', 'denomia\xe7\xe3o', 'renda'] >>> aurelio.has_key('idade') False >>> aurelio['idade']=33 >>> aurelio.has_key('idade') True A sentena dicionario.items() retorna uma lista de tuplas contendo o par (chave,valor atribudo): >>> aurelio.items() [('idade', 33), ('popula\xe7\xe3o', 23000), ('voca\xe7\xe3o', 'turismo'), ('denomia\xe7\xe3o', 'ilha solteira'), ('renda', 1700)]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 22
Vejamos alguns exemplos: >>> constante=3.14 >>> print 'O valor de pi %f' %constante O valor de pi 3.14 Observe a consistncia do cdigo com o esquema mostrado acima: O marcador % colocado no lugar onde ser inserido o valor contido no nome constante. Como o valor de constante um float (3.14), logo depois do marcador vem a indicao do tipo da varivel, f. Novos exemplos com outros tipos de variveis:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 23
>>> nome='abacaxi' >>> caracteristica='amarelo' >>> print '%s uma fruta' %nome abacaxi uma fruta >>> print '%s %s' %(nome,caracteristica) abacaxi amarelo assim que inserimos vrios valores dentro de uma string: usando uma tupla contendo os valores, na ordem em que aparecero. >>> print '%f %ss tambm so %ss' %(constante,nome,caracteristica) 3.14 abacaxis tambm so amarelos Desta vez trs valores foram inseridos na string. Eles aparecem na string na mesma ordem em que foram dispostos na tupla. Os tipos acompanham: primeiro f, que indica que constante um float, e depois s e s, que indica que nome e caracterstica so strings. Observe: no segundo e no terceiro marcadores temos %ss com dois s. O primeiro s indica o tipo da varivel, string, e o terceiro faz parte da string neste caso est fazendo o plural de abacaxi e amarelo.
O que seria impresso se a string comeasse com %fs .....? Inserindo .x depois de um marcador de floats dizemos que o valor aparecer com somente x casas decimais:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 24
>>> valor = 3.1415926535897931 >>> print 'O valor %.2f' %valor O valor 3.14 >>> print 'O valor %.3f' %valor O valor 3.141 >>> print 'O valor %f' %valor O valor 3.1415926535897931 Inserindo y. depois de um marcador qualquer, dizemos que o valor que for inserido no lugar do marcador dever ter exatamente y caracteres: >>> print 'O valor %8.2f' %valor O valor 3.14 Somando o inteiro 3, o marcador decimal e as duas casas decimais, todo o float inserido no marcador tem 4 caracteres. Ento so adicionados mais 4 espaos em branco antes do valor, para completar os 8 caracteres exigidos. >>> print 'O valor %12.4f' %valor O valor 3.1416 O inteiro 3, o marcador decimal e quatro casas decimais exigem mais 6 espaos em branco para completar 12 caracteres.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 25
Se em vez de adicionar um valor ao final da lista voc quiser inserir num endereo especfico, pode usar insert(endereo,valor). Aqui inserimos o valor start no endereo zero da lista a. >>> a.insert(0,'start') >>> a ['start',1,2,3,'poncan',0,0,-3] Tambm h algumas ferramentas para retirar valores de listas. Remove uma delas. Fazendo remove(0) sobre a lista a, retiramos dela o primeiro valor 0 que apareceu. Depois fazemos o mesmo com poncan: >>> a.remove(0) >>> a ['start', 1, 2, 3, 'poncan', 0, -3 >>> a.remove('poncan') >>> a ['start', 1, 2, 3, 0, -3] Um pouco parecido com remove o mtodo pop. objeto1.pop(i) remove de objeto1 o valor armazenado no endereo i: >>> a.pop(0) 'start' >>> a [1, 2, 3, 0, -3]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 27
J o mtodo count retorna o nmero de vezes que seu argumento aparece na lista. Para saber o nmero de vezes (uma vez) que o valor 3 aparece na lista a, fazemos: >>> a.count(3) 1 J vimos que a varredura ao contrrio, por meio de incremento negativo, poderia ser aplicada a qualquer seqncia, no somente a strings. Vamos aproveitar que estamos falando de manipulao de listas para verificar isso. Veja como fcil obter a Escala de Dureza de Mohs em ordem decrescente de dureza: >>> Mohs=['Talco', 'Gipsita', 'Calcita', 'Fluorita', 'Apatita', 'Ortoclsio', 'Quartzo', 'Topzio', 'Corndon', 'Diamante'] >>> Mohs[::-1] ['Diamante', 'Cor\xedndon', 'Top\xe1zio', 'Quartzo', 'Ortocl\xe1sio', 'Apatita', 'Fluorita', 'Calcita', 'Gipsita', 'Talco']
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 28
indentao2. Quando fizermos um programa completo, voc perceber que quando no quiser que um loop faa mais nada, basta retornar o alinhamento. Uma caracterstica do IDLE que ele percebe onde o cdigo deve ser indentado e faz isso automaticamente. Voc percebe esta caracterstica quando pressioa ENTER depois de ... b < 5:. A linha inferior j automaticamente alinhada debaixo do while. Ao pressionar ENTER novamente depois de print b e b=b+1, acontece a mesma coisa, isso porque o IDLE espera que voc informe mais um procedimento. Pressionando ENTER mais uma vez, o Shell perceber que no h mais procedimentos a serem adicionados e s ento executa o cdigo. Vejamos um loop mais elaborado; desta vez, com string formatada. >>> b=1 >>> while b < 5: print '%i dlares valem %.2f reais' %(b,b*2.98) b=b+1 1 dlares valem 2.98 reais 2 dlares valem 5.96 reais 3 dlares valem 8.94 reais 4 dlares valem 11.92 reais Se no for includa a ltima linha que incrementa o valor de b, este valor permanecer sempre 1, e o loop testar sempre 1 < 5. Como isso sempre verdade, ser impressa eternamente a linha 2 Voc tambm pode encontrar na literatura de Python o termo endentao, que significa a mesma coisa. No entanto, indentao mais utilizado, embora seja um neologismo (j incorporado por alguns dicionrios da lngua portuguesa).
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 30
1 dlares valem 2.98 reais Se voc fizer este teste, basta pressionar CTRL+C (se estiver usando o IDLE) para cancelar o loop. Ao estudar o script de outras pessoas, pode ser que voc encontre algo como b+=1, ou b*=2. Escrever b+=n o mesmo que b=b+n, e escrever b*=n o mesmo que b=b*n.
2. Entrada de dados
Veja como ler uma entrada do usurio em Python: >>> x = raw_input('Informe a fase: ') Informe a fase: vapor >>> x 'vapor' A funo raw_input(s) mostra a string s para o usurio e espera que ele fornea algum dado. Ento, a funo transforma este dado numa string e a retorna para o cdigo. Para salvar esta entrada e usa-la posteriormente, basta salvar o resultado da funo sob um nome qualquer, como fizemos acima com x. Todo mundo, ao menos uma vez na vida, se esquece de que raw_input() retorna sempre uma string e perde tempo com isso. Veja um clssico:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 31
>>> a = raw_input('Tamanho do lado do quadrado: ') Tamanho do lado do quadrado: 23 >>> print 'O permetro deste quadrado ', 4*a O permetro deste quadrado 23232323 >>> type(a) <type 'str'> Obviamente, o permetro de um quadrado de lado 23 no 23232323. Este o resultado da multiplicao da string a pelo inteiro 4: quatro vezes a string. Se voc quer solicitar um inteiro ou um float do usurio, deve converter o resultado de raw_input() para o tipo de objeto desejado usando as funes int ou float: >>> a = raw_input('Tamanho do lado do quadrado: ') Tamanho do lado do quadrado: 23 >>> a = int(a) >>> type(a) <type 'int'> >>> print 'O permetro deste quadrado ', 4*a O permetro deste quadrado 92 Ou mais diretamente, >>> a = int(raw_input('Tamanho do lado do quadrado: '))
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 32
Tamanho do lado do quadrado: 23 >>> type(a) <type 'int'> Analogamente, a funo para transformar o tipo de um objeto em float float(x): >>> r = float(raw_input('Raio: ')) Raio: 4.5 >>> type(r) <type 'float'>
Ateno para a indentao: tudo que desejamos que seja feito caso x seja lquido est alinhado. O mesmo com elif e else. >>> x = raw_input('Informe a fase: ') Informe a fase: vapor >>> if x == 'lquido': print 'Menos de 100o C' elif x == 'vapor' : print 'Mais de 100o C' else: print 'Menos de 0o C' Mais de 100o C Vamos dar uma melhorada no cdigo da pgina 30, j que a impresso que fizemos l deu uma escorregada gramatical (1 dlares valem...): >>> b = 1 >>> while b < 4: if b==1: print '%i dlar vale %.2f reais' %(b,b*2.98) else: print '%i dlares valem %.2f reais' %(b,b*2.98) b+=1
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 34
1 dlar vale 2.98 reais 2 dlares valem 5.96 reais 3 dlares valem 8.94 reais
4. For
O for de Python uma ferramenta poderosa e muito caracterstica desta linguagem. Com for, podemos varrer qualquer seqncia (strings, listas, tuplas) ou dicionrio. >>> a = ['Joo', 'Rafael', 'Douglas'] >>> a ['Jo\xe3o', 'Rafael', 'Douglas'] >>> for i in a: print i Joo Rafael Douglas >>> for x in a: print '%s tem %i letras' %(x,len(x)) Joo tem 4 letras Rafael tem 6 letras Douglas tem 7 letras
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 35
Para x em a: - x o nome que demos a cada um dos elementos da seqncia a neste exemplo. Desta forma, quando falamos em x estamos falando do elemento da seqncia a que est sendo estudado no momento. Primeiro, x Joo (o primeiro elemento de a). por isso que quando imprimimos x e len(x) imprimimos na verdade Joo e len(Joo). Simples, no? Poderamos ter usado qualquer nome no lugar de x. Uma funo importante para usarmos o for a funo range(ni,nf,i), que cria uma lista comeando do inteiro ni at o inteiro nf-1, com incremento i entre os nmeros: >>> range(1,11,2) [1, 3, 5, 7, 9] Os valores de ni, nf e i precisam ser nmeros inteiros. Podemos omitir ni e i se estes valores forem zero e 1, respectivamente: >>> range(10) [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] >>> range(1,11) [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] >>> for elefante in range(5): print elefante**2 0 1 4 9
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 36
16 Assim o for sobre a seqncia range(5). Para voc perceber que podemos chamar os elementos da seqncia de qualquer coisa, usamos o (bizarro) nome elefante. O tamanho da seqncia a (tamanho 3) um nmero inteiro, ento no h problema em aplicar a funo range sobre len(a). Na verdade, isso o mesmo que range(3): >>> for i in range(len(a)): print i 0 1 2 E aqui vemos porque interessante que range gere uma lista de nmeros inteiros o endereamento das seqncias em Python tambm feito por nmeros inteiros: >>> for i in range(len(a)): print i,a[i] 0 Joo 1 Rafael 2 Douglas
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 37
Nesta altura do campeonato voc j deve ter percebido que a indentao tambm o mecanismo que regula quais aes fazem parte do raio de ao do bloco de for. E assim encadeamos alguns blocos de for: >>> matriz = ((1,0,0),(0,1,0),(0,0,1)) >>> for i in range(len(matriz)): # o mesmo que range(3), ok? print '\n' for j in range(len(matriz)): # idem print matriz[i][j], 1 0 0 0 1 0 0 0 1 O cdigo print '\n' imprime uma linha em branco. Podemos fazer print '\n'*k para imprimir k linhas em branco... Neste exemplo e no anterior, aparece uma vrgula na linha de print. Ela serve para imprimir os termos todos na mesma linha. Ao aplicar uma varredura de for sobre um dicionrio, o nome dado aos elementos interpretado como sendo a chave do dicionrio. Melhor mostrar com exemplos:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 38
>>> oramento = {'asa':1200,'combustvel':200,'motor':4500} >>> for qualquer in orcamento: print qualquer combustvel asa motor O nome qualquer assume o valor das chaves do dicionrio orcamento. Sendo assim, >>> for teste123 in orcamento: print teste123,'custa R$',orcamento[teste123] combustvel custa R$ 200 asa custa R$ 1200 motor custa R$ 4500 Por exemplo: j que teste123 assume o valor das chaves de orcamento, seu primeiro valor assumido 'combustvel', a primeira chave deste dicionrio. Assim, fazer orcamento[teste123] o mesmo que fazer orcamento['combustvel'], que retornar o valor associado a esta chave.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 39
Voc sabe que dicionario.items() retorna uma lista de tuplas contendo o par (chave, valor). Usando este mtodo, como voc faria para imprimir os mesmos resultados do ltimo bloco de for? Lembre-se que o for pode vasculhar qualquer seqncia, inclusive uma lista de tuplas.
5. Break
O comando break serve para abortar um bloco de while ou for. Voc se lembra do operador %? Ele retorna o resto da diviso entre dois inteiros. O cdigo abaixo faz uma varredura na lista [2,3,4,5,6,...,n-1] e testa se n divisvel por algum destes valores. Se for divisvel (ou seja, se n % i == 0), ento n no um nmero primo e a varredura do for cancelada com break. >>> n = int(raw_input('Nmero a testar: ')) Nmero a testar: 28 >>> for i in range(2,n): if n % i == 0: print 'Nmero no primo'; break Nmero no primo Nota: Se o cdigo a ser executado aps um sinal de dois pontos contm poucas linhas, voc pode coloca-lo na frente dos dois pontos e separar os procedimentos com ponto e vrgula, como fizemos acima no bloco de if. O mesmo sinal de dois pontos aparece em blocos de for, while e em definies de funes.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 40
A seguir demonstramos o break abortando um bloco de while: >>> b=10 >>> while 1: b=b-1 print b if b<7: break 9 8 7 6 Observao: while 1: um loop infinito. Use isto sempre que quiser executar um cdigo indefinidamente. Voc pode cancela-lo com um break. O cdigo break cancela somente o loop a que est submetido, se houver vrios loops encadeados: loop1: blablabla blebleble loop2: bliblibli blobloblo break2 # Cancela somente o loop2 break1 # Est alinhado com o loop1, portanto cancela somente loop1.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 41
6. Tratamento de erros
Nem sempre podemos garantir que as operaes que efetuamos sobre uma seqncia ou ao longo de um loop so vlidas, principalmente se a seqncia for muito grande ou for fornecida pelo usurio. Num exemplo simples, vamos recolher uma lista de 3 inteiros fornecidos pelo usurio. O que nos garante que o usurio no tentar inserir uma string na lista? >>> lst=[] >>> for nome in range(3): lst.append(int(raw_input('Digite um valor: '))) Digite um valor: 33 Digite um valor: -4 Digite um valor: a Traceback (most recent call last): File "<pyshell#26>", line 2, in -toplevellst.append(int(raw_input('Digite um valor: '))) ValueError: invalid literal for int(): a A funo int(qualquercoisa) no pde transformar a string a em um inteiro (claro!). Como evitar ento que o usurio fornea uma string ou outro valor qualquer que no possa ser transformado em inteiro? Uma das formas :
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 42
>>> lst=[] >>> for i in range(3): while 1: try: lst.append(int(raw_input('Digite um valor: '))) break except: print 'Digite somente nmeros!!!' Digite um valor: 2 Digite um valor: a Digite somente nmeros!!! Digite um valor: r Digite somente nmeros!!! Digite um valor: t Digite somente nmeros!!! Digite um valor: 33 Digite um valor: e Digite somente nmeros!!! Digite um valor: w Digite somente nmeros!!! Digite um valor: -4.99 >>> lst [2, 33, -4]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 43
A sintaxe de try :
try: primeiro tenta fazer isso e tudo que esteja neste alinhamento (indentao, lembra?) isso tambm e isso except: se qualquer linha dentro do bloco try der erro, ento sero executadas estas linhas.
No exemplo anterior, tentamos concatenar lista lst o valor coletado pelo raw_input, porm transformado em nmero inteiro pela funo int(). Se fornecermos uma string, a funo no conseguir transform-la em inteiro. A entra em ao o bloco except:, com o qual imprimimos um aviso para que o usurio informe um nmero. Como estamos dentro de um bloco de loop infinito, voltamos novamente ao try para recolher um valor do usurio e transforma-lo em inteiro. Se no houver erro desta vez, iremos para a prxima linha, break, que cancelar o loop infinito. Acaba por a? No. Ainda estamos fazendo a varredura do for, ento iremos para o prximo valor da seqncia range(3), e assim por diante. A seguir, recolhemos um inteiro do usurio e tentamos calcular seu inverso. Se o usurio digitar um inteiro menor que zero, teremos problemas quando x for igual a zero, j que no h inverso de zero. Assim, simplesmente tentamos calcular o inverso de x; se no for possvel, informamos que no foi possvel.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 44
>>> x = int(raw_input('Digite um nmero inteiro: ')) Digite um nmero inteiro: -3 >>> while x<3: try: print 1./x except: print 'No possvel dividir 1 por %i' %x x=x+1 # Tambm pode ser x+=1 -0.333333333333 -0.5 -1.0 No possvel dividir 1 por 0 1.0 0.5
Podemos evitar este problema se pularmos a diviso 1./x quando x for igual a zero. Como voc faria isso? Observe tambm a diferena entre 1/3 e 1./3: no primeiro, temos uma diviso de dois inteiros que gera um resultado inteiro, 0. No segundo, temos uma diviso entre o float 1. (que o mesmo que 1.0) e o inteiro 3, que gera um resultado float, 0.33333333333333331. Se quisermos o resultado em float, basta que um dos termos da operao seja um float:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 45
>>> 1+2 3 >>> 1+2.0 3.0 >>> 2*9 18 >>> 2*9. 18.0 >>> 1/3. 0.33333333333333331
mesmo de comear a escrev-lo, assim voc poder contar desde o incio com o auxlio das cores. Quando quiser testar um programa, v a Run > Run Module, ou simplesmente F5. Vamos fazer um programa que teste se um nmero fornecido pelo usurio perfeito. Dizemos que um nmero perfeito se ele for igual soma dos seus divisores menores que ele prprio. O menor nmero perfeito, por exemplo, 6 (6=1+2+3). # perfeitos.py n = int(raw_input('Digite o nmero a ser testado: ')) teste = 0 for i in range(1,n): if n % i == 0: teste=teste+i if teste == n: print n, ' um nmero perfeito' else: print n, 'no um nmero perfeito' Vamos analisar este programa: Linha 1: aqui lemos do usurio o valor de n. Inicialmente o valor uma string, como qualquer valor recebido pela funo raw_input, e depois o transformamos em inteiro pela funo int, para s ento atribuir o resultado ao nome n. Linha 2: Uma forma de fazer o teste se n perfeito ou no criar uma varivel inicialmente valendo zero e ir somando a ela todos os divisores de n
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 47
que encontrarmos. Ao final, testamos se o valor desta varivel igual a n. Se for, n um nmero perfeito. Linha 3: Ok, ento vamos procurar os divisores de n. Sabemos que todos os inteiros candidatos a divisores de n esto na seqncia range(1,n) (agora voc j enxerga essa seqncia como [1,2,3,4,...,n-1], no ?). Ento basta fazer uma varredura (isto , um for) sobre range(1,n) testando quais destes so divisores de n. Linha 4: Se o termo atual da seqncia divisor de n, ento... Linha 5: ...adicione-o a teste! Linha 6: Terminamos a busca dos divisores. Perceba que voltamos ao alinhamento do for. Segundo a indentao, tudo o que escrevermos agora no ser mais executado para cada um dos valores da seqncia. Basta ento testar se a soma dos divisores de n, agora salvo sob o nome teste, corresponde a n. Linha 7: Isto o que ser executado caso o valor de teste seja igual ao de n (ou seja, se n for igual soma dos seus divisores um nmero perfeito, portanto). Linha 8: Seno, (se teste no igual soma dos divisores de n)... Linha 9: ...n no um nmero perfeito. Executando o programa (pressionar F5 o jeito mais rpido), >>> Digite o nmero a ser testado: 6 6 um nmero perfeito
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 48
Voltando janela onde est o programa e pressionando F5 novamente para testar outro valor, >>> Digite o nmero a ser testado: 7 7 no um nmero perfeito Chato pressionar F5 toda hora, no ? Encapsulando todo este cdigo num loop infinito (while 1:, por exemplo), voc pode executar isso infinitas vezes: while 1: o cdigo aqui (indentado corretamente)
Encapsulando o programa todo num loop infinito, o problema passa a ser outro: preciso pressionar CTRL+C para cancelar o teste. Como voc faria para o programa parar os testes de forma mais elegante, de preferncia perguntando ao usurio se ele deseja testar mais algum nmero?
Como bom curioso, voc no est se perguntando que outros nmeros so perfeitos alm do 6? Agora voc tem o poder em suas mos (Python)! Tente fazer um programa para descobrir quais so os 3 nmeros perfeitos que existem at 10000 alm do 6.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 49
Voc tambm pode executar seu programa clicando em seu cone, na pasta em que o programa foi salvo. Desta forma, o programa no ser executado no IDLE, e sim num ambiente mais veloz. Um detalhe quando se usa esta forma de execuo que a janela que aberta enquanto o programa roda se fecha imediatamente aps o trmino do programa, no permitindo ao usurio estudar os dados de sada. Uma forma simples de mant-la aberta at que o usurio deseje incluir uma linha raw_input(Pressione ENTER) ao final do programa, j que raw_input espera por uma entrada do usurio. Se o usurio vai pressionar ENTER ou outra tecla qualquer, no importa, j que esta informao no ser usada para nada. Para editar o programa no IDLE, clique sobre o cone com o boto da direita e escolher Editar com IDLE. Observao: se voc estiver usando Linux (meus parabns!), necessrio adicionar uma linha no comeo do seu programa especificando a localizao do Python: #!/usr/bin/python A linha acima informa ao sistema para procurar pelo programa /usr/bin/python e transmitir o resto do arquivo a este programa. Note que o diretrio onde se encontra o Python varia de sistema para sistema. O diretrio acima vlido para o Conectiva Linux 10.0. Agradecimentos ao parceiro Carlos Mota Castro, fervoroso defensor de Linux, por esta observao. O Carlos adaptou esta nota de uma apostila de Perl que ele estava aprendendo. Estava aprendendo, porque agora ele j ouviu falar de Python...
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 50
Outra funo, desta vez com dois argumentos: >>> def elevar(x,y): return x**y >>> elevar(9,9) 387420489 H meios de definir funes que trabalham com um nmero indefinido de argumentos, de acordo com a sintaxe: def nome_da_funo(*nome1): procedimentos... Esta sintaxe toma todos os elementos inseridos dentro dos parnteses como sendo uma tupla. Observe: >>> def impressora(*argumentos): return argumentos >>> impressora(1,2,3) (1, 2, 3)
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 52
Assim fica claro como os argumentos fornecidos pelo usurio 1, 2 e 3 so admitidos como argumentos da funo em forma de tupla. Considerando isso, podemos trabalhar estes argumentos facilmente, j que temos um certo traquejo em manipular tuplas: >>> def somatoria(*argumentos): soma = 0 for i in argumentos: soma+=i # Voc j sabe: soma+=i o mesmo que soma=soma+i return soma >>> somatoria(1,2,3,4,5) 15 A funo recm-definida assume quantos argumentos o usurio desejar. Apesar disso, alguns argumentos podem no ser admitidos pelos procedimentos definidos dentro da funo, por exemplo, >>> somatoria(1,2,3,4,'teste') Traceback (most recent call last): File "<pyshell#27>", line 1, in -toplevelsomatoria(1,2,3,4,'teste') File "<pyshell#25>", line 4, in somatoria soma+=i TypeError: unsupported operand type(s) for +=: 'int' and 'str'
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 53
Isso porque a funo somatoria no capaz de somar inteiros com a string 'teste'.
Usando o que aprendemos sobre tratamentos de erros, como voc reescreveria a funo somatoria para informar ao usurio que um de seus argumentos foi invlido? Evolua este cdigo para mostrar tambm qual dos argumentos invlido (o primeiro, o segundo, etc.). As funes no precisam ser limitadas a operaes algbricas: >>> def testa_primo(n): teste=1 for i in range(2,n): if n % i == 0: teste=teste+1 if teste != 1: print 'Nmero no primo' else: print 'Nmero primo' >>> testa_primo(28) Nmero no primo >>> testa_primo(7) Nmero primo
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 54
No h problema algum em usar o resultado da funo para outra operao. Assim, podemos usar diretamente f(3) que igual a 9 como parmetro da funo elevar: >>> elevar(f(3),4) 6561 Voc pode colocar um default para um argumento, para ser admitido pela funo caso o usurio da funo no fornea este argumento. >>> def user(nome='Labaki'): print 'Meu nome ',nome >>> user('Mamute') Meu nome Mamute >>> user() Meu nome Labaki Outro exemplo: >>> cadastro=[] >>> def add(x,y='Casado',z=3): cadastro.append([x,y,z]) >>> add('Mozart','Solteiro','sem filhos') >>> cadastro
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 55
[['Mozart', 'Solteiro', 'sem filhos']] >>> add('Beethoven') >>> cadastro [['Mozart', 'Solteiro', 'sem filhos'], ['Beethoven', 'Casado', 3]] >>> add('Bach','Vivo') >>> cadastro [['Mozart', 'Solteiro', 'sem filhos'], ['Beethoven', 'Casado', 3], ['Bach', 'Vi\xfavo', 3]] Se voc quer usar o default de y mas no o de z, pode indicar isso assim: >>> add('Salieri',z=4) >>> cadastro [['Mozart', 'Solteiro', 'sem filhos'], ['Beethoven', 'Casado', 3], ['Bach', 'Vi\xfavo', 3], ['Salieri', 'Casado', 4]] Para facilitar a vida de quem usar seus programas, e talvez a sua prpria quando tiver que fazer manuteno nos seus programas, pode-se inserir pequenas informaes sobre as funes dentro delas. So as chamadas doc strings. >>> def f(x): """Esta funo retorna o quadrado de um nmero""" return x**2 >>> f(2)
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 56
>>> orcamento={'asa':1200, 'combustvel':200, 'motor':4500} >>> def cambio(componente): print componente[0],'R$ ',componente[1] print componente[0],'US$ ',componente[1]/3. print '\n' >>> map(cambio,orcamento.items()) combustvel R$ 200 combustvel US$ 66.6666666667 asa R$ 1200 asa US$ 400.0 motor R$ 4500 motor US$ 1500.0 [None, None, None] Breve explicao sobre o cdigo acima: j vimos que orcamento.items() uma lista de tuplas, >>> orcamento.items() [('combust\xedvel', 200), ('asa', 1200), ('motor', 4500)]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 58
assim podemos aplicar a funo map sobre esta seqncia como sobre qualquer outra. Neste caso, a funo map tomar a primeira tupla, ('combust\xedvel', 200) como o argumento componente da funo cambio. Ao fazer componente[0], estamos fazendo ('combust\xedvel', 200)[0], que correspondente a 'combustvel', enquanto componente[1] correspondente a '200'. E assim com todos os termos da seqncia. Aquele estranho [None, None, None] ao final da execuo um resultado importantssimo e vai alimentar uma discusso posterior sobre return versus side effects. Outra interao funo-seqncia interessante reduce, que toma uma funo e uma seqncia e retorna um nico valor, com a sintaxe reduce(funo, seqncia): >>> def soma(alfa,beta): return alfa+beta >>> reduce(soma,[1,2,3]) 6 Primeiro, reduce assume o primeiro item da lista [1,2,3] (o inteiro 1) como sendo o argumento alfa da funo soma, e o segundo item (o inteiro 2) como sendo o argumento beta. Resultado: 1+2 = 3. Este inteiro 3 tomado ento como o argumento alfa e o prximo item da lista (o inteiro 3) assumido como beta. Resultado: 3+3 = 6. O processo repetido at o final da seqncia.
Este recurso parece ideal para o nosso algoritmo anterior de verificar se um nmero perfeito, no parece? Como voc reescreveria aquele algoritmo usando reduce?
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 59
Mais um exemplo: >>> def eleva(base,expoente): return base**expoente >>> elementos=[4,3,2,1] >>> reduce(eleva,elementos) 4096 Da mesma forma, o que acontece aqui :
base expoente base**expoente Resultado Primeiro Passo 4 3 4**3 64 Segundo Passo 64 2 64**2 4096 Terceiro Passo 4096 1 4096**1 4096 Resultado 4096
Um recurso de trabalho com funes e seqncias muito caracterstico de Python so as list comprehensions, que permitem gerar uma lista a partir de funes simples. >>> [x**2 for x in [1,2,3]] [1, 4, 9]
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 60
Com ltima sentena dizemos: retorne-me uma lista contendo os quadrados de x, sendo x cada um dos elementos da seqncia [1,2,3]. E abaixo, conseguimos uma lista de tuplas contendo o par (valor, quadrado do valor), sendo valor cada um dos itens da seqncia [0, 1, 2, 3, 4]. >>> [(x,x**2) for x in range(5)] [(0, 0), (1, 1), (2, 4), (3, 9), (4, 16)] A seguir, usamos list comprehensions para operaes no algbricas. Esta tambm uma boa ocasio para voc se convencer de que formatao de strings no serve s quando a string est aps um print, mas sim em qualquer lugar onde haja uma string. >>> ['%ss' %nome for nome in ['Proclise','Jilo','Bozo','Cauchy']] ['Proclises', 'Jilos', 'Bozos', 'Cauchys'] Tambm simples usar list comprehensions com seqncia de seqncias. Com uma tupla de tuplas fica assim: >>> valores=((4,5,6),(7,8,9),(5,6,-7)) >>> [a+b+c for (a,b,c) in valores] [15, 24, 4] Aqui, 15 = 4+5+6, 24 = 7+8+9 e 4 = 5+6-7. Os resultados das list comprehensions podem ser usados naturalmente como qualquer lista, inclusive em blocos de for e funes como reduce e map.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 61
>>> def metade(t): return t/2. >>> map(metade,[rrr*2 for rrr in [66,77,88,99]]) [66.0, 77.0, 88.0, 99.0] Neste exemplo, [rrr*2 for rrr in [66,77,88,99]] gera a lista [132, 154, 176, 198]. Ento a funo map aplica a funo metade que divide cada um destes termos por 2, voltando lista original. Mais fcil ainda, voc pode usar as list comprehensions para fazer mapeamentos, aplicando a funo metade diretamente sobre os termos rrr: >>> [metade(rrr*2) for rrr in [66,77,88,99]] [66.0, 77.0, 88.0, 99.0]
Os mesmos princpios se aplicam s funes em Python. Nossas funes podem tomar um ou mais argumentos e sempre retornaro um e somente um resultado, definido pelo cdigo return: def nome_da_funo(argumento_1,argumento_2,...,argumento_n): return parmetro Nesta funo genrica, o (nico) valor de retorno parmetro. Observe uma funo j conhecida, em que o parmetro de retorno o quadrado do argumento de entrada: >>> def f(x): return x**2 >>> f(3) 9 Qualquer outra coisa que a funo imprima, concatene, modifique, etc., no considerada valor de retorno, e sim side effect (efeito colateral). >>> lista=[] >>> def f(x): lista.append(x) print x return x**2
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 63
>>> f(2) 2 4 >>> lista [2] Aqui, o 2 impresso e o 2 concatenado lista so side effects e o 4 impresso o valor de retorno da funo. Isso pode ser percebido facilmente quando o resultado de execuo da funo usado para outra finalidade, como no exemplo abaixo: >>> 2*f(9) 9 162 >>> lista [2, 9] Multiplicar f(9) por um nmero s ter efeito sobre seu valor de retorno, 81 = 9**2. Somente o valor de retorno dessa funo f (o 89) multiplicado por 2, resultando em 162. Os side effects no sofrem alterao alguma o argumento 9 foi impresso e concatenado lista. Quando no definido nenhum valor de retorno com o comando return, a funo entende que seu valor de retorno None. Neste caso, qualquer operao sobre os resultados de execuo da funo gera erros: >>> def h(t): print t**3
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 64
>>> h(7) 343 >>> 2*h(7) 343 Traceback (most recent call last): File "<pyshell#38>", line 1, in -toplevel2*h(7) TypeError: unsupported operand type(s) for *: 'int' and 'NoneType' O erro era previsvel, j que no faz sentido multiplicar None por 2. O famigerado [None, None, None], portanto, se refere aos valores de retorno da funo cambio da pgina 58, quando aplicados sobre os argumentos da lista orcamento.items(). Nas pginas de exerccios h um teste bem interessante e famoso que ilustra muito bem a diferena entre returns e side effects. Tente prever o resultado da execuo do script sem execut-lo.
Quaisquer resultados obtidos com as funes builtins de Python, como map, reduce, raw_input, type, tambm so valores de retorno e/ou side effects. A funo map, por exemplo, gera uma lista a partir da execuo de uma funo sobre uma seqncia. Sugira uma forma simples de descobrir se a lista gerada por map um valor de retorno ou um side effect desta funo.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 65
4. Mdulos
Em algumas ocasies, voc poder precisar das mesmas funes vrias vezes em programas diferentes. Quando isso ocorrer, pode usar mdulos, que so como pacotes de funes. Vamos criar nosso primeiro mdulo, contendo quatro funes: uma que calcula a circunferncia de um crculo, outra que calcula a rea do crculo, outra que eleva uma base qualquer a uma potncia qualquer e uma funo intil. O procedimento para criar um mdulo o mesmo para escrever um programa completo: abra uma nova janela. No IDLE, v a File > New Window, ou simplesente CTRL+N. Digite as funes neste programa e salve-o com a extenso .py necessrio que seja num diretrio que esteja no path3. Para saber quais diretrios do seu computador esto no path, voc pode usar este pequeno script (que voc entender melhor como funciona ao final deste captulo): >>> import sys >>> for i in sys.path: print i Chamaremos nosso mdulo atual de circulos.py. #circulos.py def perimetro(r): """Esta funcao calcula o perimetro de 3 Sem muito rigor, o path uma relao de todos os diretrios em que Python vai procurar pelo seu mdulo. Se voc quiser, pode salvar seu mdulo na pasta Python23, em C:, que certamente estar no path. No entanto, evite salvar todos os seus mdulo l, para no sobrecarregar este diretrio.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 66
um circulo de raio r.""" try: r=float(r) return 2*3.14*r except: print 'O argumento da funcao deve ser um numero.' def area(r): """Esta funcao calcula a area de um circulo de raio r.""" try: r=float(r) return 3.14*(r**2) except: print 'O argumento da funcao deve ser um numero.' def potencia(x,y=2): """potencia(x,y) eleva x a potencia y. Default: y=2.""" try: return x**y except: print 'Argumentos invalidos' def nada(): """Esta funcao nao faz nada.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 67
So serve para mostrar doc strings.""" pass No pressione F5 para executar este programa! Pode fechar a janela, inclusive. Abra uma nova janela de IDLE (voc pode fazer isso pela janela do programa, em Run > Python Shell) e tente usar alguma das funes recm criadas. >>> potencia(2,3) Traceback (most recent call last): File "<pyshell#9>", line 1, in -toplevelpotencia(2,3) NameError: name 'potencia' is not defined A funo ainda no est definida no escopo deste programa. Uma das formas de usar as funes importando o mdulo: >>> import circulos Se voc salvou o mdulo na pasta C:/Python23 ou em qualquer outra que esteja no path, a importao do mdulo deve dar certo. Uma vez importado o mdulo, a sintaxe para utiliza-lo modulo.funo(parmetros)
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 68
Agora sim, podemos utilizar a funo potencia: >>> circulos.potencia(2,3) 8 >>> print circulos.potencia.__doc__ potencia(x,y) eleva x a potencia y. Default: y=2. >>> print circulos.nada.__doc__ Esta funcao nao faz nada. So serve para mostrar doc strings. >>> circulos.perimetro(33.) 207.24000000000001 >>> circulos.outrafuncao(1,2) Traceback (most recent call last): File "<pyshell#6>", line 1, in -toplevelcirculos.outrafuncao(1,2) AttributeError: 'module' object has no attribute 'outrafuncao' Este erro ocorre porque no existe a funo outrafuncao no mdulo circulos.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 69
Outra forma de usar as funes importa-las diretamente, em vez de importar o mdulo que as contm. Para evitar problemas, abra outra janela do IDLE. A forma agora : from mdulo import funo1, funo2, ... Se usarei as funes potencia e nada, >>> from circulos import potencia,nada >>> potencia(2,3) 8 >>> potencia(3) 9 >>> print potencia.__doc__ potencia(x,y) eleva x a potencia y. Default: y=2. >>> nada() Usando a sintaxe from modulo import * so importadas todas as funes do mdulo.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 70
>>> from circulos import * >>> perimetro(4.2) 26.376000000000001 >>> area(-1.) 3.1400000000000001 >>> print area.__doc__ Esta funcao calcula a area de um circulo de raio r. Uma das preciosidades de Python que esta linguagem possui milhares de mdulos para resolver muitos dos seus problemas. Por isso em Python deve-se ter cuidado para no reinventar a roda, isto , talvez voc perca um tempo precioso para desenvolver um programa que faa algo quando j existe um mdulo pronto que o faa. Antes de comear a escrever seu script, se voc suspeitar que algum talvez j tenha feito algo parecido, procure no Google ele seu amigo! (Nosso amigo: o Google escrito em Python!). Nas funes perimetro e area acima, usamos o valor aproximado de pi, 3.14. O valor preciso de pi uma das coisas que pode ser obtido do mdulo nativo math: >>> import math >>> pi Traceback (most recent call last): File "<pyshell#1>", line 1, in -toplevelpi NameError: name 'pi' is not defined >>> math.pi
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 71
3.1415926535897931 >>> from math import pi >>> pi 3.1415926535897931 Assim como as principais relaes matemticas: >>> from math import sin,cos >>> sin(pi) 1.2246063538223773e-016 >>> cos(pi) -1.0 >>> sin(cos(sin(pi))) 0.8414709848078965 >>> from math import * >>> log(1) # o log usado aqui o logaritmo neperiano (base e) 0.0 >>> log10(2) # log10, por sua vez, o logaritmo de base 10. 0.3010299956639812 >>> pow(2,4) 16.0 >>> print pow.__doc__ pow(x,y) Return x**y (x to the power of y).
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 72
Tambm h um similar cmath muito eficiente para trabalhar com nmeros complexos, e diversos outros. No endereo http://docs.python.org/modindex.html voc pode consultar a relao e descrio de todos os mdulos que j esto disponveis quando voc instala o interpretador de Python. No incio deste captulo usamos outro deles, o mdulo sys. Este mdulo contm uma funo path que retorna uma lista contendo todos os diretrios do seu computador que esto no path. D uma olhada novamente naquele pequeno cdigo da pgina 66 que agora voc entender melhor.
81 >>> x 1000 O nome x definido na primeira linha do script global, enquanto o x dentro da funo f local. Quando executamos f(argumento1), o interpretador tem que lidar com dois valores associados ao mesmo nome x: argumento1 e 1000. Qual deles o interpretador escolher vai depender da hierarquia. Neste caso, o nome x = argumento1 local (definido dentro da funo), enquanto x = 1000 global (definido ao longo do script). A hierarquia faz com que o interpretador reconhea argumento1 como sendo o valor associado ao nome x naquele momento e faz finalmente (argumento1)**2. Depois disso, o usurio consulta o valor de x. Agora s existe o valor global de x definido, que vale 1000. Outro exemplo, agora com nomes builtins: >>> type(12) <type 'int'> >>> def type(m): print 'Digitastes',m >>> type(12) Digitastes 12 >>> def tipologia(x): def type(y): print 'Voc digitou',y type(x)
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 74
>>> tipologia(12) Voc digitou 12 >>> type(12) Digitastes 12 Aqui, tnhamos o objeto builtin type, que nos retorna o tipo do objeto fornecido como argumento. Definimos globalmente uma nova funo type que executa um procedimento diferente e, dentro da funo tipologia (quer dizer, localmente), este nome foi redefinido para um terceiro procedimento, que s vale no escopo da funo. Ao usar a funo tipologia, sua funo interna type no se torna global. Globalmente, continua valendo a type que imprime Digitastes tal, em sobreposio builtin de mesmo nome. Sempre que aparece algum nome no programa que no foi definido localmente, o interpretador procura por ele entre os nomes globais. Se ele tambm no foi definido globalmente, o ltimo local de busca entre os nomes builtins. S ento informado o erro de que o nome no foi definido. Veja o exemplo. Dentro da funo, usamos um nome global e um builtin, alm dos nomes locais (argumentos da funo). Na segunda funo, usamos o nome no definido t, que gera um erro. Observe a ltima linha da mensagem de erro - O nome global 't' no definido. >>> from math import pi >>> r=2.3 >>> def area(incremento): return pi*(r+incremento)**2 >>> area(.7) 28.274333882308138 >>> def volume(inc):
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 75
return (pi*(t+inc)**3)/3. >>> volume(.4) Traceback (most recent call last): File "<pyshell#108>", line 1, in -toplevelvolume(.4) File "<pyshell#107>", line 2, in volume return (pi*(t+inc)**3)/3. NameError: global name 't' is not defined Por outro lado como j vimos, no possvel contar globalmente com os nomes definidos localmente. A seguir, define-se um nome t dentro do escopo da funo volume, que s existe localmente funo. Tentar acessar este nome globalmente tambm d erro. >>> def volume(inc): t=0.6 return (pi*(t+inc)**3)/3. >>> volume(.4) 1.0471975511965976 >>> t Traceback (most recent call last): File "<pyshell#112>", line 1, in -toplevel- t
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 76
NameError: name 't' is not defined Se for necessrio tomar um nome local de uma funo para uso no restante do script, possvel torn-lo global assim: >>> def volume(inc): global t t=0.6 return (pi*(t+inc)**3)/3. >>> volume(0.4) 1.0471975511965976 >>> t 0.59999999999999998 Deve-se usar este recurso com cautela e somente quando indispensvel. Transformar nomes locais em globais pode gerar conflitos com os nomes globais e builtins j existentes. Pelos mesmos motivos, evite usar a sintaxe from modulo import * para usar funes de qualquer mdulo, principalmente os que no foram criados por voc. Os mdulos podem conter dezenas de funes e valores, inclusive com nomes que voc desconhece, o que pode gerar conflitos com os nomes que voc definiu ao longo do script. Imagine o caso hipottico do sujeito que resolve definir uma funo sin
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 77
(uma abreviao inocente de sintaxe), e ao final do programa resolve calcular um cosseno qualquer, importando para isso todo o pacote math. >>> def sin(pronome): print 'D-%s ajuda!' %pronome >>> sin('lhe') D-lhe ajuda! >>> from math import * >>> cos(pi) -1.0 >>> sin('me') Traceback (most recent call last): File "<pyshell#131>", line 1, in -toplevelsin('me') TypeError: a float is required E assim se perde uma funo definida com tanto carinho... A perda da funo sin poderia ser evitada se o usurio tivesse importado o mdulo math e usado a sintaxe math.cos(math.pi) para calcular o cosseno de pi.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 78
Cuidado! Antes de executar a linha acima, v at sua pasta C: e verifique se voc j no tem um arquivo chamado tabela.txt. Se tiver, seu contedo ser apagado pelo cdigo anterior! Copie o arquivo para outra pasta. V sua pasta C e veja: acabou de ser criado um arquivo tabela.txt. Para fechar o arquivo, usa-se o mtodo close: >>> abertura.close() No se esquea: sempre que acabar de trabalhar com um arquivo, voc precisa fecha-lo antes de fazer qualquer coisa com ele. Se no for fechado, o arquivo continua como um objeto sendo usado por Python. Se voc tentar deletar o arquivo tabela.txt antes de executar abertura.close(), seu sistema operacional dar uma mensagem de violao de compartilhamento por causa disso. Se simplesmente abri-lo sem ter executado close, no conseguir ver o que acabou de ser escrito nele. Vamos digitar novamente a tabela de dlares, porm desta vez num arquivo que poderemos usar vrias vezes, imprimir, mandar por e-mail, etc. >>> abrindo=open('c:/tabela.txt','a') Podemos abrir no modo append porque este arquivo agora j existe, no ? Para escrever no arquivo aberto, usamos o mtodo write(s), sendo s uma string: >>> abrindo.write('Tabela de dlares\n') Usando os conceitos de formatao de strings fica assim:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 80
>>> abrindo.write('%s\n' %'Tabela de dlares') Isto serve ainda para reforar o que j vimos antes, que a formatao de strings no se aplica somente aps um print, mas tambm em qualquer lugar onde houver uma string. Dentro das funes write e raw_input, por exemplo. Ah! bom lembrar que a string '\n' corresponde a pular uma linha.
Estranho... Abrindo o arquivo tabela.txt que est em C:, no consigo ver o texto que acabei de escrever nele, Tabela de dlares. Voc sabe o que aconteceu? A resposta est na pgina anterior. >>> abrindo.close() >>> vamosdigitar=open('c:/tabela.txt','a') >>> for i in range(100): vamosdigitar.write('%i dlares valem %f reais\n' %(i,i*2.98)) >>> vamosdigitar.close() D uma olhada do contedo do arquivo tabela.txt em C:... Usamos agora o modo r para ler o contedo de um arquivo. >>> abertura=open('c:/tabela.txt','r')
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 81
H dois mtodos para ler o contedo do arquivo recm-aberto: readline l o contedo do arquivo linha por linha. Quanto acabar o arquivo, retorna '', uma string vazia. >>> abertura.readline() 'Tabela de d\xf3lares0' >>> abertura.readline() '0 dlares valem 0.00 reais' >>> abertura.close() O outro mtodo, readlines, retorna todo o contedo do arquivo em uma lista; cada termo da lista uma linha do arquivo. Readlines interessante porque gera uma lista, com a qual j sabemos trabalhar muito bem. >>> w=open('c:/tabela.txt','r') >>> lst=w.readlines() >>> print lst[0] Tabela de dlares >>> print lst[-1] 44 dlares valem 131.1 Voc pode escrever em diversos formatos alm de txt, mas alguns podem apresentar formatao insatisfatria. Se voc escrever num arquivo do MsExcel, por exemplo, todo o texto ser empacotado na primeira clula do arquivo.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 82
Entre os formatos interessantes para se escrever, est o .bat, em que voc pode escrever um cdigo de execuo do DOS, o .scr, em que voc pode escrever um script de AutoCAD, como veremos adiante, etc.
Voc tambm pode usar este recurso para abrir arquivos ou executar programas: >>> a('c:/tabela.txt') 0 E tambm pode usa-lo para causar problemas (cuidado!): >>> a('format c:') Deixando o terrorismo de lado, digamos que voc desenvolveu um programa com vrios mdulos que devem ser copiados para a pasta C:/Python23, alguns arquivos de dados em texto que devem ser copiados diretamente para C: e outros que ficariam melhor se estivessem bem vista, em INICIAR > PROGRAMAS, por exemplo, e salvou todos eles em um disquete para distribuir para seu cliente. Neste exemplo, temos: meumodulo.py deve ser copiado para C:/Python23, seno no funcionar; meuprograma.py o programa que supostamente vai usar o mdulo meumodulo.py. Este programa dever ficar na pasta C:/Python-NOME, onde NOME o nome do seu cliente; leiame.txt deve ser aberto para o usurio depois da instalao dos itens acima. Crie estes arquivos num disquete. No importa o que eles faam; podem ser arquivos vazios. Neste exemplo, importar somente o que faremos com eles.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 84
Agora precisamos criar um novo programa (este com contedo) que instalar todos os itens acima nos seus devidos lugares. Podemos chama-lo de instalar, ou setup, algo assim: #instalar.py import os comando=os.system comando('copy a:\meumodulo.py c:\Python23') NOME=raw_input('Seu nome: ') # Linha 4 NOME='Python-'+NOME comando('md c:\''+NOME) # Linha 6 comando('copy a:\meuprograma.py c:\''+NOME) abrir=os.startfile abrir('a:/leiame.txt') Cabe uma pequena explicao sobre a linha 6 do programa. O comando DOS md serve para criar um novo diretrio. Neste caso criamos um novo em C chamado Python-NOME; NOME o valor informado pelo usurio na quarta linha do programa. Com os recursos que voc viu neste ltimo captulo, j d pra criar um pequeno instalador, que espalha seus programas, mdulos e leia-mes nas pastas adequadas, inclusive abrindo para o usurio os arquivos de instruo de uso.
Agora com voc! Escreva um desintalar.py que apague todos os diretrios e programas que voc espalhou pelo computador do cliente!
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 85
3. Time e Random
Voc pode inserir a hora certa ou calcular o tempo de execuo do seu programa com o mdulo time. O mtodo asctime mostra a data e hora locais como uma string, enquanto localtime mostra esta informao em forma de tupla. >>> import time >>> time.asctime() 'Thu May 20 11:35:40 2004' >>> time.localtime() (2004, 5, 20, 11, 36, 45, 3, 141, 0) J temos certo traquejo com endereamento, ento podemos, entre outras coisas, fazer: >>> salve=time.localtime() >>> print 'Hoje ',salve[2],'do',salve[1],'de',salve[0] Hoje 20 do 5 de 2004 J o mtodo time mostra o tempo em segundos a partir de uma referncia meio estranha, no muito legvel para ns, terrqueos, mas interessante quanto trabalhamos com ele para medir variaes de tempo. Para medir quanto tempo demorou a execuo do seu programa, voc pode salvar o tempo inicial nas primeiras linhas de cdigo com
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 86
>>> from time import time >>> t0=time() e ao final do programa, >>> print 'A execuo durou', time()-t0, 'segundos' A execuo durou 50.6560001373 segundos O mdulo random pode ser til para fazer jogos e sorteios diversos. Dois mtodos muito interessantes so choice e random. O primeiro sorteia um elemento de uma seqncia, enquanto o segundo escolhe um nmero aleatrio entre 0 e 1. >>> import random >>> sorteio=random.choice >>> a=[] >>> for i in range(10): a.append(raw_input('Nome dos candidatos: ')) Nome dos candidatos: Woiski Nome dos candidatos: Barata Nome dos candidatos: Labaki Nome dos candidatos: Bestinha Nome dos candidatos: Osvaldo Nome dos candidatos: Fred Nome dos candidatos: Thiago
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 87
>>> a ['Woiski', 'Barata', 'Labaki', 'Bestinha', 'Osvaldo', 'Fred', 'Thiago'] >>> sorteio(a) 'Fred' >>> sorteio(a) 'Ricardo' >>> sorteio(a) 'Labaki' Estou com sorte hoje... Vejamos o mtodo random. O resultado gerado (R [0,1]) pode ser trabalhado normalmente multiplicado por um inteiro, transformado em inteiro, etc.: >>> import random >>> num=random.random >>> num() 0.8077947641664176 >>> num() 0.23901444156442075 >>> int(100*num()) 43 >>> for i in range(5): print int(10*num()), 0 5 3 9 4
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 88
rectangle 0,0 10,10 necessrio que voc pressione ENTER aps 10,10. como se estivesse pressionando ENTER na linha de comandos para confirmar este ponto final do retngulo. J imaginou como deve dar trabalho para desenvolver um script para desenhar 50 quadrados concntricos tendo cada um o dobro da rea de seu interno? Voc teria que calcular o incremento de rea de cada um e converter isso em termos de pontos coordenados; o script ficaria imenso. A entra nosso bom e velho Python: j sabemos como escrever o resultado de um programa num arquivo, inclusive .scr. Basta ento desenvolver um programa que calcule os pontos dos retngulos e escreva o script para ns. Uma das solues o programa abaixo: #retangulos.py import math raiz=math.sqrt A=float(raw_input('Area inicial: ')) w=open('c:/nosso_script.scr','w') for i in range(1,51): a=raiz(A) w.write('rectangle\n') pontoi=str(-a/2)+','+str(-a/2)+'\n' pontof=str(a/2)+','+str(a/2)+'\n'
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 90
w.write(pontoi) w.write(pontof) A=2*A w.write('Zoom\n') w.write('Extents\n') w.close() Analisemos o programa: Linha 4: assim criamos o arquivo nosso_script.scr em C:. Linha 6: a o comprimento da aresta do quadrado, igual raiz quadrada da sua rea. A rea do quadrado inicial informada pelo usurio. Linha 7: escrevemos o comando a ser interpretado pelo AutoCAD, rectangle e pulamos uma linha. Linha 8: escrevemos o ponto inicial do retngulo. A funo str transforma a/2 em uma string. Concatenamos a vrgula a essa string, concatenamos a coordenada y do ponto inicial, e concatenamos por fim o caracter para pular uma linha (\n). Faa um teste interativo usando o IDLE que voc entender melhor. Para uma rea inicial (A) igual a 20, a string resultante : '-2.2360679775,-2.2360679775\n' Linha 9: escrevemos o ponto final da mesma forma, porm as coordenadas agora so (a/2,a/2). Linha 10: escrevemos o ponto inicial do retngulo no arquivo (j com o comando para pular linha) e Linha 11: a mesma coisa com o ponto final. Linha 12: Duplicamos a rea para que no prximo item da varredura for o retngulo desenhado tenha o dobro da rea.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 91
Linha 13 e 14: Nestas linhas, ajustamos os retngulos desenhados tela, para facilitar sua visualizao.
Reescreva o programa para desenhar crculos concntricos. Tente faze-los com o dobro da rea do anterior, depois com o dobro do raio.
5. Msica, maestro!
T certo, no l um concerto sinfnico, mas seria interessante inserir uns sons do PC nos programas, no ? O mdulo winsound, e o mtodo, Beep(freqncia,durao), sendo freqncia a freqncia do som, em Hertz, no intervalo 37 32767, e a durao em milissegundos. Ambos os parmetros devem ser nmeros inteiros. >>> import winsound >>> b=winsound.Beep >>> b(1000,1000) >>> b(500,500) Ouviu? Pode ser til a funo sleep(s) do mdulo time, que d uma pausa de s segundos no programa. Para uma volta nostlgica aos tempos de SuperNintendo,
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 92
>>> import time >>> s=time.sleep >>> for i in range(1,5): for j in range(1,5): b(100*j*i,50) s(0.01) s(0.01)
Que tipo de erro ocorrer se em vez de range(1,5) colocssemos range(4)? Tente descobrir sem executar o cdigo.
6. Calendar
E para trabalhar com datas, o mdulo calendar mais eficiente que o time. O mtodo isleap(y), por exemplo, testa se o ano y bissexto4: >>> import calendar >>> calendar.isleap(2004) True >>> calendar.isleap(2055) False 4 Leap year: Ano Bissexto.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 93
O mtodo monthrange(x,y) retorna uma tupla contendo em que dia da semana (segunda-feira=1, tera-feira=2, etc.) comea o ms y do ano x e quantos dias tem esse ms. >>> calendar.monthrange(2004,5) (5, 31) E um muito til, prmonth(x,y), que imprime de forma bem comportada o calendrio do ms y do ano x. Para voc no se esquecer de quando aconteceu a Semana de Arte Moderna, a vai: >>> calendar.prmonth(1922,2) February 1922 Mo Tu We Th Fr Sa Su 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 94
teste=teste+1 if teste <> 0: print valor,'no primo' else: print valor,' primo' def testa_perfeito(gretagarbo): verifica=1 for qualquerum in range(1,gretagarbo): if gretagarbo % qualquerum == 0: verifica=verifica+qualquerum if verifica == gretagarbo: print gretagarbo,' perfeito' else: print gretagarbo,'no perfeito' def procura_divisores(n): lista_de_divisores=[] for i in range(1,n): if n % i == 0: lista_de_divisores.append(i) if len(lista_de_divisores)==0: print n,'No tem divisores' else: print 'Divisores de',n,':' for i in lista_de_divisores:
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 96
print i, def soma_algarismos(n): n=str(n) while len(n)>1: m=0 for i in n: m=m+int(i) n=str(m) print n n=int(raw_input('Digite o nmero a ser analisado: ')) testa_par(n) testa_primo(n) testa_perfeito(n) procura_divisores(n) soma_algarismos(n) Alguns comentrios sobre a funo soma_algarismos: na primeira linha da funo, transformamos n em uma string porque mais simples somar os termos de uma seqncia. At que n tenha somente um algarismo (while len(n) > 1), somamos os termos da string n (quarta e quinta linhas da funo) e salvamos esta soma novamente no nome n.
necessrio encapsular o programa inteiro dentro de um while 1: para que sejam testados vrios nmeros diferentes?
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 97
for i in mail: print i, w.write(i) w.close() Teste o programa: escolha algum e-mail que voc recebeu encaminhado (aqueles com subject repleto de Fwd, Re, etc.), pressione CTRL+A no seu navegador para selecionar tudo e CTRL+C para copiar e cole isto quando o programa pedir Mail:. Viu s quantos endereos so capturados? Valem alguns comentrios: o bloco de try tenta abrir o arquivo mailcollection.txt, que estar em C: se voc j usou este programa alguma vez. Se der errado (dar, se o arquivo ainda no existir), ento um novo arquivo com este nome criado. Na sexta linha vemos um mtodo de strings, split(w), que serve para separar uma string nos caracteres w e colocar os termos separados numa lista. O caracter w o espao em branco por default. >>> 'araraquara'.split('a') ['', 'r', 'r', 'qu', 'r', ''] Assim cortamos a string araraquara em todos os caracteres a que aparecem, e colocamos os termos separados em uma lista. >>> a='Boa tarde!' >>> a.split() ['Boa', 'tarde!']
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 99
J na oitava linha usamos o mtodo a.replace(b,c) que substitui todos os valores b da string a por c: >>> a='Boa tarde!' >>> a.replace('Boa','Pssima') 'P\xe9ssima tarde!' Finalmente, na linha 11 aparece sequencia1.__contains__(termo1). Contains serve para verificar se h termo1 em sequencia1. >>> a='Good Night Vietnam' >>> a.__contains__('Vie') True >>> a.split().__contains__('Vie') False >>> a.split() ['Good', 'Night', 'Vietnam'] >>> a.split().__contains__('Vietnam') True
comportada as colunas alinhadas, as linhas espaadas, mesmo nmero de casas decimais dos valores, etc. ordem=int(raw_input('Ordem da matriz: ')) matriz=[] print 'Digite os termos da matriz A' for i in range(ordem): matriz.append([]) for j in range(ordem): termo='A'+str(i+1)+str(j+1) matriz[i].append(float(raw_input('Termo'+termo+': '))) print '\n' print 'Matriz A\n' for k in range(ordem): for w in range(ordem): print '%7.2f' %matriz[k][w], print '\n' Neste programa temos um bloco de for dentro de outro. Para cada valor de i, for i insere na lista matriz uma linha vazia (na verdade, uma lista vazia). O for j (que executado para cada valor de i por causa da indentao!) adiciona os n termos a esta linha e o for i ainda imprime uma linha em branco (o print '\n' tambm est indentado sob o for i). J a linha print 'Matriz A\n' no ser impressa para cada valor de i porque no est mais na indentao de for. A indentao tambm acontece no ltimo bloco de for. As trs ltimas linhas do programa so executadas para cada valor de k: o bloco de for w imprime uma
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 101
linha da matriz (fixa o valor de k e varia w de 0 a ordem-1). Ento terminado o for w, uma linha em branco impressa e o processo se repete para o prximo valor de k.
if morse[i][j]=='.': Beep(500,50*vel) sleep(0.1*vel) elif morse[i][j]=='-': Beep(500,150*vel) sleep(0.1*vel) else: sleep(0.3*vel) sleep(0.3*vel) Conseguiu descobrir? Este programa um gerador de cdigo Morse. Ele traduz o texto do usurio para traos e pontos, e tambm toca a traduo. O processamento de verdade est todo neste bloco de for. Toda a parte anterior, que ocupa mais da metade do programa, s pr-processamento, quando recebemos os dados do usurio e informamos ao programa o dicionrio morse, que o alfabeto Morse retirado do site WikiPedia. O primeiro for varre a string informada pelo usurio e executa os procedimentos indentados debaixo dele para cada letra dessa string. O primeiro procedimento imprimir esta letra (representada por i) e o valor associado chave i no dicionrio morse. Por exemplo, se o texto do usurio amerindio, o primeiro valor assumido por i a. Ento ser impresso a e morse[a], que vale .- (veja o dicionrio). O bloco for j varrer o valor associado chave i transformando pontos (.) em sons curtos e hfens (-) em sons com o triplo da durao, todos eles com 500 Hz. A ltima linha, sleep(0.3*vel), uma pausa de 0.3*vel segundos dada ao final do bloco de for j, ou seja, entre uma letra e outra do texto. O else do bloco de if identifica os valores do dicionrio que no so nem pontos nem hfens (os espaos) e tambm d a mesma pausa que dada entre uma letra e outra. Certamente mais fcil entender o cdigo olhando pra ele do que lendo este pargrafo... , comparado com Python, at o Portugus difcil!
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 103
xx +
1
= xf )( )('
n
from math import * from os import startfile from time import time,localtime print 'Operadores especiais disponveis:' print 'sin, cos, log (neperiano),' print 'log10 (base 10), e sqrt (raiz quadrada).\n' funcao=raw_input('Funo: ') derivada=raw_input('Derivada: ') x0=float(raw_input('Valor inicial: ')) t0=time()
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 104 k k k xf
Aqui comea o processamento de verdade, por isso imediatamente antes marcamos o tempo para conferir depois quanto tempo levou a resoluo do problema. f='x-(('+funcao+')/('+derivada+'))' Esta string f contm a equao geral de Newton-Raphson, j mostrada. w=open('c:/newton-raphson.txt','w') w.write(' Newton-Raphson\n\n') if localtime()[3]<=12: w.write(' Bom dia!\n\n') elif 12<localtime()[3]<18: w.write(' Boa tarde!\n\n') else: w.write(' Boa noite!\n\n') As linhas anteriores abrem o arquivo newton-raphson.txt em C:, que conter os nossos dados de sada, e escreve nele o ttulo e um pequeno cumprimento ao usurio. Voc se lembra, o mtodo localtime() retorna uma tupla, cujo endereo 3 contm a hora local. w.write('Procedimentos para encontrar a raiz de '+funcao+'\n\n') w.write('Valor inicial: '+str(x0)+'\n\n')
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 105
Aqui escrevemos com que funo estamos trabalhando e tambm o valor inicial escolhido. Como x0 um float, precisamos transforma-lo numa string antes de concatena-lo string Valor inicial: e por fim string '\n\n' (duas linhas em branco). x=x0 for p in range(10): w.write('Iterao '+str(p+1)+':\n') x=eval(f) w.write(str(x)+'\n') w.write('\n') E assim escrevemos os valores obtidos em 10 iteraes sobre a equao de Newton-Raphson. A funo eval(f) avalia a string f desde que ela contenha o termo x e se anteriormente tivermos definido um valor para x. Veja um exemplo (as prximas cinco linhas no fazem parte do programa newton-raphson!): >>> string='2+a+b' >>> a=1 >>> b=5 >>> eval(string) 8 Continuando o programa... O valor atual de x corresponde a x
10
obtido pelo Mtodo e um float. Se quisermos escreve-lo no arquivo, devemos transforma-lo tambm em string. Perceba
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 106
que sempre somamos '\n' ao final das strings usadas pelo mtodo write. Se no fizermos isso, o prximo texto ser empilhado ao final deste e o arquivo ficar ilegvel. w.write('Valor aproximado da raiz: '+str(x)+'\n\n') E como time() retorna o tempo atual, time()-t0 nos dar o tempo decorrido deste que definimos t0: w.write('Tempo para clculos: '+str(time()-t0)+'\n') No h mais nada para escrevermos no arquivo, ento o fechamos usando o mtodo close(), e o abrimos para o usurio usando startfile. w.close() startfile('c:/newton-raphson.txt') Faamos o script para plotar o grfico da funo. Para facilitar sua busca pelo script, j salvaremos este arquivo na pasta que aberta por default quando voc executa Run Script no AutoCAD. Se voc estiver usando uma verso do AutoCAD ou do Windows que no seja a 2000, pode haver pequenas variaes neste endereo. t=open('c:/Arquivos de Programas/ACAD2000/graficos.scr','w') J que a funo range s gera uma lista de inteiros, criamos uma nova funo arange que gera uma lista de floats, por motivos que veremos adiante.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 107
def arange(xi,xf,i): a=[] while xi<=xf: a.append(xi) xi=xi+i return a O script color serve para mudar a cor usada pelo AutoCAD. A cor 20, por exemplo, o vermelho. t.write('color\n') t.write('20\n') Escrevemos ento o comando que desenhar o grfico, e logo depois os pontos que a linha ter que percorrer, exatamente (x,f(x)), para x variando do valor aproximado da raiz at o valor inicial x0. Depois um Zoom e pronto! t.write('line\n') funcao=funcao.replace('x','elemento') for elemento in arange(x,x0,0.1): t.write(str(elemento)+','+str(eval(funcao))+'\n') t.write('\n') t.write('Zoom\n') t.write('Extents\n') t.close()
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 108
Aps executar o programa, no AutoCAD v a Tools > Run Script e execute graficos.scr.
import random r=random.random print 'Memoria\n' print 'Nao joque usando IDLE\n' print 'Digite todos os numeros que forem' print 'informados pelo programa' Aqui comea o processamento. O objeto escolha um inteiro entre 0 e 9.
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 109
escolha=int(10*r()) print escolha a=str(escolha) user=str(int(raw_input('Tente: '))) Aps receber a primeira tentativa do usurio, o programa testa se ele acertou e comea o jogo de memria at que o jogador erre. while user==a: escolha=int(10*r()) print escolha a=a+str(escolha) user=str(int(raw_input('Tente: '))) print '\n'*100 A linha print '\n'*100 imprime 100 linhas em branco que limpam a tela de forma que o usurio no consegue ver as escolhas feitas anteriormente pelo jogo. Assim, ter que recorrer memria mesmo. por isso que este jogo no pode ser executado no IDLE, porque rolando a tela o jogador poderia ver os nmeros sorteados anteriormente. Se a condio do while no for mais satisfeita, quer dizer que a ltima resposta dada pelo usurio tem ao menos um termo no correspondente srie de nmeros que o computador escolheu at o momento, ou seja, print 'Voce perdeu!'
JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 110
E o usurio pode saber quanto foi seu ltimo acerto: print 'Seu saldo:',len(a) Como este programa ser rodado no prompt do DOS e no pelo IDLE, assim que terminar toda a impresso anterior a janela ser fechada, e o usurio no saber seu saldo. Assim, para manter a janela aberta podemos fazer raw_input('Pressione ENTER para sair')
Parte IX Exerccios
C
Fceis
F
Mdios
C
Por que no exemplo numanalise, a funo testa_primo faz uma varredura sobre range(2,...) e testa_perfeito faz uma varredura sobre range(1,...)? Por que ambas no fazem os testes sobre range(x), por exemplo?
C
Aproveitando os recursos do mdulo calendar, desenvolva um programa para calcular quantos dias de vida um usurio est completando hoje, se informado o dia de nascimento.
F
Em vez de imprimir nmeros, modifique o exemplo memoria para tocar o nmero sorteado. Se o computador escolher 4, sero tocados 4 bips, e assim por diante. E o usurio continua informando nmeros. Faa tambm com que seja salvo um ranking de jogadores num arquivo de texto e com que o jogador seja informado sempre que bater um recorde. Melhor ainda se o recorde for acompanhado com msica (veja o som da pg. 53). JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 111
F
Desenvolva um banco de dados simples de forma que o usurio possa inserir um produto e seu preo, alm de poder consultar o preo de um produto, tudo isso sem ter que acessar o arquivo de texto.
F
Leitura instrumental a tcnica com a qual se pode compreender uma leitura em um idioma desconhecido aprendendo apenas as palavras mais utilizadas do idioma. O rol das palavras mais utilizadas feito por pesquisa em centenas de livros famosos no idioma, que obviamente no feita manualmente. Desenvolva um programa que vasculhe a obra Os Lusadas e retorne um arquivo com o ranking das 200 palavras mais utilizadas neste livro. Voc pode encontrar um e-book de Os Lusadas facilmente na internet.
F
Tente prever que resultados sero gerados pelo seguinte script, sem execut- lo: x, y = 5, 3 print x print y def one(): x = 1 print x two(x) print x def two(y): print y y = 2 print y z = one y = z() print x print y JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 112
Vem a...
Como dissemos na primeira parte deste mdulo, Python uma linguagem orientada a objetos, mas que permite a programao procedural. O que voc viu at agora a parte mais bsica, em que Python se assemelha um pouco s outras linguagens de script. agora, no Mdulo B Python Orientado a Objetos, que vem o pulo do gato. Principalmente se voc j programa usando este paradigma, vai se surpreender com a facilidade com que a nossa linguagem (agora sua tambm!) se integra com a Orientao a Objetos (OO). Programar orientado a objetos, em Python, como respirar! Quando seus projetos comearem a tomar grandes propores, a OO importante para garantir a estabilidade e facilidade de manuteno do cdigo, por exemplo. A OO tambm evita as montanhas de variveis globais que causam conflitos como os que vimos na nossa discusso sobre escopo. No prximo mdulo voc tambm entender o que significam aquelas sintaxes do tipo modulo.funo, lista.append, string.split, etc. E finalmente, a Orientao a Objetos ser uma introduo indispensvel para o Mdulo C a to esperada programao de interfaces grficas. Nos vemos no Mdulo B! Sua participao muito importante para a constante melhoria deste material. Ficarei muito honrado em conhecer suas opinies, sugestes, crticas ou dvidas sobre ele. Para isso, posso ser encontrado em labaki@feis.unesp.br. Agradeo a todos que j colaboraram, em especial ao Prof. E. R. Woiski, Ph.D, pelas observaes e a toda a lista python-brasil por diversas mensagens que ajudaram a melhorar muitos pontos ao longo do texto.
J. Labaki
labaki@feis.unesp.br http://labaki.tk
Este documento pode ser distribudo livremente, desde que mantidos os crditos do autor. JOSU LABAKI GRUPO PYTHON UNESP ILHA SOLTEIRA 113