Mudanças entre as edições de "PRG-2011-1-sobral"

De MediaWiki do Campus São José
Ir para navegação Ir para pesquisar
(Criou página com 'Sistemas Operacionais e Introdução à Programação - 2010-2 = Informações gerais = ''Professor'': Marcelo Maia Sobral <br>''Email'': msobral@gmail.com <br>''Skype'': mm-so...')
 
 
(118 revisões intermediárias por 2 usuários não estão sendo mostradas)
Linha 1: Linha 1:
Sistemas Operacionais e Introdução à Programação - 2010-2
+
Introdução à Programação - 2011-1
  
 
= Informações gerais =
 
= Informações gerais =
Linha 6: Linha 6:
 
<br>''Email'': msobral@gmail.com
 
<br>''Email'': msobral@gmail.com
 
<br>''Skype'': mm-sobral
 
<br>''Skype'': mm-sobral
<br>''Lista de email (forum)'': [http://groups.google.com/group/sop-ifsc sop-ifsc@googlegroups.com]
+
<br>''Lista de email (forum)'': [http://groups.google.com/group/prg-ifsc prg-ifsc@googlegroups.com]
<br>''Atendimento paralelo'': 2a feira 10h - 11 h, 3a feira 10h - 12 h ou 16h - 17 h, 5a feira 16h - 17h (no Laboratório de Desenvolvimento de Tele)
+
<br>''Atendimento paralelo'': 4a feira 10:00 - 11:30 h, 5a feira 8:00 - 9:30 h e 13:30 - 15:30 h (no Laboratório de Desenvolvimento de Tele)
 
<br>''Reavaliação (recuperação)'': no final do semestre
 
<br>''Reavaliação (recuperação)'': no final do semestre
 +
<br> '''IMPORTANTE:''' o direito de recuperar uma avaliação em que se faltou somente existe mediante justificativa reconhecida pela coordenação. Assim, deve-se protocolar a justificativa no prazo de 48 horas, contando da data e horário da avaliação, e aguardar o parecer da coordenação. O não cumprimento desse procedimento implica a impossibilidade de fazer a recuperação, e assim a reprovação na disciplina.
  
* [http://www.sj.ifsc.edu.br/~msobral/SOP/plano.html Plano da disciplina]
+
* [http://tele.sj.ifsc.edu.br/~msobral/SOP/plano.html Plano da disciplina]
 
* [[Sistemas_Operacionais_e_Introdu%C3%A7%C3%A3o_a_Programa%C3%A7%C3%A3o|Ementa e bibliografia]]
 
* [[Sistemas_Operacionais_e_Introdu%C3%A7%C3%A3o_a_Programa%C3%A7%C3%A3o|Ementa e bibliografia]]
  
 
== Softwares ==
 
== Softwares ==
  
Será usado como plataforma de estudo o sistema operacional Ubuntu Linux 10.04 LTS. Para obtê-lo há essas opções:
+
Será usado como plataforma de estudo o sistema operacional Ubuntu Linux 10.04 LTS ou posterior. Para obtê-lo há essas opções:
  
 
# Trazer um CD-R virgem para que eu faça a cópia aqui no IFSC
 
# Trazer um CD-R virgem para que eu faça a cópia aqui no IFSC
Linha 26: Linha 27:
  
 
== Listas de exercícios ==
 
== Listas de exercícios ==
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/listas/lista1-2010-2.pdf 1a lista (parte teórica)]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/listas/lista1-pratica-2010-2.pdf 1a lista (parte prática)]
 
  
 
== Referências adicionais ==
 
== Referências adicionais ==
  
* [[Sistemas Operacionais e Introdução a Programação (diário 2010-1) - Prof. Tiago Semprebom|Diário de aula do Prof. Tiago Semprebom]]
+
* [[PRG-2011-1-tiago|Diário de aula do Prof. Tiago Semprebom]]
* [http://www.sj.ifsc.edu.br/~msobral/SOP/ Antiga página da disciplina (2009-2)]
+
* [http://tele.sj.ifsc.edu.br/~msobral/SOP/ Antiga página da disciplina (2009-2)]
 
* Valle, Odilson Tadeu.  [[Media:Gerencia_de_redes.pdf|Gerência de Redes]]. IFSC - Unidade São José. 2009. ''(ver capítulos 1 a 9)''
 
* Valle, Odilson Tadeu.  [[Media:Gerencia_de_redes.pdf|Gerência de Redes]]. IFSC - Unidade São José. 2009. ''(ver capítulos 1 a 9)''
 
* [https://help.ubuntu.com/community Ubuntu Documentation]
 
* [https://help.ubuntu.com/community Ubuntu Documentation]
Linha 40: Linha 38:
 
* Demais referências contidas na [[Sistemas Operacionais e Introdução a Programação (página)#Sobre_a_Disciplina|página principal de SOP]].
 
* Demais referências contidas na [[Sistemas Operacionais e Introdução a Programação (página)#Sobre_a_Disciplina|página principal de SOP]].
  
= Sistemas operacionais =
+
= Aulas =
  
== 27/07: Apresentação ==
+
== 16/03: Introdução ==
  
Apresentação da disciplina: plano de ensino, avaliação, abordagem pedagógica.
+
'''Apresentação da disciplina:''' plano de ensino, avaliação, abordagem pedagógica.
  
== 02/08: Introdução a sistemas computacionais ==
+
'''Assunto:''' Lógica de programação
  
* Sistemas computacionais [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula1.pdf (resumo)]:
+
''Referências auxiliares:''
** Visão geral de sistemas operacionais
+
* [http://wiki.sj.ifsc.edu.br/images/0/02/L%C3%B3gica_de_Programa%C3%A7%C3%A3o.pdf Apostila auxiliar], escrita pelo prof. Paulo Sérgio de Moraes.
** Histórico de sistemas operacionais
+
* Ver [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula10.pdf transparências].
  
=== Introdução ao Linux ===
 
  
Uso básico da interface gráfica e de texto.
+
[http://www.sj.ifsc.edu.br/~msobral/SOP/slides/pao.png Um primeiro algoritmo]
Apostila [[Media:Gerencia_de_redes.pdf|Gerência de Redes]] (prof. Odilson), capítulos 3 a 5, e capítulos [http://www.guiafoca.org/guia/iniciante/ch-cmdd.html 6] e [http://www.guiafoca.org/guia/iniciante/ch-cmd.html 7] do Guia Foca Linux Iniciante.
 
  
'''Objetivos:'''
 
# Conhecer o sistema operacional Linux
 
# Usar a interface gráfica
 
# Usar a interface de linha de comando (shell)
 
# Usar comandos básicos para manipulação de arquivos e diretórios
 
# Conhecer a estrutura de diretórios do Linux
 
  
* [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/aula2.pdf Roteiro da aula]
+
''Videos sobre algoritmos:''
 +
* [http://www1.folha.uol.com.br/folha/videocasts/ult10038u689083.shtml Resolvendo o cubo de Rubik (cubo mágico)]
 +
** [http://www.sj.ifsc.edu.br/~msobral/SOP/coisas/1285780.flv Cópia desse vídeo aqui no Ifsc]
 +
* [http://www.youtube.com/watch?v=k0xgjUhEG3U Outro algoritmo ...]
  
==== Exercício ====
+
=== Problemas exemplo ===
  
# Partindo do subdiretório /home/aluno/Desktop, e usando somente caminhos (''pathnames'') relativos, indique a sequência de comandos necessária para visitar sucessivamente os seguintes subdiretórios:
+
'''Problema dos três recipientes'''
#* /home/aluno
 
#* /home/visita
 
#* /home/visita/Documentos
 
#* /home/visita/public_html/imagens
 
#* /etc/default
 
#* /usr/share/docs
 
#* /usr/share/docs/python/manual/html
 
#* /usr/lib/browser-plugins
 
#* /usr/local/etc
 
#* /usr/local/man/man1
 
#* /home/aluno
 
#* /home/aluno/Desktop
 
#Partindo do subdiretório /home/aluno/Desktop, informe  a sequência de subdiretórios (indique-os com seus caminhos absolutos) visitados se forem seguidos os seguintes caminhos relativos:
 
#* ../Imagens/Fotos
 
#* 2010/07
 
#* ../../2009
 
#* ../../../bin
 
#* ../../manuel/docs
 
#* ../../../etc/init.d
 
#* ../rc5.d
 
#* ../../var/log/apache2
 
#* ../../tmp
 
#* ../../lib/modules
 
#* ../../usr/bin
 
#* ../lib/X11
 
#* ../..
 
  
=== Videos ilustrativos ===
+
Há três recipientes com tamanhos distintos: um com 8 litros, outro com 5 litros e o terceiro com 3 litros. O recipiente com 8
 +
litros está completamente cheio. Deseja-se colocar 4 litros em dois recipientes. Considere que os recipientes não são graduados.
  
* [http://www.youtube.com/watch?v=6X2B8Z_DCo0&feature=related Eniac]
+
'''Problema da travessia'''
* [http://www.google.com.br/search?source=ig&hl=pt-BR&rlz=1G1GGLQ_PT-BRBR356&=&q=saga+do+windows&btnG=Pesquisa+Google&meta=lr%3D A Saga do Windows]
+
* [http://www.youtube.com/watch?v=nV6CReLELwU Funcionamento de um processador]
+
Um barqueiro precisa levar um saco de milho, uma galinha e uma raposa para o outro lado do rio. Porém o barco somente é capaz de levar uma coisa de cada vez (além do barqueiro). Qual a sequência de travessias necessário para atravessar o milho, a galinha e a raposa ?
* [http://www.youtube.com/watch?v=Y7U8M6UsEwE&feature=related Funcionamento de um disco rígido]
 
* [http://www.youtube.com/watch?v=9QePnWV7ztE&feature=related Outro video sobre funcionamento de um disco rígido]
 
  
=== Textos interessantes sobre história dos sistemas operacionais ===
+
'''Torres de Hanoi'''
  
* [http://cm.bell-labs.com/cm/cs/who/dmr/cacm.html The Unix Operating System:] um artigo publicado em 1974 pelos criadores do Unix, Ken Thompson e Dennis Ritchie.
+
Há três hastes. Uma das hastes serve de suporte para três discos de tamanhos diferentes. Um disco menor sempre é colocado sobre um disco maior. A figura abaixo ilustra as hastes e os discos:
* [https://netfiles.uiuc.edu/rhasan/linux/ History of Linux (inglês)]
 
* [http://www.brasilescola.com/informatica/historia-do-linux.htm História do Linux (português)]
 
* [http://pt.wikipedia.org/wiki/Linux Visão geral sobre o Linux]
 
* [http://en.wikipedia.org/wiki/History_of_Microsoft_Windows História do Microsoft Windows (inglês)]
 
* [http://en.wikipedia.org/wiki/History_of_operating_systems História dos Sistemas Operacionais (inglês)]
 
* [http://www.baixaki.com.br/info/2031-a-historia-dos-sistemas-operacionais.htm História dos Sistemas Operacionais (em português)]
 
* [[Media:History_of_operating_system_Moumina.pdf| History of Operating Systems (inglês)]]
 
* [http://www.pcworld.com/article/162866/the_10_worst_operating_systems_of_all_time.html Os 10 piores sistemas operacionais de toda a história (inglês)]
 
  
== 03/08: Introdução ao Linux (continuação) ==
+
[[imagem:Hanoi.png]]
  
*Uso básico da interface gráfica e de texto.
+
Desejam-se mover os três discos para a haste da direita. Porém só pode se mover um disco por vez, e um disco maior nunca pode ficar sobre um disco menor.
**Apostila [[Media:Gerencia_de_redes.pdf|Gerência de Redes]] (prof. Odilson), capítulos 3 a 5
 
**Capítulos [http://www.guiafoca.org/guia/iniciante/ch-cmdd.html 6] e [http://www.guiafoca.org/guia/iniciante/ch-cmd.html 7] do Guia Foca Linux Iniciante.
 
  
'''Objetivos:'''
+
Operação possível: '''Move''' ''disco'' para ''haste''
# Conhecer o sistema operacional Linux
 
# Usar a interface de linha de comando (shell)
 
# Usar comandos básicos para manipulação de arquivos e diretórios
 
# Conhecer a estrutura de diretórios do Linux
 
# Usar comandos básicos para visualização e manipulação de processos
 
  
* [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/aula2.pdf Roteiro da aula]
+
Qual a sequência de operações para mover os discos de uma haste para outra ?
* [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/ex-dirs.tgz Arquivo para o exercício de reorganização de arquivos]
 
*  [[media:Guia_rapido_nocat.txt|Guia rápido de instalação do NoCat (arquivo texto demonstrativo)]]
 
  
Foram vistos [http://wiki.ubuntu-br.org/ComandosBasicos#head-cd7ef0b9f7aeda593cbb2946b24f8fbb086a6a71 programas para manipulação de arquivos]:
+
=== Leitura adicional ===
* '''cat:''' mostra o conteúdo de um arquivo na tela, porém sem paginá-lo.
 
* '''more:''' mostra de forma paginada o conteúdo de um arquivo na tela.
 
* '''less:''' mostra de forma paginada o conteúdo de um arquivo na tela, com alguns recursos a mais (possibilita paginar pra cima ou pra baixo)
 
* '''cp:''' copia um ou mais arquivos
 
* '''mv:''' move um ou mais arquivos de um subdiretório para outro, ou renomeia um arquivo
 
* '''rm:''' remove um ou mais arquivos
 
* '''mkdir:''' cria um subdiretório
 
* '''rmdir:''' remove um subdiretório
 
  
=== Exercício ===
+
* [http://cs-exhibitions.uni-klu.ac.at/index.php?id=193 História dos algoritmos]
 +
* [http://en.wikipedia.org/wiki/Algorithm Uma definição de algoritmos na Wikipedia]
  
# Usando a interface gráfica, organize os arquivos contidos [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/sop.tar.bz2 aqui]. Eles devem ser colocados nos subdiretórios ''Imagens'', ''Videos'' e ''Documentos'', de acordo com seus tipos.
+
== 17/03: Algoritmos ==
# Refaça a questão anterior, porém usando a interface de linha de comando (o ''shell'').
 
  
''Obs: para extrair os arquivos do exercício faça assim:''
+
Para criar algoritmos, deve-se primeiro entender o que são instruções e sequências lógicas;
<syntaxhighlight lang=bash>
 
aluno@DX~$ tar xjf sop.tar.bz2
 
</syntaxhighlight>
 
  
... e veja que aparecerá um subdiretório ''sop'' com todos os arquivos do exercício lá dentro.
+
O jogo [http://armorgames.com/play/2205/light-bot LightBot] mostra de uma forma divertida como criar pequenos algoritmos. Nesse jogo o robô deve se movimentar de acordo com uma '''sequência''' de '''instruções elementares''' definidas pelo jogador. Apesar de sua simplicidade, ele pode se tornar bastante desafiador ! Até que fase desse jogo você consegue chegar ?
  
== 09/08: Processos e memória ==
+
[[imagem:Lightbot.png]]
  
[http://www.sj.ifsc.edu.br/~msobral/SOP/aula1.pdf Visão geral sobre Sistemas Operacionais]:
+
=== Desenhando figuras geométricas ===
* O que são ?
 
* Para que servem ?
 
  
'''Processos e multiprogramação:''' uma visão geral sobre programas, processos, ciclos de um processo, multiprogramação e escalonamento. Ver [[Media:Gerencia_de_redes.pdf|apostila do prof. Odilson]], capítulo 3, e [http://www.guiafoca.org/guia/iniciante/ch-run.html capítulo 5] do Guia Foca Linux Iniciante.
+
# [http://scratch.mit.edu/ Scratch] é um software educacional para ajudar no ensino de matemática, geometria e introdução à programação. Com ele pequenos programas podem ser escritos de foram visual, com instruções representadas por blocos que se encaixam como Lego. Ele possibilita fazer desenhos facilmente, seguindo um programa com instruções de desenho. Use-o para desenhar estas figuras:
 
+
#* triângulo equilátero
'''Gerência de memória:''' visão geral sobre o uso de memória no sistema operacional e pelos processos
+
#* triângulo isósceles
Uso do laboratório para ilustrar conceitos.
+
#* quadrado
 +
#* hexágono
 +
#* quadrado com vértices interligados
 +
#* Um círculo
  
* Continuação do [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/aula2.pdf roteiro da aula de 26/02].
+
[[imagem:Scratch.png]]
* [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/lab2 Programas usados para investigar o escalonamento de processos]
 
** '''fominha:''' tenta usar todo o tempo de processador disponível
 
** '''crash:''' cria processos indefinidamente
 
** '''lento:''' processo que trabalha pouco, ficando a maior parte do tempo ocioso
 
  
Vários utilitários (programas auxiliares) existem para obter informações do sistema operacional sobre processos e memória. Alguns trabalham em modo texto, como:
 
  
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/ps.1.html ps]:''' lista os processos existentes
+
* Agora tente algo mais avançado:
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/pstree.1.html pstree]:''' lista os processos existentes mas de forma hierárquica
+
** Desenhe um triângulo equilátero, porém perguntando na tela qual o tamanho do lado.
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/top.1.html top]:''' lista os processos mais ativos, junto com informações globais sobre uso dos recursos no sistema operacional (memória, processador, memória virtual, quantidade de processos, carga de trabalho)
+
** repita o desenho do triângulo, porém fazendo-o lentamente: espere 1 segundo após desenhar cada lado do triângulo.
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/atop.1.html atop]:''' o mesmo que '''top''', mas com maior detalhamento do uso de recursos mantidos pelo sistema operacional
+
** Faça com que um pequeno traço se movimente de um lado a outro do palco.
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/mpstat.1.html mpstat]:''' mostra estatísticas  de uso do processador
+
** Faça com que um pequeno traço se movimente em círculo indefinidamente.
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man1/free.1.html free]:''' mostra o uso de memória
+
** Faça com que uma bola role pela horizontal. Insira quatro raios na bola para ajudar no efeito visual de movimento.
* '''[http://manpages.ubuntu.com/manpages/karmic/en/man8/vmstat.8.html vmstat]:''' mostra o uso de memória, discos e processador no último intervalo de tempo.
 
  
Existem também utilitários no modo gráfico. Por exemplo, no Ubuntu há o "Monitor do sistema":
+
=== Atividade para casa ===
  
[[Imagem:Monitor1.png|600px|center]]
+
Faça as figuras geométricas indicadas na aula !
<center>''Para executar o Monitor do Sistema</center><br><br>
 
  
 +
== 23/03: Começando os primeiros algoritmos ... ==
  
[[Imagem:Monitor2.png|400px]] [[Imagem:Monitor3.png|400px]]
+
'''Algoritmo''' é uma sequência lógica de instruções, a qual possui um início e um fim.
<center>''Telas do monitor do Sistema</center>
 
  
== 10/08: Sistemas de arquivos; redirecionamentos de entrada e saída padrão; pipes ==
+
'''Sequência lógica''' é uma  sequência correta de instruções capaz de gerar o resultado esperado.
  
Resolver a 1a lista de exercícios:
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/listas/lista1-2010-2.pdf parte teórica]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/listas/lista1-pratica-2010-2.pdf parte prática]
 
  
Visão geral sobre sistemas de arquivos: definição de arquivo e diretório, formas de armazenamento em media (discos, DVD, ...), segurança (controle de acesso).
+
A escrita ou projeto de algoritmos se faz com:
 +
* ''Estruturas de controle:'' comandos que modificam  uma sequência de instruções
 +
* ''Estruturas de dados:'' dados (valores) usados e modificados durante a execução do algoritmo.
  
=== Alguns utilitários de uso comum ===
 
  
Esses utilitários são programas úteis para operações do dia-a-dia com arquivos, e serão úteis para ilustrar os conceitos de entrada e saída padrão:
+
Para introduzir esses conceitos, nada melhor que um  pequeno projeto ...
  
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/cut.1.html cut:] divide as linhas dos arquivos em colunas, e mostra colunas específicas
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/grep.1.html grep:] mostra linhas de arquivos que contenham determinada palavra ou padrão de caracteres
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/sort.1.html sort:] ordena as linhas de um arquivo
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/paste.1.html paste:] combina linhas de arquivos
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/wc.1.html wc:] conta linhas, palavras e caracteres
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/tail.1.html tail:] mostra as últimas linhas de um arquivo
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/head.1.html head:] mostra as primeiras linhas de um arquivo
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/du.1.html du:] mostra a utilização de disco de um determinado arquivo ou diretório
 
  
=== Entrada e saída padrão ===
+
=== Projeto: um jogo muito simples ===
  
Todo processo possui uma saída padrão, que corresponde a um arquivo ou dispositivo onde os dados de saída do processo (ex: mensagens de texto) serão mostrados. E assim como existe uma saída padrão, todo processo possui uma entrada padrão que corresponde ao arquivo ou dispositivo de onde por ''default'' são obtidos os dados de entrada.  
+
Você conhece o jogo clássico [http://pt.wikipedia.org/wiki/Space_Invaders Space Invaders] ?
  
[[imagem:Sop-redir.png|800px]]
+
[[imagem:Spaceinvaders.jpg]]
  
=== Redirecionamento de saída padrão ===
 
  
Normalmente a saída padrão de um processo é a tela do terminal, mas ela pode ser redirecionada para um arquivo ou para outro dispositivo. Assim, as mensagens de texto que um processo gera podem ser guardadas em um arquivo para posterior utilização. No exemplo abaixo, a listagem dos processos foi guardada no arquivo ''processos.txt'':
+
Pros padrões atuais esse é um jogo muito simples. Você conseguiria reproduzi-lo, mesmo que simplificado, no Scratch ? Ele poderia começar assim:
  
<syntaxhighlight lang=bash>
+
[[imagem:Jogo1-scratch.gif]]
aluno@D1:~$ ps ax > processos.txt
 
aluno@D1:~$
 
</syntaxhighlight>
 
  
O redirecionamento de saída padrão se faz com o símbolo '''>''' seguido do nome do arquivo. Note que nada aparece na tela, uma vez que a saída se tornou o arquivo ''processos.txt''. Após executar o comando acima, veja o conteúdo de ''processos.txt'':
 
  
<syntaxhighlight lang=bash>
+
# O robô se movimenta na horizontal, comandado pelas teclas ''seta pra direita'' e ''seta pra esquerda''.
aluno@D1:~$ less processos.txt
+
# O helicóptero se movimenta de um lado a outro constantemente, refletindo  nas bordas.
  PID TTY      STAT  TIME COMMAND
+
# Ao teclar espaço,  um tiro sai do robô em direção ao topo do palco
    1 ?        Ss    0:03 /sbin/init
+
# Se o tiro acertar o helicóptero, o jogo termina.
    2 ?        S<    0:00 [kthreadd]
 
    3 ?        S<    0:00 [migration/0]
 
    4 ?        S<    0:00 [ksoftirqd/0]
 
    5 ?        S<    0:00 [watchdog/0]
 
    6 ?        S<    0:00 [migration/1]
 
    7 ?        S<    0:00 [ksoftirqd/1]
 
    8 ?        S<    0:00 [watchdog/1]
 
    9 ?        S<    0:00 [events/0]
 
  10 ?        S<    0:00 [events/1]
 
  11 ?        S<    0:00 [khelper]
 
  12 ?        S<    0:00 [kstop/0]
 
  13 ?        S<    0:00 [kstop/1]
 
  14 ?        S<    0:00 [kintegrityd/0]
 
  15 ?        S<    0:00 [kintegrityd/1]
 
  16 ?        S<    0:00 [kblockd/0]
 
  17 ?        S<    0:00 [kblockd/1]
 
  18 ?        S<    0:00 [kacpid]
 
  19 ?        S<    0:00 [kacpi_notify]
 
  20 ?        S<    0:00 [cqueue]
 
  21 ?        S<    0:00 [ata/0]
 
  22 ?        S<    0:00 [ata/1]
 
processos.txt
 
  
</syntaxhighlight>
 
  
Como se pode ver, o texto que apareceria na tela foi guardado em ''processos.txt''. Experimente executar novamente o comando ''ps ax > processos.txt'' e veja o resultado. O que aconteceu com o arquivo ''processos.txt'' ?
+
O que você precisa para fazer esse jogo ?
 +
# Movimento do helicóptero.
 +
# Movimento do robô comandado pelo teclado.
 +
# Acionamento do tiro.
 +
# Detecção se o tirou acertou o helicóptero.
  
Quando se deseja redirecionar a saída de um processo para um arquivo, porém preservando o conteúdo original desse arquivo, deve-se executar o comando da seguinte forma:
+
'''Mãos à obra !'''
  
<syntaxhighlight lang=bash>
+
== 24/03: Introduzindo algoritmos: variáveis e expressões ==
aluno@D1:~$ ps ax >> processos.txt
 
aluno@D1:~$
 
</syntaxhighlight>
 
  
... quer dizer, em vez de usar '''>''' usa-se '''>>'''.
+
Variáveis são usadas em algoritmos para guardar dados. Dados podem ser:
 +
* Números inteiros ou reais
 +
* Letras ou textos
 +
* Valores lógicos (booleanos)
 +
* ... e outros que vocês descobrirão futuramente !
  
==== Exercícios ====
+
Uma variável possui um identificador, que nada mais é que seu nome. No Scratch as variáveis são criadas e podem ser modificadas com os comandos categorizados em ''Variáveis'' (comandos laranjas):
  
# Concatene vários arquivos de texto, e usando redirecionamento de saída padrão grave o resultado em um novo arquivo. ''Dica:'' para concatenar use o utilitário '''cat'''.
 
# Execute o comando '''ps''' ''> ps.txt''. Veja o conteúdo do arquivo resultante ''ps.txt''. Execute novamente '''ps''' ''> ps.txt'', e veja como ficou o arquivo ''ps.txt''. O que você pode concluir quanto ao redirecionamento de saída para um arquivo já existente ?
 
# Repita o procedimento do ítem 2, porém executando primeiro '''ps''' ''> ps.txt'' e depois '''ps''' ''>> ps.txt''. Qual foi a diferença ?
 
  
=== Redirecionamento de entrada padrão ===
+
[[imagem:Scratch-var1.png]]
  
Normalmente a entrada padrão corresponde ao teclado do terminal, mas pode ser redirecionada para outro arquivo ou dispositivo. No exemplo abaixo, usa-se o programa '''wc''' para contar as linhas, palavras e caracteres contidos no arquivo ''processos.txt'':
 
  
<syntaxhighlight lang=bash>
+
Variáveis podem ser ''globais'' ou ''locais'':
aluno@D1:~$ ps ax > processos.txt
+
* ''Globais:'' são visíveis e podem ser usadas por qualquer objeto do Scratch.
aluno@D1:~$ wc < processos.txt
+
* ''Locais:'' são visíveis e podem ser usadas somente pelo objeto do Scratch ao qual está vinculada.
137  810 7807 processos.txt
 
aluno@D1:~$
 
</syntaxhighlight>
 
  
No exemplo acima, gravou-se em ''processos.txt'' o resultado do comando '''ps ax''' , e depois usou-se '''wc'' para contar linhas, palavras e caracteres desse arquivo. O efeito combinado é a contagem de quantos processos existem no sistema nesse momento (representado pelo número de linhas contidas em ''processos.txt'').
+
Variáveis podem ser usadas em qualquer comando que precise de um valor numérico, como por exemplo o comando para mover um certo número de passos. No exemplo abaixo, um programa faz com que um objeto desenhe um triângulo no palco, sendo que o comprimento do lado do triângulo é informado pelo usuário.
  
Ao contrário do caso da saída padrão, não é tão comum se usar redirecionamento de entrada padrão. No entanto há um recurso adicional provido pelo sistema operacional que explora a combinação de ambas, e que possibilita combinar as funcionalidades de diferentes programas.
+
[[imagem:Scratch-var2.png]]
  
=== Pipes ===
 
  
Em sistemas operacionais Unix, é possível conectar a saída padrão de um processo à entrada padrão de outro processo, e a isto se chama ''pipe'' (que pode ser traduzido como ''duto'' ou ''tubo''). Com isto, os dados de saída de um processo serão os dados de entrada de outro processo, o que pode ser explorado para realizar diferentes tarefas. Por exemplo, retomando a contagem de processos vista na seção anterior:
+
Variáveis podem ser usadas em ''expressões aritméticas'', que realizam algum cálculo com valores numéricos. No Scratch expressões são criadas combinando-se os blocos contidos em ''Operadores''(blocos verdes-claros):
  
<syntaxhighlight lang=bash>
 
aluno@D1:~$ ps ax | wc
 
137  810 7807
 
aluno@D1:~$
 
</syntaxhighlight>
 
  
O símbolo de ''pipe'' é a barra vertical '''|'''. Ao se executarem os comandos acima unidos pelo ''pipe'', a listagem de processos gerada pelo '''ps ax''' é automaticamente enviada para a entrada padrão do comando '''wc'''. O resultado é a contagem de processos existentes no sistema.
+
[[imagem:Scratch-operadores1.png]]
  
Podem-se ligar mais de dois processos usando ''pipes''. Assim, cria-se um encadeamento de processos, em que a saída padrão de um processo alimenta a entrada padrão do próximo processo. Por exemplo, para se contarem os processos do usuário aluno:
 
  
<syntaxhighlight lang=bash>
+
Usando esses blocos podem-se calcular novos valores usando variáveis ou constantes. Por exemplo, um pequeno programa que desenha um polígono qualquer precisa calcular o ângulo de quina da figura dependendo de sua quantidade de lados:
aluno@D1:~$ ps aux | grep aluno | wc
 
    47    568    5195
 
aluno@D1:~$
 
</syntaxhighlight>
 
  
  
 +
[[imagem:Scratch-operadores2.png]]
  
== 16/08: Localização de arquivos ==
 
  
Uma tarefa usual em um  sistema operacional é localizar arquivos. Essa procura pode ser baseada no nome ou parte do nome de arquivo, na data de criação ou última modificação, no nome do usuário ou grupo dono, ou outros atributos  do arquivo. Pode ser inclusive  baseada numa combinação dessas informações. Por exemplo,  pode ser necessário procurar todos os arquivos de música ou video, ou então todos os programas (arquivos executáveis) que pertençam ao superusuário (usuário ''root''). Existem programas capazes de fazer tais buscas tanto no ambiente gráfico quanto no modo texto.  
+
Usando esses conceitos sobre variáveis e expressões, faça as atividades a seguir.
  
As buscas a serem  realizadas são:
+
=== Pong ===
# Arquivos de música
 
# Arquivos de música ou video
 
# Arquivos de texto ou planilha do Openoffice que não tenham sido modificados a 7 dias ou mais
 
# Arquivos executáveis (programas)
 
# Arquivos executáveis pertencentes ao usuário ''root''
 
  
Para iniciar, vamos usar programas gráficos para fazer buscas.
+
Outro jogo clássico chamado [http://pt.wikipedia.org/wiki/Pong Pong] mostrava uma bola que refletia em uma parede, voltando e devendo ser interceptada por uma pequena base (que funcionava como raquete) comandada pelo jogador. Se conseguir interceptá-la, a bola é refletida de novo em direção à parede oposta, continuando assim o jogo. Se falhar, o jogo termina. Esse é considerado o primeiro videogame lucrativo a ter sido criado ...
  
=== Usando o ambiente gráfico ===
+
[[imagem:Pong1.jpg]]
  
O primeiro programa gráfico a ser usado é o ''nautilus'', o gerenciador de arquivos disponível no Ubuntu Linux. Para usá-lo para fazer buscas selecione a opção de pesquisa:
 
  
 +
Tente fazer uma versão do Pong usando o Scratch, mas com duas pequenas modificações. Há apenas uma base, que fica no canto inferior do palco. O placar conta quantas vezes a bola tocou o chão (i.e. o jogador falhou em interceptá-la), e se chegar a 0 o jogo termina. A segunda modificação é uma aceleração gradual da bola, que deve deve aumentar de velocidade a cada 5 segundos de jogo.
  
[[imagem:Pesquisa1.png|600px]]
 
  
 +
Para que serão necessárias variáveis nesse jogo ?
  
No ''nautilus'' aparecerá uma caixa de texto  onde se  deve digitar o nome ou parte do nome dos arquivos a serem procurados.
+
==== O problema da reflexão da bola na barra ====
  
 +
Em aula vimos que para fazer a reflexão da bola com a barra deve-se calcular o ângulo de reflexão. No caso da barra estar na horizontal, esse ângulo deve ser calculado assim:
  
[[imagem:Pesquisa2.png|600px]]
+
<math>NovaDirecao = 180^0 - Direcao</math>
  
 +
No Scratch isso deve ser feito combinando-se o comando "Aponte para direção" e uma expressão para calcular a nova  direção. A direção atual do objeto está contida em sua variável local ''direção'' (predefinida nos comandos de ''Movimentos''):
  
Tente fazer as buscas usando esse programa. Há alguma que não pode ser feita ?
 
  
==== Buscas avançadas ====
+
[[imagem:Scratch-reflexao.png]]
  
Para fazer buscas que combinem outros critérios que não somente o nome do arquivo deve-se usar outro aplicativo, que está acessível pelo Mmenu Locais:
 
  
[[imagem:pesquisa3.png|400px]]
+
==== Uma outra versão do Pong ====
  
Esse aplicativo de busca mostra uma tela em que se adicionam ou removem critérios de busca:
+
Em uma outra versão ainda encontrada, no canto superior há um conjunto de alvos que devem ser acertados pela bola. Quando se acertarem todos os alvos o jogo termina.
  
[[imagem:pesquisa4.png|400px]]
 
  
Como resultado, podem-se localizar arquivos e diretórios que atendam diferentes opçõs de localização:
+
[[imagem:Pong2.jpg]]
  
[[imagem:pesquisa5.png|400px]]
+
=== Para casa: um jogo de sinuca ===
  
=== Modo texto ===
+
Para exercitar o uso de variáveis vamos criar também um jogo de sinuca. Nesse jogo deve-se usar a bola branca para encaçapar uma outra bola colorida. As bolas se movimentam de forma natural, desacelerando gradualmente devido ao atrito com a mesa. Quando as bolas colidem, suas velocidades e direções se modificam, dependendo de seus valores anteriores e do ângulo de colisão. Quando as bolas batem nos cantos das mesas, devem ser refletidas. Como as bolas predem velocidade gradualmente, dependendo de suas velocidades iniciais e das colisões em algum momento irão fatalmente parar. As caçapas ficam nos dois cantos superiores do palco. Você notará que sem o uso de variáveis, não será possível criar esse jogo, ainda mais se quiser acrescentar mais alguns detalhes.
  
A procura de arquivos e diretórios pode ser automatizada com o comando '''find'''. Por exemplo, para localizar todos os
+
As tacadas se fazem com um taco representado por uma linha reta, cuja ponta sempre está colada à bola branca. A direção do taco pode ser controlada pelo mouse ou por teclas direita e esquerda. Para fazer uma tacada, deve-se pressionar a tecla seta pra baixo durante um tempo, e então soltá-la. Quanto mais tempo se pressionar essa tecla mais forte será a tacada, e assim mais veloz a bola branca vai disparar. Assim, para realizar tacadas com pouca velocidade deve-se pressionar a tecla seta pra baixo por pouco tempo, e para tacadas mais fortes, em que a bola branca dispara com grande velocidade, deve-se pressionar essa tecla por um tempo maior. Pode-se usar algum objeto e colocá-lo na lateral do palco para informar qual a potência da tacada a ser realizada.
arquivos ou diretórios chamados ''default'', e que estejam abaixo de ''/etc'':
 
  
<syntaxhighlight lang=bash>
+
Alguns dados serão necessários para calcular as movimentações das bolas:
aluno@D1:~$ find /etc -name default -print
+
* Peso de cada bola: 100 g
/etc/default
+
* Diâmetro das bolas: 8 cm
/etc/calendar/default
+
* Desaceleração das bolas devido ao atrito com a mesa: <math>0.1 ~m/s^2</math>
/etc/X11/xinit/xinput.d/default
+
* Velocidade mínima de tacada: 0 m/s
aluno@D1:~$
+
* Velocidade máxima de tacada: 4 m/s
</syntaxhighlight>
 
  
Uso do '''find''' :
+
== 30/03: Algoritmos: Usando variáveis, expressões e estruturas de decisão ==
  
<syntaxhighlight lang=bash>
+
'''Estruturas de decisão:''' blocos de controle usados quando devem-se executar comandos somente se uma determinada condição for verdadeira. Por exemplo, no jogo Pong a bola deve ser refletida se tocar na barra controlada pelo jogador:
find pathname [opões]
 
</syntaxhighlight>
 
  
... sendo ''pathname'' o diretório abaixo de que se deseja fazer a busca. As opções servem para selecionar arquivos e diretórios durante a busca, e também para executar ações para cada um deles que for encontrado. Algumas opções mais comuns:
+
[[imagem:Prg-se3.png]]
  
* '''-name nome:''' seleciona arquivos ou diretórios cujos nomes sejam '''''nome'''' . Essa opção aceita ''wildcards'' (* e ?)
+
Esse tipo de estrutura de decisão se chama ''Se condição então faça algo''. Como indica sua descrição, uma ou mais instruções serão executadas caso a condição seja verdadeira. No exemplo acima, a condição é ''tocando em barra'', e as instruções executados caso isso seja verdade são ''aponte para a direção 180-direção''.
* '''-type tipo:''' seleciona apenas o '''''tipo'''' indicado, que pode ser:
 
** '''d:''' diretório
 
** '''f:''' arquivo
 
** '''l:''' ''link'' simbólico (ou atalho)
 
* '''-perm modo:''' seleciona apenas arquivos ou diretórios que possuam as permissões indicadas em ''modo''
 
* '''-size tamanho:''' seleciona apenas arquivos com tamanho de acordo com o especificado (em bytes)
 
* '''-mtime n:''' seleciona apenas arquivos ou diretórios modificados a '''''n * 24''''' horas atrás
 
* '''-atime n:''' seleciona apenas arquivos ou diretórios acessados a '''''n * 24''''' horas atrás
 
* '''-user usuario:''' seleciona apenas arquivos ou diretórios pertencentes a '''''usuario'''''
 
* '''-exec comando \;:''' executa '''comando''' para cada arquivo ou diretório encontrado. Ex:
 
** ''find /tmp -type f -mtime +7 -exec rm -f {} \;'' : remove todos os arquivos dentro de /tmp que estejam há 7 dias sem serem modificados. A sequência '''''{}''''' é substituída pelo nome do arquivo encontrado.
 
  
... e muitas outras (consulte o [http://manpages.ubuntu.com/manpages/karmic/en/man1/find.1.html manual do find]).
 
  
=== Atividade ===
+
No exemplo do tiro ao helicóptero, inspirado no jogo Space invaders, o jogo deveria terminar quando o tiro acertasse o alvo. Assim, as instruções para término do jogo precisavam estar dentro de uma estrutura ''Se condição então ...'', como mostrado abaixo:
  
Usando o modo gráfico E o modo texto,  faças as seguintes pesquisas por arquivos:
 
  
# Encontre o arquivo ''lsusb'', que se encontra em algum  subdiretório dentro de ''/usr''
+
[[imagem:Prg-se1.png]]
# Encontre os arquivos dentro de ''/usr'' cujos tamanhos sejam maiores que 20 MB
 
# Encontre os subdiretórios ''lib'', que estão dentro de ''/usr''
 
# Localize todos os arquivos executáveis (i.e., os programas) que existem abaixo de ''/lib''
 
# Localize todos os arquivos pertencentes ao usuário ''aluno''  e que estejam dentro de ''/tmp''
 
# Localize o arquivo ''bits.h'' em toda a árvore de diretório (quer dizer, a partir da raiz, ou  ''/'').
 
# Localize todos os arquivos abaixo de ''/tmp'' que tenham sido criados (ou modificados) a um dia ou mais.
 
# Localize os arquivos dentro de ''/tmp'' que pertençam a seu usuário.
 
# Localize somente os diretórios dentro de ''/tmp'' que pertençam a seu usuário.
 
# Localize somente arquivos dentro de ''/tmp'' que tenham menos de 1 kB.
 
# Localize somente arquivos dentro de ''/tmp'' que tenham mais de 10 kB.
 
# Localize somente arquivos dentro de ''/tmp'' que tenham mais de 10 kB e sejam do seu usuário.
 
# Localize somente arquivos dentro de ''/tmp'' que tenham mais de 10 kB e não sejam do seu usuário.
 
# Localize os arquivos dentro de /etc que contenham o texto ''telnet''.<syntaxhighlight lang=bash>
 
find /etc -type f -exec grep -q telnet {} \; -print
 
</syntaxhighlight>
 
# Localize os arquivos dentro de /etc que não contenham o texto ''root''.<syntaxhighlight lang=bash>
 
find /etc -type f -not -exec grep -q root {} \; -print
 
</syntaxhighlight>
 
# Localize os arquivos dentro de /etc que contenham o texto ''telnet'', porém não tenham o texto ''vt'' nas mesmas linhas que ''telnet''.
 
# Conte quantos arquivos em /etc contêm o texto ''root''.
 
# '''''(Desafio)''''' Em seu diretório ''home'' (pasta pessoal) existem alguns arquivos de video obtidos com sua câmera fotográfica, porém você não sabe exatamente onde estão. Além disso, esses videos são codificados na máquina fotográfica com o codec MJPG, que gera arquivos muito grandes. Se forem recodificados para XVID, os tamanhos desses videos  serão reduzidos substancialmente (4 vezes ou mais). Sabendo que se pode recodificá-los no modo texto com o programa '''mencoder''' (usando-o assim: '''''mencoder''' -o novo_video.avi -oac mp3lame -ovc xvid -xvidencopts bitrate=1200 arquivo_video''), faça a busca e conversão desses videos para XVID.
 
  
== 17/08: Redirecionamentos, pipes, usuários e grupos ==
 
  
Uso de redirecionamentos e pipes. Para entender a utilidade desses recursos, alguns comandos adicionais serão vistos:
+
Para ilustrar o uso de ''Se condição então ...'', e o teste de condições em geral, hoje serão feitos dois projetos:
  
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/cut.1.html cut:] divide as linhas dos arquivos em colunas, e mostra colunas específicas
+
# A continuação do jogo Pong, em que a bola deve acertar alvos no topo da tela. Além disso, se a bola tocar no chão mais que 3 vezes o jogador perde.
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/grep.1.html grep:] mostra linhas de arquivos que contenham determinada palavra ou padrão de caracteres
+
#* [http://tele.sj.ifsc.edu.br/~msobral/prg/pong0.sb Uma versão inicial do jogo Pong]
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/sort.1.html sort:] ordena as linhas de um arquivo
+
# Um cronômetro a ser visualizado no palco.
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/paste.1.html paste:] combina linhas de arquivos
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/wc.1.html wc:] conta linhas, palavras e caracteres
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/tail.1.html tail:] mostra as últimas linhas de um arquivo
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/head.1.html head:] mostra as primeiras linhas de um arquivo
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/du.1.html du:] mostra a utilização de disco de um determinado arquivo ou diretório
 
  
Conceitos de usuários e grupos. Comandos relacionados:
+
== 31/03:  Algoritmos: ainda estruturas de decisão e também de repetição ==
  
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/whoami.1.html whoami:] mostra quem é o usuário dono do processo atual (que é o shell)
+
Por diversas vezes criaram-se programas no Scratch em que o comportamento de um objeto corresponde a repetir uma sequência de comandos. Por exemplo, cada alvo no topo da tela no jogo Pong deve sumir quando for tocado pela bola:
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/users.1.html users:] mostra os usuários logados no sistema
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/who.1.html who:] mostra os usuários logados no msistema, porém com mais informações sobre cada um deles
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/w.1.html w:] mostra o que os usuários logados estão fazendo agora
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/finger.1.html finger:] mostra informações sobre um ou mais usuários
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/passwd.1.html passwd:] muda a senha de um usuário
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/last.1.html last:] mostra o histórico de usuários logados
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/chfn.1.html chfn:] muda diversas informações de um usuário (aquelas visíveis com '''finger''')
 
* [http://manpages.ubuntu.com/manpages/lucid/en/man1/groups.1.html groups:] grupos a que um usuário pertence
 
  
As informações sobre os usuários estão guardadas no arquivo ''/etc/passwd''. Mas podem estar armazenadas também na rede ...
+
[[imagem:Prg-repeticao1.gif]]
As informações sobre os grupos estão guardadas no arquivo ''/etc/group''.
 
  
=== Atividade ===
+
Assim, ele deve sempre testar se a bola o está tocando, e quando isso acontecer ele deve desaparecer e contabilizar que há uma alvo a menos para ser acertado. Como esse teste deve ser feito repetidamente até que a bola o acerte, torna-se necessário usar uma estrutura de repetição. No caso do Scratch, usou-se o comando ''Sempre'', que faz com que todos os comandos nele contidos sejam executados indefinidamente (no caso, até que a bola toque o alvo). Além dessa parte do programa, o teste se todos os alvos foram acertados também implica a repetição de uma sequência de comandos.
  
# Mostre quais os usuários que existem, junto com seus nomes completos (essa informação está no arquivo /etc/passwd)
+
No jogo, a bola deve se movimentar pelo palco, refletindo nas paredes e sendo rebatida pela barra até que todos os alvos tenham sido acertados.
# Repita o item anterior, mas mostrando em ordem alfabética direta e reversa
+
Portanto, os comandos que implicam controlar seu movimento devem se repetir seguidamente até que essa condição de término seja alcançada. O comportamento da bola pode desta forma ser descrito simplificadamente pelos comandos mostrados abaixo, em que se usa a variável ''bananas'' para indicar quantos alvos faltam para serem acertados. A repetição foi obtida com o comando ''repita até bananas=0'', e dessa forma a repetição se interrompe quando a variável ''bananas'' tiver o valor 0.
# Repita o ítem anterior, mas mostrando apenas o último usuário em ordem alfabética, e depois o primeiro usuário
 
# Conte quantos usuários têm o nome completo iniciando com a letra 'A' <syntaxhighlight lang=bash>
 
# O caractere ^ na frente da palavra faz com o grep a procure somente no início da linha
 
cut -d: -f5 /etc/passwd | grep ^A | wc -l
 
</syntaxhighlight>
 
# Seja um arquivo com o conteúdo abaixo, que contém uma relação de pessoas com alguns dados pessoais (nome, idade, sexo, profissão, residência):<syntaxhighlight lang=text>
 
Joaquim Silva 34 M comerciante Imaruim
 
Vanessa Duarte 28 F engenheira Biguaçu
 
Manuel Rosa 61 M médico Anitápolis
 
Sergio Queiroz 22 M estudante São José
 
Diana Marques 37 F advogada Palhoça
 
</syntaxhighlight>
 
## Ordene-o alfabeticamente pelo nome
 
## Ordene-o numericamente pela idade <syntaxhighlight lang=bash>
 
# Imaginando que o arquivo acima se chame dados.txt
 
sort -n -k 3 dados.txt
 
</syntaxhighlight>
 
##* '''''Desafio:''''' Como se poderia resolver esse problema se os nomes das pessoas tivessem quantidades variáveis de componentes ? Ex:<syntaxhighlight lang=text>
 
Joaquim Koch Silva 34 M comerciante Imaruim
 
Vanessa Duarte 28 F engenheira Biguaçu
 
Manuel João da Silva Rosa 61 M médico Anitápolis
 
Sergio Felipe Queiroz 22 M estudante São José
 
Diana Marques 37 F advogada Palhoça
 
</syntaxhighlight>''Dica: você poderia modificar as linhas desse arquivo de forma a delimitar claramente onde está a idade de cada pessoa. Se conseguir fazer algo parecido com o mostrado a seguir, o problema seria mais facilmente resolvido:'' <syntaxhighlight lang=text>
 
Sergio Queiroz: 22 :M estudante São José
 
Vanessa Duarte: 28 :F engenheira Biguaçu
 
Joaquim Silva: 34 :M comerciante Imaruim
 
Diana Marques: 37 :F advogada Palhoça
 
Manuel Rosa: 61 :M médico Anitápolis
 
</syntaxhighlight>''Veja o programa '''sed'''''.
 
## Mostre somente os dados das mulheres
 
## Conte quantos homens e quantas mulheres há nesse arquivo <syntaxhighlight lang=bash>
 
# Mulheres ...
 
grep " F " dados.txt | wc -l
 
  
# Homens ...
+
[[imagem:Prg-repeticao2.gif]]
grep " M " dados.txt | wc -l
 
</syntaxhighlight><br> ... ou ... <syntaxhighlight lang=bash>
 
# Mulheres ...
 
grep -w F dados.txt | wc -l
 
  
# Homens ...
+
Um terceiro exemplo diz respeito à criação de um cronômetro. Uma variável pode ser decrementada uma vez por segundo, até que o cronômetro zere. Um exemplo de como fazer isso segue abaixo:
grep -w M dados.txt | wc -l
 
</syntaxhighlight>
 
## Mostre os dados da primeira pessoa, de acordo com a ordem alfabética
 
## Mostre os dados da última pessoa, de acordo com a ordem alfabética
 
# Imagine que seu diretório de trabalho está muito cheio, e você quer identificar que arquivos ou subdiretórios estão ocupando mais espaço. Combine os comandos '''du''' e '''sort''' para descobrir isto.
 
# Conte quantos processos pertencem ao usuário ''aluno'', e quantos pertencem a ''root''.
 
# Sejam dois arquivos como mostrado abaixo. Combine suas linhas e mostre o resultado em ordem alfabética.<br><syntaxhighlight lang=text>
 
abacaxi
 
laranja
 
morango
 
manga
 
goiaba
 
limão
 
</syntaxhighlight><br><syntaxhighlight lang=text>
 
graviola
 
kiwi
 
maracujá
 
coco
 
pêssgo
 
</syntaxhighlight>
 
# Sejam dois arquivos como mostrado abaixo. Combine suas linhas e mostre o resultado em ordem alfabética. Mostre também o resultado em ordem de quantidade. <br><syntaxhighlight lang=text>
 
abacaxi
 
laranja
 
morango
 
manga
 
goiaba
 
limão
 
</syntaxhighlight><br><syntaxhighlight lang=text>
 
1 kg
 
500 g
 
100 g
 
2 kg
 
1 kg
 
300 g
 
</syntaxhighlight>
 
  
== 23/08: Editores de texto ==
+
[[imagem:Prg-repeticao4.png]]
  
Ver capítulo 9 da [[Media:Gerencia_de_redes.pdf|apostila do prof. Odilson]].
 
  
Os editores a serem vistos servem pra criar e modificar arquivos de texto simples (contêm apenas texto e sem formatação). Esse tipo de arquivo é muito comum para arquivos de programa, manuais rápidos, e documentos escritos em linguagens de formatação de texto, como HTML e XML.
+
Como se vê, estruturas de repetição são muito úteis para repetir sequências de comandos. Existem estruturas de repetição incondicionais, que repetem indefinidamente (como o comando ''sempre'' do Scratch), e condicionais, que repetem enquanto determinada condição for verdadeira ou até que determinada condição se verifique (ex: comando ''repita até''). Com elas se evita que programas sejam desnecessariamente longos ... na verdade sem elas seriam inviáveis muitos tipos de programas !
  
Alguns desses editores:
+
=== Atividade ===
  
* [http://library.gnome.org/users/gedit/stable/index.html.pt_BR gedit]: um editor de texto simples para o ambiente gráfico. Possui recurso para ressaltar palavras e sentenças, chamado ''syntax highlight'', de acordo com o tipo de texto (se programa em linguagem C, C++, Python, PHP, shll script, ou um texto HTML, entre outros).
+
# Adicione o cronômetro na tela do seu jogo Pong. O cronômetro deve ser composto de dois dígitos.
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/vi.1posix.html vi]: um editor de programas com recursos avançados de edição. Mas usa uma interface textual com comandos que devem ser memorizados ... algo anacrônico, mas funciona. Como motivação para usá-lo, praticamente todo computador com Unix (não somente Linux) vem com esse editor previamente instalado.
+
# O Pong poderia ficar mais difícil se os alvos fossem móveis e fugissem da bola. Assim, quando sentisse a bola se aproximando um alvo poderia acelerar ou reduzir sua velocidade, visando evitar uma colisão. Modifique então o Pong para que os alvos movam na horizontal de uma lado para o outro, e evitem a colisão com a bola.
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/nano.1.html nano]: um outro editor mais simples, também para modo texto e com menos funcionalidades que o '''vi''', porém um pouco mais amigável.
 
  
=== gedit ===
+
== 06/04: Portugol ==
  
Para executar o ''gedit'' deve-se selecioná-lo no menu de aplicativos:
+
'''Trabalho:''' deve-se entregar o jogo Pong com o cronômetro, conforme especificado na [[PRG-2011-1-sobral#Atividade|última aula sobre Scratch]].
  
[[imagem:Gedit.png|600px]]
+
'''Prazo de entrega:'' ''14/04/2011''
  
 +
----
 +
Os algoritmos que iremos criar serão compostos de:
 +
* ''Estruturas de controle:'' estruturas para controlar a sequência de execução (realização) de algoritmos:
 +
** ''Decisão:'' para realizar testes de forma a decidir que sequências de instruções devem ser executadas
 +
** ''Repetição:'' para repetir sequências de instruções
 +
* ''Estruturas de dados:'' estruturas para representar os dados a serem  processados por algoritmos
 +
** ''Variáveis e constantes''
  
Esse editor é bastante simples, porém útil para a edição de perquenos textos e programas. Algumas funções existentes são busca e substituição de palavras e impressão. Outras funcionalidades podem ser adicionadas por ''plugins''.
+
=== Portugol ===
  
[[imagem:Gedit2.png|600px]]
+
As aulas de Lógica de Programação usarão um software de auxílio ao ensino de algoritmos chamado [http://orion.ipt.pt/%7Emanso/Portugol/ Portugol], desenvolvido na [http://www.dei.estt.ipt.pt/portal/ Escola Superior de Engenharia do Instituto Politécnico de Tomar], em Portugal.
 +
* [http://www.sj.ifsc.edu.br/~msobral/SOP/portugol.html Guia rápido de instalação e utilização do Portugol].
  
Experimente editar diferentes tipos de arquivos de texto com o ''gedit'':
+
==== Guia rápido de instalação e utilização do Portugol ====
* [http://www.sj.ifsc.edu.br/~msobral/SOP/portugol.html Um texto HTML]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/exemplos/soma.c Um programa em linguagem C]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/exemplos/Soma.java Um programa em linguagem Java]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/exemplos/soma.py Um programa em linguagem Python]
 
* [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/lab6/Leiame.txt Um texto simples]
 
  
=== nano ===
+
Abaixo segue uma breve ajuda de como obtê-lo, instalá-lo e usá-lo. Esse guia assume que você esteja usando o Ubuntu Linux 9.04 ou superior.
  
O editor ''nano'' serve para editar arquivos em modo texto. Sua tela apresenta uma área de trabalho, onde aparece o texto, com uma barra de ajuda embaixo. Nessa barra há os comandos que podem ser usados, que devem ser invocados com a combinação da tecla ''Ctrl'' e a letra indicada. Para terminá-lo basta teclar ''Ctrl X'', fornecendo o nome do arquivo onde guardar o texto.
+
# Faça o [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/portugol23.tar.gz download] do Portugol.
 +
# Descompacte-o com o seguinte comando: <syntaxhighlight lang=bash>
 +
tar xzf portugol23.tar.gz
 +
</syntaxhighlight>
 +
# Repare que existe agora um subdiretório portugol no diretório onde você o descompactou. Execute o Portugol com o seguinte comando: <syntaxhighlight lang=bash>
 +
java -jar portugol/Portugol.jar
 +
</syntaxhighlight> Obs: você precisará ter Java instalado. Caso não o tenha, execute o comando: <syntaxhighlight lang=bash>
 +
sudo apt-get install openjdk-6-jre
 +
</syntaxhighlight>
 +
# Copie esse [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/exemplos/fluxogramas.cfg arquivo] para poder ver fluxogramas coloridos, e grave-o no memso diretório onde está o Portugol.
 +
# Veja a [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/help/index.html ajuda] do Portugol, e use-a sempre que tiver dúvidas !
  
[[imagem:Nano.png|600px]]
+
<br>A tela inicial do Portugol segue abaixo, junto com um programa demonstrativo.<br>
  
 +
[[Imagem:Editor-Portugol.png]]
  
=== vi ===
+
Exemplos de programas iniciais em Portugol:
  
O editor vi é bastante simples e muito utilizado por ser encontrado em todas as distribuições Linux. Poderíamos optar por um editor um pouco mais avançado, mas com o inconveniente de encontrarmos uma distribuição/instalação onde não dispuséssemos deste editor. O editor vi não tem como objetivo formatar textos: negritos, indentações, justificação, etc. Na prática o vi é muito usado para editar textos que não necessitam de formatação em nenhum momento, como por exemplo códigos fonte de programas em alguma linguagem de programação, e que não carreguem o texto com caracteres especiais. Assim, esse editor também é chamado de ''editor de programas''.
+
# ''Lendo um número e mostrando-o na tela em seguida:'' <syntaxhighlight lang=text>
 +
Inicio
 +
  inteiro x
  
''Dicas do Maykon:''
+
  Escrever "Digite um numero: ",
* [http://aurelio.net/vim/ Guia do vim]
+
  Ler X
** [http://aurelio.net/vim/vi-teclado.png Um guia de referência]
+
  Escrever "Numero digitado: ", x
 +
Fim
 +
</syntaxhighlight>
 +
# ''Lendo dois números, somando-os e mostrando o resultado na tela:'' <syntaxhighlight lang=text>
 +
Inicio
 +
  inteiro x, y
  
==== Modos de operação ====
+
  Escrever "Digite um numero: ",
 +
  Ler x
 +
  Escrever "Digite outro numero: ",
 +
  Ler y
 +
  Escrever "Soma = ", x+y
 +
Fim
 +
</syntaxhighlight>O programa abaixo é equivalente:<syntaxhighlight lang=text>
 +
Inicio
 +
  inteiro x, y, z
  
O editor vi tem três modos  de operação distintos: modo insert ,
+
  Escrever "Digite um numero: ",
modo escape  (também chamado de modo comando), modo last line;
+
  Ler x
 +
  Escrever "Digite outro numero: ",
 +
  Ler y
 +
  z <- x + y
 +
  Escrever "Soma = ", z
 +
Fim
 +
</syntaxhighlight>
  
* '''insert:''' usado para a digitação do texto. Neste modo o vi funciona como uma máquina de escrever, com a diferença de que você pode retroceder sobre o texto já digitado para corrigir eventuais erros. Cada caracter que for digitado aparecerá na tela exatamente como foi digitado.
+
=== Atividades ===
* '''escape:'''  os  caracteres  comuns  (letras,  números  e  sinais  de pontuação) têm um significado especial e quase todos os caracteres funcionam como comandos; portanto, existem muitos comandos. Alguns comandos servem para passar para o ''modo insert'', outros para movimentar o cursor sobre as linhas do texto, alterar trechos do texto, buscar palavras, etc.
 
* '''last  line:'''  os  comandos são digitados em  uma  linha  especial  que aparece no final da tela quando se digita ''':''' (dois pontos) no ''modo escape''. Parte dos comandos do ''modo escape'' possuem similares no ''modo last line'', como por exemplo os comandos  de edição, que veremos mais adiante. Os comandos no ''modo last line'' devem ser seguidos por ENTER, contrariamente ao que acontece no modo escape.
 
  
==== Criação e edição de arquivos ====
+
# ''Média de três números:'' escreva um programa para calcular a média de três números.
 
+
# ''Média balanceada de três números:'' escreva um programa para calcular a média balanceada de três números.
[http://dmitry.eti.br/vim Outro guia rápido sobre vi]
+
# ''Sequência de Fibonacci:'' em matemática corresponde aos números: <syntaxhighlight lang=text>
 
+
1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, ...
Para criarmos um arquivo simplesmente digitamos vi seguido do nome do arquivo. Por exemplo:
+
</syntaxhighlight>... que pode ser descrita pela relação de recorrência <br><br><math>F_{n} = F_{n-1} + F_{n-2}</math><br><br>... com valores iniciais <math>F_{0} = 1</math> e <math>F_{1} = 1</math>.<br><br>Numerosas formas na natureza apresentam essa sequência, como neste girassol (cujas flores se dispõem em uma espiral):<br><br>[[Imagem:Sunflower.jpg]]  [[Imagem:espiral_fibonacci.png]] [[Imagem:FibonacciBlocks.png]]<br><br>Usando o Portugol escreva um programa que mostre os 10 primeiros números dessa sequência.
 +
# ''Distância até o horizonte:'' escreva um programa que calcule a distância dos seus olhos até o horizonte. Assuma que a Terra é perfeitamente esférica, e que seu raio tem 6.378 km. Considere que você esteja no nível do mar (seus pés tocando a água do mar ;-), e que o horizonte esteja num mar perfeitamente liso. ''Dica: faça um diagrama desse problema para visualizar sua geometria, e use trigonometria para resolvê-lo.
  
<syntaxhighlight lang=bash>vi primeiro.arquivo</syntaxhighlight>
+
== 13/04: Algoritmos ==
  
Após isto será aberto o editor com conteúdo vazio, no modo comando. Para podermos editar qualquer coisa devemos entrar no modo inserção, para isto basta teclarmos <nowiki><i></nowiki> (aparecerá -- INSERT -- na base da janela). Em seguida digitamos o texto propriamente dito, usando o teclado normalmente.
+
=== Variáveis e constantes ===
  
Para salvarmos o texto devemos teclar <Esc> (modo comando), <:> (modo last line> e <w> (write). Assim teremos o nosso texto salvo.
+
* '''Variável:''' capaz de guardar um dado a ser usado no algoritmo. Pode ser entendida como uma caixa, onde se coloca um dado e se pode consultá-lo quantas vezes for necessário. O dado pode ser modificado (substituído por outro). Exemplo em Portugol: <syntaxhighlight lang=text>
 +
Inicio
 +
  inteiro anos, dias
  
Agora vamos a alguns comandos úteis:
+
  Escrever "Quantos anos se passaram ? "
 +
  Ler anos
 +
  dias <- anos * 365
 +
  Escrever "... então se passaram ", dias, " dias"
 +
Fim
 +
</syntaxhighlight> Nesse exemplo há duas variáveis: ''dias'' e ''anos''
 +
* '''Constante:''' semelhante à variável, porém o dado armazenado não pode ser modificado. Exemplo em Portugol:<syntaxhighlight lang=text>
 +
Inicio
 +
  constante inteiro diasPorAno <- 365
 +
  inteiro anos, dias
  
* '''Copiar algumas linhas do texto:''' colocamos o cursor no início do texto a ser copiado e, no modo de comando, teclamos <n>+<y>+<y>, onde n é número de linhas que desejamos copiar. Por exemplo se digitarmos <5>+<y>+<y> copiaremos 5 linhas para o buffer.
+
  Escrever "Quantos anos se passaram ? "
* '''Excluir algumas linhas do texto:''' colocamos o cursor no início do texto a ser excuído e, no modo de comando, teclamos <n>+<d>+<d>, onde n é número de linhas que desejamos copiar. Por exemplo se digitarmos <3>+<y>+<y> apagaremos 3 linhas do texto mas que serão armazenadas no buffer.
+
  Ler anos
* '''Colar o conteúdo do buffer para alguma parte do texto:''' colocamos o cursor no ponto onde pretendemos inserir o texto e, no modo de comando, teclamos <p> (paste) para inserirmos o texto abaixo da linha do cursor e <P> para inserir o texto acima da linha do cursor.
+
  dias <- anos * diasPorAno
* '''Encontrar alguma palavra:''' no modo de comando, teclamos </> <palavra> <Enter>. O vi mostrará a primeira ocorrência da mesma. Para ir para a próxima ocorrência teclamos <n> (next), e para a ocorrência anterior teclamos <nowiki><p></nowiki> (previous)
+
  Escrever "... então se passaram ", dias, " dias"
* '''Substituir uma palavra por outra:''' no modo de comando, teclamos <:s/><palavra></><outra><Enter>. Por exemplo se quisermos substituir velha por nova: <:s/><velha></><nova><Enter>, assim teremos a troca da primeira ocorrência de velha por nova. Para substituirmos todas as ocorrências acrescentamos <%> entre <:> e <nowiki><s></nowiki> do caso anterior. Exemplo: <:%s/><velha></><nova><Enter>
+
Fim
* '''Inserir o conteúdo de um texto externo em nosso texto:''' no modo de comando, teclamos <:r>+<caminho/arquivo>+<Enter>.
+
</syntaxhighlight> Nesse exemplo há uma constante: ''diasPorAno''
* '''Salvar com outro nome:''' no modo de comando, teclamos <:w>+<novo.nome>+<Enter>.
 
  
 +
<br>
 +
Variáveis e constantes devem ser declaradas antes de serem usadas (algumas poucas linguagens, como [http://www.python.org Python] e [http://www.perl.org Perl], não exigem isto). A declaração consiste do tipo e identificador da variável. O tipo corresponde ao tipo de valor que pode ser guardado, e o identificador é o nome da variável. No exemplo abaixo:
  
Uma lista de comandos resumida segue abaixo:
+
<syntaxhighlight lang=text>
 
+
  constante inteiro diasPorAno <- 365
* '''0''': vai para o início da linha
+
  inteiro anos, dias
* '''$''': vai para o final da linha
+
Fim
* '''gg''': vai para a primeira linha
+
</syntaxhighlight>
* '''G''': vai para a última linha
 
* ''':n''': vai para a linha n (n é um número)
 
* '''w''': vai para a próxima palavra
 
* '''b''': vai para a palavra anterior
 
* '''x''': apaga um caractere à direita do cursor
 
* '''X''': apaga um caractere à esquerda do cursor
 
* '''dd''': apaga-se uma linha inteira
 
* '''D''': apaga do cursor até o final da linha
 
* '''dw''': apaga até a próxima palavra
 
* '''dfs''': apaga até a próxima ocorrência do caractere s
 
* '''a''': entra-se em inserção uma posição à frente do cursor
 
* '''i''': entra-se em inserção uma posição antes do cursor
 
* '''rc''': substitui o caractere sob o cursor pelo caractere c
 
* '''cwnova''': substitui a palavra a partir do cursor pela palavra nova
 
* ''':w''': grava o arquivo
 
* ''':w um_arquivo''': grava o conteúdo no arquivo um_arquivo
 
* ''':w! um_arquivo''': grava o conteúdo no arquivo um_arquivo, sobrescrevendo-o caso exista
 
* ''':r um_arquivo''': lê um_arquivo e insere seu conteúdo a partir do cursor
 
* ''':r!comando''': executa comando no shell, e insere seu resultado a partir do cursor
 
* ''':q''': sai do vi (termina o editor)
 
* ''':q!''': sai do vi, descartando as modificações
 
 
 
=== Outros editores ? ===
 
  
Há muitos outros editores de texto ! Se tiver curiosidade, faça um pesquisa na Web sobre os editores de texto que existem para o Linux.
+
No  exemplo acima, há duas variáveis do tipo ''inteiro'', e seus identificadores são ''dias'' e ''anos''. O tipo ''inteiro'' indica que essas variáveis podem guardar somente números inteiros.
  
 +
'''''Tipos de variáveis e constantes no Portugol:'''''
  
=== Atividade ===
+
{| border="1" cellpadding="2"
 +
!Tipo
 +
!Descrição
 +
!Exemplo
 +
|-
 +
|Inteiro||Número inteiro entre -2 147 483 648 e 2 147 483 647 || Inteiro x <- 10
 +
|-
 +
|Real||Número real entre -1.7 E 308 e 1.7 E 308 || Real y <- 10.789
 +
|-
 +
|Lógico||Valor booleano, com valores "Verdadeiro" e "Falso" || Logico ok <- Falso
 +
|-
 +
|Caracter||Um caractere da [http://pt.wikipedia.org/wiki/ASCII tabela ASCII] || Caracter letra <- "A"
 +
|-
 +
|Texto||Uma sequência de caracteres (ou ''string'') || Texto palavra <- "um teste"
 +
|}
  
Usando os três editores faça os seguintes exercícios:
+
A declaração de constantes é semelhante à de variáveis, bastanto prefixá-las com a palavra-chave ''constante''.
  
# Crie e edite um arquivo contendo seus dados pessoais, de forma que possam ser usados em sua assinatura de email. Salve-o com nome signature.txt.
+
=== Expressões aritméticas ===
# Edite o arquivo [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/lab6/Leiame.txt Leiame.txt]. Logo após suas duas primeiras linhas (título de seu conteúdo), adicione uma linha com seu nome, matrícula e turma. Salve o arquivo modificado.
 
# Em Leiame.txt acrescente a data de hoje após a última linha.
 
# Em Leiame.txt remova a 2a questão. Apague também a linha em branco logo a seguir. Em seguida renumere as próximas questões, para que fiquem na sequência correta.
 
# Em Leiame.txt localize a 3a ocorrência das palavras "diretorio" ou "diretorios", e remova-a. Localize então a ocorrência imediatamente anterior, e substitua por maiúsculas.
 
# Em Leiame.txt localize as ocorrências da palavra “diretorio”, e substitua por “subdiretorio”, com exceção de quando se referir ao diretório raiz. Salve o arquivo modificado.
 
# Em Leiame.txt, acrescente o resultado da listagem do diretório /home logo após a questão 5.
 
# Salve o arquivo resultante com o nome Leiame2.txt. O arquivo resultante deve ter mais ou menos [http://www.sj.ifsc.edu.br/~msobral/SOP/roteiros/lab6/Leiame2-professor.txt este conteúdo].
 
  
== 24/08: Compactadores de arquivos ==
+
Um conjunto de operações sobre variáveis, constantes e funções numéricas, e que gera um determinado resultado numérico.
  
Uma primeira forma de compactar arquivos no Linux é usando o gerenciador de arquivos ''nautilus''. Por exemplo, ra compactar um diretório deve-se selecioná-lo e em seguida chamar o ''pop-up menu'' (botão direito do mouse):
+
Exemplos de expressões aritméticas: <syntaxhighlight lang=text>
 +
# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
 +
3600*horas + 60*minutos + segundos
  
 +
# Uma expressão que calcula a velocidade instantânea, segundo um MRV
 +
vel_inicial + aceleracao*tempo;
  
[[imagem:Compress1.png|600px]]
+
# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
 +
raiz(x^2 + y^2)
 +
</syntaxhighlight>
  
 +
Os resultados de expressões podem ser mostrados na tela, ou armazenados em variáveis:
  
Ao se escolher a opção ''Comprimir'', uma  janela irá surgir para que se defina o nome do arquivo compactado e o tipo de compactação a ser usada:
+
<syntaxhighlight lang=text>
 +
# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
 +
segundos <- 3600*horas + 60*minutos + segundos
  
 +
# Uma expressão que calcula a velocidade instantânea, segundo um MRV
 +
escrever 'Velocidade no instante ', tempo, ' = ', vel_inicial + aceleracao*tempo;
  
[[imagem:Compress2.png|600px]]
+
# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
 +
modulo <- raiz(x^2 + y^2)
 +
</syntaxhighlight>
  
 +
Repare que uma expressão fica sempre do lado direito, quando atribuída a uma variável. A expressão é primeiro calculada, e em seguida seu resultado é armazenado na variável:
  
O arquivo compactado aparecerá na listagem de arquivos do ''nautilus'':
+
<syntaxhighlight lang=text>
 +
segundos <- 3600*horas + 60*minutos + segundos
 +
</syntaxhighlight>
  
 +
==== Operadores aritméticos ====
  
[[imagem:Compress3.png|600px]]
+
Expressões aritméticas sao compostas por números e operadores aritméticos:
  
 +
Obs: para os exemplos abaixo são usadas estas variáveis: <syntaxhighlight lang=text>
 +
Real x, area, lado
 +
inteiro dias, horas
 +
</syntaxhighlight>
  
Para descompactá-lo e visualizar seu conteúdo, basta clicar duas vezes nele:
+
{| border="1" cellpadding="2"
 
+
!Operador
 
+
!Descrição
[[imagem:Compress4.png|600px]]
+
!Exemplo
 
+
|-
 
+
|<nowiki>+</nowiki>||Adição|| x <- x + 1
==== Exercício ====
+
|-
 
+
|<nowiki>-</nowiki>||Subtração|| x <- x - 1
Obtenha [http://www.sj.ifsc.edu.br/~msobral/soft/hostapd-0.7.2-com-madwifi.tgz este arquivo], e descompacte-o em algum subdiretório. Em seguida recompacte-o usando os diferentes formatos de compactação.
+
|-
 +
|*||Multiplicação|| x <- x*x*x
 +
|-
 +
|/||Divisão|| dias <- horas / 24
 +
|-
 +
|%||Módulo (resto de divisão)|| horas <- horas % 24
 +
|-
 +
|^||Potenciação|| area <- lado^2
 +
|}
  
=== Compactadores no modo texto ===
+
Precedência dos operadores (nesta ordem): ^, *, /, %, + e -
  
Vários compactadores de arquivos existem no Linux, e o ''nautilus'' simplifica sua seleção e uso. Esses compactadores podem ser usados também no modo texto ... aliás, eles originalmente foram criados para serem usados dessa forma ! O que o ''nautilus'' faz é facilitar seu uso por meio de uma interface gráfica.
+
A precedência pode ser modificada com o uso de parênteses. Ex:
  
Os principais compactadores são:
+
<syntaxhighlight lang=text>
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/compress.1.html compress]: compactador mais antigo e comum nos Unix em geral, porém em desuso. Gera arquivos compactados com extensão .Z. Precisa do pacote de software ''ncompress'' no Ubuntu. Exemplo de uso: <syntaxhighlight lang=bash>
+
escrever 1 + 2 * 3
msobral@dell-iron:~$ ls -l API-changes.txt
+
escrever (1 + 2)*3
-rw-r--r-- 1 msobral professores 51759 2010-03-24 11:32 API-changes.txt
 
msobral@dell-iron:~$ compress API-changes.txt
 
msobral@dell-iron:~$ ls -l API*  
 
-rw-r--r-- 1 msobral professores 22781 2010-03-24 11:32 API-changes.txt.Z
 
msobral@dell-iron:~$ uncompress API-changes.txt.Z
 
msobral@dell-iron:~$ ls -l API*
 
-rw-r--r-- 1 msobral professores 51759 2010-03-24 11:32 API-changes.txt
 
msobral@dell-iron:~$
 
 
</syntaxhighlight>
 
</syntaxhighlight>
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/gzip.1.html GNU zip]: compactador bastante utilizado, com maior poder de compactação que ''compress''. Gera arquivos compactados com extensão .gz. Já vem instalado no Ubuntu. Exemplo de uso: <syntaxhighlight lang=bash>
+
 
msobral@dell-iron:~$ gzip API-changes.txt
+
No Portugol, existem também algumas [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/help/funcoes_biblioteca.html funções úteis], como a função '''''raiz''''': <syntaxhighlight lang=text>
msobral@dell-iron:~$ ls -l API*
+
r <- raiz(x^2 + y^2)
-rw-r--r-- 1 msobral professores 17651 2010-03-24 11:32 API-changes.txt.gz
 
msobral@dell-iron:~$ gunzip API-changes.txt.gz
 
msobral@dell-iron:~$ ls -l API*
 
-rw-r--r-- 1 msobral professores 51759 2010-03-24 11:32 API-changes.txt
 
msobral@dell-iron:~$
 
</syntaxhighlight>
 
* [http://manpages.ubuntu.com/manpages/karmic/en/man1/bzip2.1.html bzip2]: vem sendo bastante usado por ter um poder de compactação ainda maior, porém à custa de maior processamento (compactação fica mais lenta). Gera arquivos compactados com extensão .bz2. Também já vem instalado no Ubuntu. <syntaxhighlight lang=bash>
 
msobral@dell-iron:~$ bzip2 API-changes.txt
 
msobral@dell-iron:~$ ls -l API*
 
-rw-r--r-- 1 msobral professores 15804 2010-03-24 11:32 API-changes.txt.bz2
 
msobral@dell-iron:~$ bunzip2 API-changes.txt.gz
 
msobral@dell-iron:~$ ls -l API*
 
-rw-r--r-- 1 msobral professores 51759 2010-03-24 11:32 API-changes.txt
 
msobral@dell-iron:~$
 
 
</syntaxhighlight>
 
</syntaxhighlight>
* ... outros menos populares no mundo do Linux, tais como [http://manpages.ubuntu.com/manpages/karmic/en/man1/zip.1.html zip], [http://manpages.ubuntu.com/manpages/karmic/en/man1/rar.1.html rar] e [http://manpages.ubuntu.com/manpages/karmic/en/man1/zoo.1.html zoo].
 
  
Note que os compactadores ''compress'', ''gzip'', e ''bzip2'' compactam um arquivo por vez. Assim, com eles não é possível juntar vários arquivos e diretórios dentro de um único arquivo compactado (o que se faz corriqueiramente com '''zip''' ou '''rar''' ...). Portanto, se for necessário compactar um diretório ou um conjunto de arquivos, o melhor é combinar um  compactador com o programa [http://manpages.ubuntu.com/manpages/karmic/en/man1/tar.1.html tar].
+
O resultado de expressões aritméticas depende dos tipos numéricos das variáveis e constantes: <syntaxhighlight lang=text>
 
+
inicio
=== tar ===
+
  real x
 
+
  inteiro y
O programa ''tar'' é um  utilitário do mundo Unix originalmente criado para backups em fita (daí seu nome: ''TApe aRchiver'', se bem que ''tar'' é também um trocadilho que pode significar piche, pois ele de certa forma gruda um arquivo ao outro). O resultado da execução do ''tar'' é um arquivo contendo todos os arquivos e diretórios que foram selecionados para inclusão. Esse arquivo ''tar'' pode ser então compactado, obtendo-se algo parecido com o que faz zip ou rar.
+
  inteiro resultadoInteiro
 
+
  real resultadoReal
Mas porque não usar então zip e rar ? Afinal, eles existem também no Linux ... No entanto, esses compactadores nasceram no mundo do antigo MS-DOS, e assim não são capazes de armazenar todos os atributos de arquivos que existem em sistemas Unix (informações tais como usuário e grupo dono do arquivo, permissões de acesso, tipo do arquivo, datas de último acesso e modificação). O ''tar'', pelo contrário, consegue preservar esses atributos, e por isto se torna mais adequado para uso no Unix em geral.
+
 
+
  x <- 9
Uso do ''tar'':
+
  y <- 9
* '''Criação de arquivo tar:''' <syntaxhighlight lang=bash>tar cf nome_arquivo.tar arquivo_ou_diretorio1 [arquivo_ou_diretorio2 ...]</syntaxhighlight>
+
* '''Mostrar o conteúdo de arquivo tar:''' <syntaxhighlight lang=bash>tar tvf nome_arquivo.tar </syntaxhighlight>
+
  escrever "O resultado de uma expressão aritmética depende dos tipos das variáveis e constantes\n"
* '''Extrair conteúdo de arquivo tar:''' <syntaxhighlight lang=bash>tar xf nome_arquivo.tar</syntaxhighlight>
+
  escrever "usadas na expressão. Se forem todas inteiras, então o resultado será inteiro.\n"
 +
  escrever "Veja este exemplo: \n"
 +
  escrever "Variável inteira y=", y
 +
  escrever "\nExpressão: y/2=", y/2
 +
 +
  escrever "\n\nNeste segundo exemplo, repete-se a mesma expressão, porém usando-se uma\n"
 +
  escrever "variável real:\n"
 +
  escrever "Variável real x=", x
 +
  escrever "\nExpressão: x/2=", x/2
 +
 +
  x <- 4
 +
  y <- 5
 +
  escrever "\n\nSe as variáveis de diferentes tipos forem combinadas, o resultado da\n"
 +
  escrever "expressão será real:\n"
 +
  escrever "Variável real x=", x, " e inteira y=", y
 +
  escrever "\nExpressão: (x+y)/2=", (x+y)/2
 +
 +
  escrever "\n\nNo entanto, se uma expressão tiver um resultado real, mas este for\n"
 +
  escrever "atribuído a uma variável inteira, então apenas a parte inteira será guardada:\n"
 +
  escrever "Variável real x=", x, " e inteira y=", y
 +
  y <- (x+y)/2
 +
  escrever "\nExpressão: y <- (x+y)/2 ... após executada, y=", y
 +
 +
fim
  
Os usos acima não compactam os arquivos incluídos dentro do arquivo ''tar''. Para compactá-los deve-se adicionar uma opção de compactação:
+
</syntaxhighlight>
* '''Criação de arquivo tar compactado:'''
 
** '''''Com compress:''''' <syntaxhighlight lang=bash>tar cZf nome_arquivo.tar.Z arquivo_ou_diretorio1 [arquivo_ou_diretorio2 ...]</syntaxhighlight>
 
** '''''Com gzip:''''' <syntaxhighlight lang=bash>tar czf nome_arquivo.tar.gz arquivo_ou_diretorio1 [arquivo_ou_diretorio2 ...]</syntaxhighlight>
 
** '''''Com bzip2:''''' <syntaxhighlight lang=bash>tar cjf nome_arquivo.tar.bz2 arquivo_ou_diretorio1 [arquivo_ou_diretorio2 ...]</syntaxhighlight>
 
* '''Mostrar o conteúdo de arquivo tar:'''
 
** '''''Com compress:''''' <syntaxhighlight lang=bash>tar tvZf nome_arquivo.tar.Z </syntaxhighlight>
 
** '''''Com gzip:''''' <syntaxhighlight lang=bash>tar tvzf nome_arquivo.tar.gz </syntaxhighlight>
 
** '''''Com bzip2:''''' <syntaxhighlight lang=bash>tar tvjf nome_arquivo.tar.bz2 </syntaxhighlight>
 
* '''Extrair conteúdo de arquivo tar:'''
 
** '''''Com compress:''''' <syntaxhighlight lang=bash>tar xZf nome_arquivo.tar.Z </syntaxhighlight>
 
** '''''Com gzip:''''' <syntaxhighlight lang=bash>tar xzf nome_arquivo.tar.gz </syntaxhighlight>
 
** '''''Com bzip2:''''' <syntaxhighlight lang=bash>tar xjf nome_arquivo.tar.bz2 </syntaxhighlight>
 
  
 
=== Atividade ===
 
=== Atividade ===
  
Faça os exercícios abaixo usando o ''nautilus'' e os compactadores no modo texto:
+
Para os exercícios abaixo, escreva o algoritmo no Portugol.
  
# Procure um arquivo de texto (extensão .txt, .html, ou .doc) com ao menos 100 kB e compacte-o com compress, gzip e bzip2. Compare os tamanhos dos arquivos resultantes.
+
# Faça um algoritmo que, usando somente duas variáveis, calcule a média de quatro números.
# Usando o programa tar, compacte todo o diretório home do usuário ''aluno''. Experimente criar o arquivo tar sem compactação, e depois compactado com cada um dos compactadores vistos. Você terá portanto como resultado quatro arquivos tar diferentes ...
+
# Escreva um algoritmo que calcule as raizes de uma equação de 2o grau.
# Descompacte os arquivos tar gerados na questão anterior. Para evitar sobreposição, descompacte cada um deles em um diretório separado.
+
# Um equipamento conta o tempo desde que foi ligado. No entanto, essa contagem é feita em segundos. Faça um algoritmo que converta o valor desse contador para horas, minutos e segundos. Ex: se o contador tiver o valor 43397, o seu programa deve mostrar: 12:03:17. <syntaxhighlight lang=text>
# Resolva a [http://www.sj.ifsc.edu.br/~msobral/SOP/prova1/prova1-rec-2009-2.tgz prova] que foi aplicada no semestre 2009.2.
+
inicio
 +
  inteiro h,m,s
  
== 30/08: Avaliação ==
+
  escrever "Quantos segundos mostrou o equipamento ? "
 +
  ler s
 +
 
 +
  h <- s /3600
 +
  s <- s % 3600  // s <- s - h*3600
 +
  m <- s /60
 +
  s <- s % 60 // s <- s - m*60
 +
 
 +
  escrever h, ":", m, ":", s
 +
 
 +
fim
 +
</syntaxhighlight>
  
[http://www.sj.ifsc.edu.br/~msobral/SOP/tmp/prova1-2010-2.tgz Avaliação teórica e prática]
+
== 14/04: Variáveis e expressões ==
  
= Lógica de Programação =
+
Continuação ...
  
== 31/08: Introdução ==
+
=== Exercícios ===
  
'''Assuntos:''' Lógica de programação: instrução, sequência lógica, algoritmo.
+
# Crie um conversor de decimal para binário (limite: 4 bits). Ex.: tendo o número 10 (decimal) de entrada, deve-se obter o número 1010 (binário) de saída.
 +
# Escreva um algoritmo que leia o nome, sobrenome e idade de uma pessoa, e escreva na tela (no lugar dos colchetes devem aparecer os valores lidos):<syntaxhighlight lang=text>
 +
Nome: [sobrenome, nome]
 +
Idade: [idade] anos</syntaxhighlight>
 +
# A função exponencial <math>f(x) = e^x</math> pode ser calculada por uma série de Taylor:<br><math>f(x) \approx 1+x+\frac{x^2}{2}+\frac{x^3}{6}+...+\frac{x^n}{n!}</math><br> Faça um  algoritmo para calcular o valor desta função.
  
''Referências auxiliares:''
+
== 16/04: Estruturas de decisão ==
* [http://wiki.sj.ifsc.edu.br/images/0/02/L%C3%B3gica_de_Programa%C3%A7%C3%A3o.pdf Apostila auxiliar], escrita pelo prof. Paulo Sérgio de Moraes.
+
 
* Ver [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula10.pdf transparências].
+
O exemplo abaixo está contido no livro ''Classification and Regression Trees'', de Leo Breiman, Jerome Friedman, Richard Olshen e Charles Stone, editora Chapman & Hall, 1984.
  
 +
Na Universidade da California, no centro Médico de San Diego, quando um paciente com ataque cardíaco é recebido, dezenove variáveis são medidas ao longo das primeiras 24 horas. Essas incluem pressão sanguínea, idade, e 17 outras variáveis ordenadas e binárias (booleanas) que sumarizam os sintomas médicos considerados importantes indicadores da condição do paciente.
  
[http://www.sj.ifsc.edu.br/~msobral/SOP/slides/pao.png Um primeiro algoritmo]
+
O objetivo  de um  estudo médico feito entre os anos 70 e 80 foi o desenvolvimento de um método para identificar pacientes de alto risco (que não sobreviverão ao menos 30 dias) com  base nos dados obtidos nas primeiras 24 horas. O diagrama abaixo mostra uma regra de classificação que foi produzida nesse estudo. A letra F  significa que não há um risco alto, e a letra G quer dizer paciente de alto risco.
  
 +
[[imagem:Crt-decisao.png]]
  
''Videos sobre algoritmos:''
+
Essa regra classifica os pacientes como F ou G dependendo de respostas do tipo sim/não a no máximo três perguntas. Como  seria um algoritmo que a implementasse ?
* [http://www.youtube.com/watch?v=k0xgjUhEG3U Outro algoritmo ...]
 
* [http://www1.folha.uol.com.br/folha/videocasts/ult10038u689083.shtml Resolvendo o cubo de Rubik (cubo mágico)]
 
** [http://www.sj.ifsc.edu.br/~msobral/SOP/coisas/1285780.flv Cópia desse vídeo aqui no Ifsc]
 
  
'''Exemplificando com ''shell scripts'':'''
+
----
  
<syntaxhighlight lang=bash>
+
Algoritmos para resolver problemas como  o exemplo acima dependem de se poder expressar a '''''tomada de decisões'''''. Deve ser possível testar '''''condições''''', como  por exemplo "''a idade é maior que 62.5 anos ?''", e executar instruções dependendo do resultado do teste. Em Lógica de Programação, e em linguagens de programação em geral, isso se faz com '''''estruturas de decisão'''''.
#!/bin/bash
 
  
# Cada linha no script abaixo corresponde a uma instrução ...
+
'''''Estruturas de decisão''''' possibilitam que se executem diferentes sequências de instruções de um programa, dependendo de uma condição a ser avaliada. Por exemplo, um jogo poderia armazenar a maior pontuação já obtida, e verificar se foi ultrapassada ao final de cada partida:
# O conjunto de instruções na ordem apresentada forma uma sequência lógica ...
 
  
echo Iniciando o script ...
+
<syntaxhighlight lang=text>
echo Vou procurar todos os arquivos de texto existentes neste diretório
+
  Se pontuacao > recorde então
 +
    recorde <- pontuação
 +
  FimSe
 +
</syntaxhighlight>
  
find . -type f -name "*.doc" > .tmp
+
O exemplo em Portugol acima mostra a estrutura de decisão ''Se condição entao comandos Fimse''. O equivalente em  fluxograma é:
find . -type f -name "*.txt" >> .tmp
 
find . -type f -name "*.rtf" >> .tmp
 
find . -type f -name "*.odt" >> .tmp
 
  
echo Os arquivos são:
+
[[imagem:Seentao.png]]
  
cat .tmp
+
Veja o próximo exemplo:
  
rm -f .tmp
+
<syntaxhighlight lang=text>
 +
  Se conceito > 70 entao
 +
    escrever 'Voce esta aprovado'
 +
  senao
 +
    escrever 'Desta vez não deu ... tente de novo !'
 +
  Fimse
 
</syntaxhighlight>
 
</syntaxhighlight>
  
=== Problemas exemplo ===
+
O uso de ''Se condição entao comandos Senao comandos Fimse'' possibilita que se execute uma sequência de comandos se a condição for verdadeira, e outra sequência se for falsa. Seu equivalente em  fluxograma é:
  
'''Problema dos três recipientes'''
+
[[imagem:Seentaosenao.png]]
  
Há três recipientes com tamanhos distintos: um com 8 litros, outro com 5 litros e o terceiro com 3 litros. O recipiente com 8
+
Para fazer um bom uso de estruturas de decisão deve-se primeiro conseguir identificar as condições a serem avaliadas. Condições são escritas com expressões lógicas, e estas são compostas de operadores lógicos e relacionais aplicados a variáveis e constantes.
litros está completamente cheio. Deseja-se colocar 4 litros em dois recipientes. Considere que os recipientes não são graduados.
 
  
 +
=== Condições ===
  
'''Problema da travessia'''
+
Condições representam os testes que precisam ser feitos com dados do problema. Esses testes são expressados por '''''expressões lógicas''''', cujos resultados são ''verdadeiro'' ou ''falso''. Expressões lógicas são construídas usando-se '''''operadores lógicos''''' e '''''operadores relacionais''''' para comparar e avaliar os valores de variáveis e constantes.
 
Um barqueiro precisa levar um saco de milho, uma galinha e uma raposa para o outro lado do rio. Porém o barco somente é capaz de levar uma coisa de cada vez (além do barqueiro). Qual a sequência de travessias necessário para atravessar o milho, a galinha e a raposa ?
 
  
'''Torres de Hanoi'''
+
Obs: para os exemplos abaixo são usadas estas variáveis: <syntaxhighlight lang=text>
 +
Logico correto, multa, aprovado, barato, bombear_agua
 +
Logico descartar, baixo, reprovado, erro, enviado, recebido
 +
inteiro erros, pontuacao, preco, endereco, velocidade
 +
Real faltas, nivel_agua, altura
 +
</syntaxhighlight>
  
Há três hastes. Uma das hastes serve de suporte para três discos de tamanhos diferentes. Um disco menor sempre é colocado sobre um disco maior. A figura abaixo ilustra as hastes e os discos:
+
==== Operadores relacionais ====
  
[[imagem:Hanoi.png]]
+
Operadores relacionais são usados para comparar valores, e estão listados  na tabela abaixo. Cada comparação feita com esses operadores forma uma ''expressão lógica simples''. Portanto, o resultado de uma comparação é um valor lógico que pode ser ''verdadeiro'' ou ''falso''.
  
Desejam-se mover os três discos para a haste da direita. Porém só pode se mover um disco por vez, e um disco maior nunca pode ficar sobre um disco menor.
+
{| border="1" cellpadding="2"
 
+
!Operador
Operação possível: '''Move''' ''disco'' para ''haste''
+
!Descrição
 +
!Exemplo
 +
|-
 +
|=||Igualdade|| correto <- (erros = 0)
 +
|-
 +
|>||Maior|| multa <- (velocidade > 80)
 +
|-
 +
|>=||Maior ou igual|| aprovado <- (pontuacao >= 70)
 +
|-
 +
|<||Menor|| barato <- (preco < 100)
 +
|-
 +
|<=||Menor ou igual|| bombear_agua <- (nivel_agua <= 0.7)
 +
|-
 +
|=/=||Diferente|| descartar <- (endereco =/= 12345)
 +
|}
  
Qual a sequência de operações para mover os discos de uma haste para outra ?
+
==== Operadores lógicos ====
  
=== Atividade extra ===
+
Operadores lógicos são usados para combinar expressões lógicas simples, criando assim ''expressões lógicas compostas'', e estão listados  na tabela abaixo.
 
 
* O jogo [http://armorgames.com/play/2205/light-bot LightBot] mostra de uma forma divertida como criar pequenos algoritmos. Até que fase desse jogo você consegue chegar ?
 
 
 
=== Leitura adicional ===
 
 
 
* [http://cs-exhibitions.uni-klu.ac.at/index.php?id=193 História dos algoritmos]
 
* [http://en.wikipedia.org/wiki/Algorithm Uma definição de algoritmos na Wikipedia]
 
 
 
=== Conceitos da 1a avaliação ===
 
  
 
{| border="1" cellpadding="2"
 
{| border="1" cellpadding="2"
  !Aluno
+
  !Operador
  !Teorica
+
  !Descrição
  !Pratica
+
  !Exemplo
 
  |-
 
  |-
  |Fernando || A || D
+
  |NAO||Negação|| baixo <- NAO (altura > 1.8)
 
  |-
 
  |-
  |Gabriel || D || D
+
  |E||Conjunção|| aprovado <- NAO (conceito = "D") E (faltas <= 0.25)
 
  |-
 
  |-
  |Levi || D || D
+
  |OU||Disjunção|| reprovado <- (conceito = "D") OU (faltas > 0.25)
 
  |-
 
  |-
  |Lindalvo || D || D
+
  |XOU||Disjunção exclusiva|| erro <- enviado XOU recebido
|-
+
|}
|Marine || D || D
 
|-
 
|Mario Allan || A || B
 
|-
 
|Mário André || B || C
 
|-
 
|Mário Sérgio || C || D
 
|-
 
|Maykon || A || C
 
|-
 
|Nathany || B || D
 
|-
 
|Paulo Igor || A || D
 
|-
 
|Ricardo || C || D
 
|-
 
|Roberto || A || B
 
|-
 
|Thiago Martins || C || D
 
|-
 
|Thiago Ramos || C || D
 
|-
 
|Vinicius Kachniacz || C || C
 
|-
 
|Vinicius Rodrigues || A || D
 
|-
 
|}
 
  
== 06/09: Algoritmos ==
+
Precedência dos operadores (nesta ordem): NAO, E, OU e XOU
  
Ver [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula11.pdf transparências].
+
Lembre que a precedência pode ser modificada com o uso de parênteses.
  
Introdução a lógica de programação:
+
=== Atividades ===
* '''Algoritmos:''' sequência finita de passos que leva à execução de uma tarefa. Exemplos:
 
*# [http://unesp.br/~jroberto/rubiks Como resolver o cubo de Rubik (cubo mágico)]
 
*# ''Cálculo das raízes de uma equação de 2o grau (método de Baskara)''
 
*# [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/pao.png Receita de pão]
 
*# ''Soma de dois números:'' <syntaxhighlight lang=text>
 
Ler primeiro número
 
Ler segundo número
 
Somar os dois números lidos
 
Mostrar o resultado
 
</syntaxhighlight>
 
* '''Programa:''' tradução de um algoritmo para uma linguagem de programação, para que possa ser executado por um computador.
 
*# ''Soma de dois números em shell script:'' <syntaxhighlight lang=bash>
 
#!/bin/bash
 
  
echo -n Entre com o primeiro numero:
+
# Faça um programa que leia um número e então informe se ele é par ou ímpar.
read x
+
# Um radar de trânsito faz a medição de velocidade de veículos e, dependendo do valor, calcula a multa a ser aplicada. Em uma determinada via esse radar foi configurado da seguinte forma:
echo -n Entre com o segundo numero:
+
#* Se a velocidade for maior que 80 km/h, a multa é de R$ 360.
read y
+
#* Se a velocidade for maior que 60 km/h, a multa é de R$ 180.
z=$(($x + $y))
+
#* Se a velocidade for menor ou igual a 60 km/h, não há multa.<br><br>Escreva um algoritmo que calcule a multa de acordo com a velocidade de um veículo.
 
+
# Faça um algoritmo que leia três números do teclado, e mostre o maior e menor números.
echo A soma de $x e $y eh $z
+
# Escreva um  algoritmo  que faça o [[SOP-2010-2-sobral#20.2F09:_Estruturas_de_decis.C3.A3o|teste de risco de paciente cardíaco]], mostrado no início da aula.
</syntaxhighlight> Para executar esse programa, grave-o em um arquivo chamado ''soma.sh'' e em seguida faça o seguinte: <syntaxhighlight lang=bash>
+
# Um estudo sobre sensibilidade de pessoas a temperaturas da água identificou que a maioria das pessoas considera fria a água com temperaturas abaixo de 25 graus, morna entre 25 e 30 graus, e quente acima de 30 graus. Escreva um algoritmo que mostre as palavras "fria", "morna" ou "quente" dependendo da temperatura da água que for informada.
bash soma.sh
+
# Em uma rede de computadores, o ''firewall'' restringe o acesso a Internet dependendo do horário e do tipo de usuário que faz o acesso. Os tipos de usuário abaixo têm as seguintes restrições:
</syntaxhighlight>
+
#* Funcionario: apenas entre 0:00 e 7:30, entre 18:30 e 0:00, e entre 12:00 e 13:30
*# ''Soma de dois números em Linguagem C:'' <syntaxhighlight lang=c>
+
#* Financeiro: qualquer horario
#include <stdio.h>
+
#* Diretoria: qualquer horário<br>Escreva um algoritmo que informe se um acesso foi permitido, em função do horário e do tipo de usuário.
 
+
# Modifique o algoritmo acima para adicionar a seguinte restrição:
int main() {
+
#* Funcionario: nao pode acessar sites de jornal (ex: www.rbs.com.br)
  int x, y, r;
+
#* Financeiro: nao pode acessar sites de jornal durante o expediente
 +
#* Diretoria: sem restrições a sites
 +
# Faça um algoritmo para fazer a divisão de dois números reais. Antes de dividi-los deve ser feito um teste de validade. Caso não seja possível dividi-los, deve ser mostrada uma mensagem de erro. Se for possível, deve-se mostrar o resultado da divisão.
 +
# Faça um jogo de par ou ímpar, em que o jogador aposta contra o computador. O jogador deve digitar um número entre 0 e 5 e optar entre par ou ímpar. O computador deve sortear um número também entre 0 e 5. Se a paridade da soma dos números do jogador e do computador for a mesma que o jogador optou, então ele ganha a partida, senão o computador vence.
 +
# Escreva um algoritmo para identificar se há, dentre três palavras lidas do teclado, ao menos duas palavras distintas que pertençam ao conjunto {azul, preto, vermelho}. Exemplos:
 +
#* Se o usuário digitar ''verde'', ''preto'', ''vermelho'', o programa deve mostrar na tela ''verdadeiro''
 +
#* Se o usuário digitar ''verde'', ''preto'', ''preto'', o programa deve mostrar na tela ''falso''
 +
#* Se o usuário digitar ''azul'', ''preto'', ''azul'', o programa deve mostrar na tela ''verdadeiro''
 +
<syntaxhighlight lang=text>
  
 +
inicio
 +
  texto cor1, cor2, cor3
 +
  inteiro cont <- 0
 +
   
 +
  ler cor1
 +
  ler cor2
 +
  ler cor3
 +
 
 +
  se cor1 = "preto" ou cor2 = "preto" ou cor3 = "preto" entao
 +
    cont <- cont + 1
 +
  fimse
  
   printf("Entre com o primeiro numero: ");
+
   se cor1 = "azul" ou cor2 = "azul" ou cor3 = "azul" entao
   scanf("%d", &x);
+
    cont <- cont + 1
 +
   fimse
  
   printf("Entre com o segundo numero: ");
+
   se cor1 = "vermelho" ou cor2 = "vermelho" ou cor3 = "vermelho" entao
   scanf("%d", &y);
+
    cont <- cont + 1
 +
  fimse
 +
       
 +
  se cont > 1 entao
 +
    escrever "verdadeiro"
 +
   senao
 +
    escrever "falso"
 +
  fimse
 +
fim
 +
</syntaxhighlight>
  
  r = x + y;
+
=== Fluxogramas ===
  
  printf("A soma de %d e %d eh %d\n", x, y, r);
+
Diagramas de bloco para auxiliar a descrição de algoritmos. Ajudam na compreensão do algoritmo, por poder visualizar o fluxo de execução.
  
  return 0;
 
}
 
</syntaxhighlight> Para executar esse programa, grave-o em um arquivo chamado ''soma.c'' e em seguida faça o seguinte: <syntaxhighlight lang=bash>
 
gcc -o soma soma.c
 
./soma
 
</syntaxhighlight>
 
*# ''Soma de dois números em Java:'' <syntaxhighlight lang=java>
 
import java.util.Scanner;
 
  
class Soma
+
[[imagem:Fluxograma-soma.png]] Fluxograma para o algoritmo da média de trẽs números.
 
        public static void main(String args[])
 
        {
 
  int  x, y, r;
 
  Scanner scanner = new Scanner(System.in);
 
 
          System.out.print("Entre com o primeiro numero: ");
 
  x = scanner.nextInt();
 
  
          System.out.print("Entre com o segundo numero: ");
+
<br>
  y = scanner.nextInt();
 
  
  r = x + y;
+
'''Blocos de uso mais comum'''<br>
 +
{| border="1" cellpadding="2"
 +
!Bloco
 +
!Descrição
 +
!Exemplo
 +
|-
 +
|[[Imagem:Inicio.png]]||Inicio do fluxograma||
 +
|-
 +
|[[Imagem:Processamento.png]]||Processamento||
 +
|-
 +
|[[Imagem:Entrada.png]]||Entrada de dados (ler do teclado)||
 +
|-
 +
|[[Imagem:Saida.png]]||Saída de dados (mostrar na tela)||
 +
|-
 +
|[[Imagem:Decisao.png]]||Decisão (testar uma condição e bifurcar)||[[imagem:Ex-decisao.png]]
 +
|-
 +
|[[Imagem:Conector.png]]||Conector (juntar dos ou mais ramos do fluxograma)||[[imagem:Ex-conector.png]]
 +
|-
 +
|[[Imagem:Fim.png]]||Fim||
 +
|}
  
  System.out.println("A soma de " + x + " e " + y + " eh " + r);
+
Obs: [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/exemplos/fluxogramas.cfg Arquivo de configuração das cores do fluxograma ] do Portugol.
        }
 
}
 
</syntaxhighlight>Para executar esse programa, grave-o em um arquivo chamado ''Soma.java'' e em seguida faça o seguinte: <syntaxhighlight lang=bash>
 
javac Soma.java
 
java Soma
 
</syntaxhighlight>
 
*# ''Soma de dois números em Python:'' <syntaxhighlight lang=python>
 
#!/usr/bin/python
 
  
import sys
+
== 20/04: estruturas de decisão ==
  
print "Entre com o primeiro numero: ",
+
Há situações em que se precisa fazer um conjunto de comparações, como mostrado abaixo:
x = int(sys.stdin.readline())
 
  
print "Entre com o segundo numero: ",
+
<syntaxhighlight lang=text>
y = int(sys.stdin.readline())
+
// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
 +
// dia, nome do mês, ano.
 +
Inicio
 +
  inteiro dia, mes, ano
 +
  texto nome_mes
  
r = x + y
+
  escrever "Dia: "
 +
  ler dia
 +
  escrever "Mes: "
 +
  ler mes
 +
  escrever "Ano: "
 +
  ler ano
  
print "A soma de", x, "e", y, "eh", r
+
  se mes = 1 entao
 
+
    nome_mes <- "Janeiro"
sys.exit(0)
+
  senao
</syntaxhighlight>Para executar esse programa, grave-o em um arquivo chamado ''soma.py'' e em seguida faça o seguinte: <syntaxhighlight lang=bash>
+
    se mes = 2 entao
python soma.py
+
      nome_mes <- "Fevereiro"
</syntaxhighlight>
+
    senao
 
+
      se mes = 3 entao
Como mostrado acima, o mesmo algoritmo foi traduzido para programas em diferentes linguagens de programação. Ao serem executados, geraram o mesmo resultado.
+
        nome_mes <- "Março"
 
+
      senao
=== Exercícios: desenho de figuras geométricas ===
+
        se mes = 4 entao
 
+
          nome_mes <- "Abril"
# Usando apenas as instruções: <syntaxhighlight lang=text>
+
        senao
acendeCaneta
+
          se mes = 5 entao
apagaCaneta
+
            nome_mes <- "Maio"
avança X
+
          senao
giraDireita angulo
+
            se mes = 6 entao
giraEsquerda angulo
+
              nome_mes <- "Junho"
fazArco X graus, raio Y
+
            senao
</syntaxhighlight> escreva algoritmos para desenhar a figura entregue pelo professor. Não mostre sua figura para as outras equipes.
+
              se mes = 7 entao
# [http://edu.kde.org/kturtle/ kturtle] é um software educacional para ajudar no ensino de matemática, geometria e introdução à programação. Ele possibilita fazer desenhos facilmente, seguindo um programa com instruções de desenho. Usando as instruções: <syntaxhighlight lang=bash>
+
                nome_mes <- "Julho"
reset (move a tartaruga para o centro da tela)
+
              senao
clear (limpa a tela)
+
                se mes = 8 entao
pendown (põe a caneta no papel - acendeCaneta)
+
                  nome_mes <- "Agosto"
penup (levanta a caneta - apagaCaneta)
+
              senao
forward X (avança X posições)
+
                  se mes = 9 entao
turnright angulo (gira pra direita)
+
                    nome_mes <- "Setembro"
turnleft angulo (gira pra esquerda)
+
                  senao
</syntaxhighlight> ... escreva programas para os algoritmos criados no ítem  anterior.<br><br>[http://docs.kde.org/stable/pt/kdeedu/kturtle/reference.html Guia de referência completo do Kturtle]
+
                    se mes = 10 entao
# Usando o '''kturtle''' escreva algoritmos para desenhar as seguintes figuras:
+
                      nome_mes <- "Outubro"
#* triângulo equilátero
+
                    senao
#* triângulo isósceles
+
                      se mes = 11 entao
#* triângulo escaleno
+
                        nome_mes <- "Novembro"
#* quadrado
+
                      senao
#* hexágono
+
                        nome_mes <- "Dezembro"
#* octógono
+
                      fimSe
#* quadrado com vértices interligados
+
                    fimSe
#* Um círculo
+
                  fimSe
#* 7 hexágonos interligados (um central e seis periféricos).
+
                fimSe
#* 7 círculos interligados (um central e seis periféricos), parecendo uma flor
+
              fimSe
# Fractais são figuras nas quais cada pedacinho é semelhante à toda figura. Um fractal  bastante simples de criar é o Floco de Neve Koch, cuja criação pode ser entendida na sequência de  figuras abaixo:
+
            fimSe
 
+
          fimSe
[[imagem:Koch0.png]]
+
        fimSe
<br>''Primeira iteração''
+
      fimSe
 
+
    fimSe
[[imagem:Koch1.png]]
+
   fimSe
<br>''Segunda iteração''
+
 
 
+
  escrever dia, " de ", nome_mes, " de ", ano
[[imagem:Koch2.png]]
+
fim
<br>''Terceira iteração''
 
 
 
[[imagem:Koch3.png]]
 
<br>''Próximas iterações''
 
 
 
Usando o '''kturtle''' desenhe essas figuras, reproduzindo o maior número de iterações que conseguir. Você consegue descobrir uma  regra para gerá-las ?
 
 
 
''Obs: [http://pt.wikibooks.org/wiki/Matem%C3%A1tica_divertida/Fractais_e_o_infinitamente_pequeno#Fractais fonte dessas figuras]''
 
 
 
==== Dicas para o kturtle ====
 
 
 
# Pode-se repetir uma sequência de instruções usando-se o comando ''repeat''. Por exemplo, para desenhar um pentágono usando ''repeat'': <syntaxhighlight lang=text>
 
repeat 5 {
 
  forward 50
 
  turnleft 72
 
}
 
</syntaxhighlight>
 
# Podem-se usar variáveis para generalizar os algoritmos. Uma variável tem um identificador, que deve ser precedido do caractere ''$'': <syntaxhighlight lang=text>
 
$x=5
 
$lado=50
 
repeat $x {
 
  forward $lado
 
   turnleft 360/$x
 
}
 
</syntaxhighlight>
 
# Pode-se usar o comando ''ask'' para ler do teclado valores a serem usados no algoritmo de desenho: <syntaxhighlight lang=text>
 
$x = ask "Quantos lados tem o polígono ?"
 
$lado = ask "Qual o comprimento de cada lado ?"
 
repeat $x {
 
  forward $lado
 
  turnleft 360/$x
 
}
 
</syntaxhighlight>
 
# Por fim, com o comando ''learn'' é possível criar novos comandos no ''kturtle''. Por exemplo, para criar um comando que desenhe um triângulo: <syntaxhighlight lang=text>
 
learn triangulo $lado {
 
repeat 3 {
 
  forward $lado
 
  turnleft 120
 
}
 
}
 
 
</syntaxhighlight>
 
</syntaxhighlight>
Esse novo comando desenha um triângulo equilátero cujos lados tem comprimento dado pelo parâmetro ''$lado''. Ele poderia ser usado assim em um algoritmo: <syntaxhighlight lang=text>
 
learn triangulo $lado {
 
repeat 3 {
 
  forward $lado
 
  turnleft 120
 
}
 
}
 
triangulo 50
 
triangulo 100
 
triangulo 100
 
</syntaxhighlight>
 
Nesse exemplo, três triângulos são desenhados: um com lado 50, outro com lado 100, e um terceiro com lado 150. Veja que o número escrito logo após o comando ''triangulo'' irá aparecer no lugar de ''$lado'' no algoritmo que desenha o triângulo.
 
  
== 13/09:  Algoritmos ==
+
Além de ser trabalhoso esse encadeamento de ''Se ... entao ... senao'', o algoritmo resultante fica pouco legível. Quer dizer, ele fica feio e difícil de entender.
  
Sobre a [[SOP-2010-2-sobral#Exerc.C3.ADcios:_desenho_de_figuras_geom.C3.A9tricas|atividade da aula passada]]:
+
Existe uma estrutura de decisão criada justamente para casos como esse, e que resulta em um algoritmo mais limpo e compreensível:
* [http://www.sj.ifsc.edu.br/~msobral/SOP/exemplos/kturtle/ Algoritmos dos fractais no Kturtle]
 
----
 
Os algoritmos que iremos criar serão compostos de:
 
* ''Estruturas de controle:'' estruturas para controlar a sequência de execução (realização) de algoritmos:
 
** ''Decisão:'' para realizar testes de forma a decidir que sequências de instruções devem ser executadas
 
** ''Repetição:'' para repetir sequências de instruções
 
* ''Estruturas de dados:'' estruturas para representar os dados a serem  processados por algoritmos
 
** ''Variáveis e constantes''
 
  
=== Portugol ===
+
<syntaxhighlight lang=text>
 +
// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
 +
// dia, nome do mês, ano.
 +
Inicio
 +
  inteiro dia, mes, ano
 +
  texto nome_mes
  
As aulas de Lógica de Programação usarão um software de auxílio ao ensino de algoritmos chamado [http://orion.ipt.pt/%7Emanso/Portugol/ Portugol], desenvolvido na [http://www.dei.estt.ipt.pt/portal/ Escola Superior de Engenharia do Instituto Politécnico de Tomar], em Portugal.
+
  escrever "Dia: "
* [http://www.sj.ifsc.edu.br/~msobral/SOP/portugol.html Guia rápido de instalação e utilização do Portugol].
+
  ler dia
 +
  escrever "Mes: "
 +
  ler mes
 +
  escrever "Ano: "
 +
  ler ano
  
==== Guia rápido de instalação e utilização do Portugol ====
+
  Escolhe mes
 
+
    caso 1:
Abaixo segue uma breve ajuda de como obtê-lo, instalá-lo e usá-lo. Esse guia assume que você esteja usando o Ubuntu Linux 9.04 ou superior.
+
      nome_mes <- "Janeiro"
 
+
    caso 2:
# Faça o [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/portugol23.tar.gz download] do Portugol.
+
      nome_mes <- "Fevereiro"
# Descompacte-o com o seguinte comando: <syntaxhighlight lang=bash>
+
    caso 3:
tar xzf portugol23.tar.gz
+
      nome_mes <- "Março"
 +
    caso 4:
 +
      nome_mes <- "Abril"
 +
    caso 5:
 +
      nome_mes <- "Maio"
 +
    caso 6:
 +
      nome_mes <- "Junho"
 +
    caso 7:
 +
      nome_mes <- "Julho"
 +
    caso 8:
 +
      nome_mes <- "Agosto"
 +
    caso 9:
 +
      nome_mes <- "Setembro"
 +
    caso 10:
 +
      nome_mes <- "Outubro"
 +
    caso 11:  
 +
      nome_mes <- "Novembro"
 +
    Defeito:  
 +
      nome_mes <- "Dezembro"
 +
  fimEscolhe
 +
 
 +
  escrever dia, " de ", nome_mes, " de ", ano
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
# Repare que existe agora um subdiretório portugol no diretório onde você o descompactou. Execute o Portugol com o seguinte comando: <syntaxhighlight lang=bash>
+
 
java -jar portugol/Portugol.jar
+
A estrutura de decisão ''escolhe ... caso'' tem uma certa flexibilidade. No exemplo abaixo, mostra-se a possibilidade de testar mais de um valor no mesmo ''caso'':
</syntaxhighlight> Obs: você precisará ter Java instalado. Caso não o tenha, execute o comando: <syntaxhighlight lang=bash>
 
sudo apt-get install openjdk-6-jre
 
</syntaxhighlight>
 
# Copie esse [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/exemplos/fluxogramas.cfg arquivo] para poder ver fluxogramas coloridos, e grave-o no memso diretório onde está o Portugol.
 
# Veja a [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/help/index.html ajuda] do Portugol, e use-a sempre que tiver dúvidas !
 
  
<br>A tela inicial do Portugol segue abaixo, junto com um programa demonstrativo.<br>
+
<syntaxhighlight lang=text>
 +
inicio   
 +
  caracter sexo   
  
[[Imagem:Editor-Portugol.png]]
+
  escrever "Qual o seu sexo (f/m):"   
 +
  ler sexo   
  
Exemplos de programas iniciais em Portugol:
+
  escrever "Olá "   
  
# ''Lendo um número e mostrando-o na tela em seguida:'' <syntaxhighlight lang=text>
+
  escolhe sexo       
Inicio
+
    caso "m", "M" :           
   inteiro x
+
      escrever "senhor"       
 +
    caso "f","F" :           
 +
      escrever "senhorita"       
 +
    defeito :            
 +
      escrever "Sexo indefinido"   
 +
   fimescolhe   
  
   Escrever "Digite um numero: ",
+
   escrever ", benvindo ao portugol"  
  Ler X
+
fim
  Escrever "Numero digitado: ", x
 
Fim
 
 
</syntaxhighlight>
 
</syntaxhighlight>
# ''Lendo dois números, somando-os e mostrando o resultado na tela:'' <syntaxhighlight lang=text>
 
Inicio
 
  inteiro x, y
 
  
  Escrever "Digite um numero: ",
+
=== Atividades ===
  Ler x
 
  Escrever "Digite outro numero: ",
 
  Ler y
 
  Escrever "Soma = ", x+y
 
Fim
 
</syntaxhighlight>O programa abaixo é equivalente:<syntaxhighlight lang=text>
 
Inicio
 
  inteiro x, y, z
 
  
  Escrever "Digite um numero: ",
+
# Faça um algoritmo que converta um número de 1 a 7 para o respectivo dia da semana (ex: 1 = domingo, 2 = 2a feira, e assim por diante).
  Ler x
+
# Faça uma calculadora com as quatro operações aritméticas. Sua calculadora deve ler (nesta ordem) o primeiro número, a operação aritmética (que deve ser informada usando o símbolo da respectiva operação: +, -, * ou /), e depois o segundo número. Ao final, seu algoritmo deve mostrar o resultado, ou uma mensagem de erro se a operação não for possível de realizar (ex: divisão por zero).
  Escrever "Digite outro numero: ",
+
# A previsão do tempo na costa brasileira pode ser feita de forma aproximada usando-se um barômetro e um termômetro. Uma estimativa com boa chance de acerto se baseia na tabela abaixo: <br>[[imagem:Previsao-barometro.png|800px]]<br><br>Faça um algoritmo que forneça uma estimativa da previsão do tempo usando essa tabela.
  Ler y
+
# Faça um algoritmo que mostre qual o último dia de um determinado mês informado pelo teclado. Caso seja informado o mês 2 (fevereiro), seu algoritmo deve identificar se é [http://pt.wikipedia.org/wiki/Bissexto ano bissexto] (assim o mês tem 29 dias), ou não (mês tem 28 dias). Obs: anos bissextos são dados pelas regras (segundo o [http://pt.wikipedia.org/wiki/Calend%C3%A1rio_Gregoriano calendário Gregoriano]):
  z <- x + y
+
## De 400 em 400 anos é ano bissexto.
  Escrever "Soma = ", z
+
## De 100 em 100 anos não é ano bissexto.
Fim
+
## De 4 em 4 anos é ano bissexto.<br>''Solução vista em aula: (sem considerar anos bissextos):''
</syntaxhighlight>
 
  
=== Atividades ===
+
== 27/04: Estruturas de decisão ==
  
# ''Média de três números:'' escreva um programa para calcular a média de três números.
+
=== Exercícios ===
# ''Média balanceada de três números:'' escreva um programa para calcular a média balanceada de três números.
 
# ''Sequência de Fibonacci:'' em matemática corresponde aos números: <syntaxhighlight lang=text>
 
1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, ...
 
</syntaxhighlight>... que pode ser descrita pela relação de recorrência <br><br><math>F_{n} = F_{n-1} + F_{n-2}</math><br><br>... com valores iniciais <math>F_{0} = 1</math> e <math>F_{1} = 1</math>.<br><br>Numerosas formas na natureza apresentam essa sequência, como neste girassol (cujas flores se dispõem em uma espiral):<br><br>[[Imagem:Sunflower.jpg]]  [[Imagem:espiral_fibonacci.png]] [[Imagem:FibonacciBlocks.png]]<br><br>Usando o Portugol escreva um programa que mostre os 10 primeiros números dessa sequência.
 
# ''Distância até o horizonte:'' escreva um programa que calcule a distância dos seus olhos até o horizonte. Assuma que a Terra é perfeitamente esférica, e que seu raio tem 6.378 km. Considere que você esteja no nível do mar (seus pés tocando a água do mar ;-), e que o horizonte esteja num mar perfeitamente liso. ''Dica: faça um diagrama desse problema para visualizar sua geometria, e use trigonometria para resolvê-lo.
 
  
== 14/09: Algoritmos ==
+
# Um estudo sobre sensibilidade de pessoas a temperaturas da água identificou que a maioria das pessoas considera fria a água com temperaturas abaixo de 25 graus, morna entre 25 e 30 graus, e quente acima de 30 graus. Escreva um algoritmo que mostre as palavras "fria", "morna" ou "quente" dependendo da temperatura da água que for informada.
 +
# Faça um algoritmo para fazer a divisão de dois números reais. Antes de dividi-los deve ser feito um teste de validade. Caso não seja possível dividi-los, deve ser mostrada uma mensagem de erro. Se for possível, deve-se mostrar o resultado da divisão.
 +
# Escreva um programa que leia cinco números do teclado e mostre os dois maiores números.
 +
# Faça um  programa que leia o nome de um usuário e sua senha, e em seguida informe se o acesso foi concedido ou negado. Os usuários possíveis de entrar são "joao", "maria" e "zé".
 +
# Jogo de Pedra, papel e tesoura: nesse jogo cada jogador faz sua escolha, e vence aquele que escolher um objeto que seja capaz de vencer o outro:
 +
#* Tesoura corta papel
 +
#* Pedra quebra tesoura
 +
#* Papel embrulha a pedra<br>Faça um algoritmo que pergunte ao usuário o objeto escolhido, e em seguida faça o computador escolher o seu próprio objeto (dica: use o gerador de números aleatórios visto no jogo do par ou ímpar). Finalmente o algoritmo deve informar o vencedor.
  
* Diagrama de blocos (fluxograma)
+
== 28/04: Estruturas de repetição ==
* Variáveis e constantes
 
  
=== Fluxogramas ===
+
Alguns algoritmos vistos anteriormente possuem sequências repetitivas de instruções. Por exemplo, o algoritmo da média de quatro avaliações:
  
Diagramas de bloco para auxiliar a descrição de algoritmos. Ajudam na compreensão do algoritmo, por poder visualizar o fluxo de execução.
+
<syntaxhighlight lang=text>
 +
Inicio
 +
  real m1, m2, m3, m4
 +
  real media
  
 +
  escrever "Avaliação 1:"
 +
  Ler m1
 +
  escrever "Avaliação 2:"
 +
  Ler m2
 +
  escrever "Avaliação 3:"
 +
  Ler m3
 +
  escrever "Avaliação 4:"
 +
  Ler m4
  
[[imagem:Fluxograma-soma.png]] Fluxograma para o algoritmo da média de trẽs números.
+
  media <- (m1 + m2 + m3 + m4) / 4
  
<br>
+
  escrever "Média: ", media
 +
Fim
 +
</syntaxhighlight>
  
'''Blocos de uso mais comum'''<br>
+
O algoritmo acima repete quatro vezes a sequência que lê uma nota do teclado. Porém há uma forma de expressar a repetição de sequências de instruções, usando-se o que se chama de ''estrutura de repetição''. A estrutura de repetição '''''enquanto''''' ''condição'' '''''faz''''' repete todas as instruções enquanto a condição for verdadeira. A condição é escrita como uma expressão lógica, da mesma forma que na estrutura de decisão '''''se''''' ''condição'' '''''então ... senão'''''. Veja como fica o algoritmo acima ao ser reescrito para usar essa estrutura de repetição:
{| border="1" cellpadding="2"
 
!Bloco
 
!Descrição
 
!Exemplo
 
|-
 
|[[Imagem:Inicio.png]]||Inicio do fluxograma||
 
|-
 
|[[Imagem:Processamento.png]]||Processamento||
 
|-
 
|[[Imagem:Entrada.png]]||Entrada de dados (ler do teclado)||
 
|-
 
|[[Imagem:Saida.png]]||Saída de dados (mostrar na tela)||
 
|-
 
|[[Imagem:Decisao.png]]||Decisão (testar uma condição e bifurcar)||[[imagem:Ex-decisao.png]]
 
|-
 
|[[Imagem:Conector.png]]||Conector (juntar dos ou mais ramos do fluxograma)||[[imagem:Ex-conector.png]]
 
|-
 
|[[Imagem:Fim.png]]||Fim||
 
|}
 
  
Obs: [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/exemplos/fluxogramas.cfg Arquivo de configuração das cores do fluxograma ] do Portugol.
+
<syntaxhighlight lang=text>
 
 
=== Variáveis e constantes ===
 
 
 
* '''Variável:''' capaz de guardar um dado a ser usado no algoritmo. Pode ser entendida como uma caixa, onde se coloca um dado e se pode consultá-lo quantas vezes for necessário. O dado pode ser modificado (substituído por outro). Exemplo em Portugol: <syntaxhighlight lang=text>
 
 
Inicio
 
Inicio
   inteiro anos, dias
+
   constante inteiro NUMEROS <- 4
 +
  real m
 +
  real media <- 0
 +
  inteiro contador <- 0
  
   Escrever "Quantos anos se passaram ? "
+
   enquanto contador < NUMEROS faz 
  Ler anos
+
    escrever "Avaliação ", contador, ":"
  dias <- anos * 365
+
    ler m
  Escrever "... então se passaram ", dias, " dias"
+
    media <- media + m
Fim
+
    contador <- contador + 1
</syntaxhighlight> Nesse exemplo há duas variáveis: ''dias'' e ''anos''
+
   fimEnquanto
* '''Constante:''' semelhante à variável, porém o dado armazenado não pode ser modificado. Exemplo em Portugol:<syntaxhighlight lang=text>
 
Inicio
 
  constante inteiro diasPorAno <- 365
 
   inteiro anos, dias
 
  
   Escrever "Quantos anos se passaram ? "
+
   escrever "Média: ", media/NUMEROS
  Ler anos
 
  dias <- anos * diasPorAno
 
  Escrever "... então se passaram ", dias, " dias"
 
 
Fim
 
Fim
</syntaxhighlight> Nesse exemplo há uma constante: ''diasPorAno''
+
</syntaxhighlight>
  
<br>
+
Esse algoritmo funciona somente para médias de quatro números. Porém calcular a média de qualquer quantidade de números usa a mesma lógica: ler os números, somá-los e dividir o total pela quantidade de números lidos. Por exemplo, para fazer com que ele calcule a média de 7 números é necessário escrevê-lo assim:
Variáveis e constantes devem ser declaradas antes de serem usadas (algumas poucas linguagens, como [http://www.python.org Python] e [http://www.perl.org Perl], não exigem isto). A declaração consiste do tipo e identificador da variável. O tipo corresponde ao tipo de valor que pode ser guardado, e o identificador é o nome da variável. No exemplo abaixo:
 
  
 
<syntaxhighlight lang=text>
 
<syntaxhighlight lang=text>
   constante inteiro diasPorAno <- 365
+
Inicio
   inteiro anos, dias
+
   constante inteiro NUMEROS <- 7
 +
  real m
 +
  real media
 +
   inteiro contador <- 0
 +
 
 +
  enquanto contador < NUMEROS faz 
 +
    escrever "Avaliação ", contador, ":"
 +
    ler m
 +
    media <- media + m
 +
    contador <- contador + 1
 +
  fimEnquanto
 +
 
 +
  escrever "Média: ", media/NUMEROS
 
Fim
 
Fim
 
</syntaxhighlight>
 
</syntaxhighlight>
  
No  exemplo acima, há duas variáveis do tipo ''inteiro'', e seus identificadores são ''dias'' e ''anos''. O tipo ''inteiro'' indica que essas variáveis podem guardar somente números inteiros.
+
Note que o algoritmo praticamente não foi modificado, pois somente se alterou o valor da constante NUMEROS.  
 +
 
 +
Esses dois algoritmos podem também ser visualizados como fluxogramas, como mostrado na figura abaixo:
 +
 
 +
[[imagem:Ex-repeticao.png]]
  
'''''Tipos de variáveis e constantes no Portugol:'''''
+
A estrutura de repetição junto com a sequência de instruções a ser repetida é comumente chamada de '''laço''. No fluxograma acima o laço está destacado em vermelho, e no algoritmo em Portugol o laço aparece em:
  
{| border="1" cellpadding="2"
+
<syntaxhighlight lang=text>
!Tipo
+
  enquanto contador < NUMEROS faz  
!Descrição
+
    escrever "Avaliação ", contador, ":"
!Exemplo
+
    ler m
|-
+
    media <- media + m
|Inteiro||Número inteiro entre -2 147 483 648 e 2 147 483 647 || Inteiro x <- 10
+
    contador <- contador + 1
|-
+
  fimEnquanto
|Real||Número real entre -1.7 E 308 e 1.7 E 308 || Real y <- 10.789
+
</syntaxhighlight>
  |-
 
|Lógico||Valor booleano, com valores "Verdadeiro" e "Falso" || Logico ok <- Falso
 
|-
 
|Caracter||Um caractere da [http://pt.wikipedia.org/wiki/ASCII tabela ASCII] || Caracter letra <- "A"
 
|-
 
|Texto||Uma sequência de caracteres (ou ''string'') || Texto palavra <- "um teste"
 
|}
 
  
A declaração de constantes é semelhante à de variáveis, bastanto prefixá-las com a palavra-chave ''constante''.
+
----
  
=== Expressões aritméticas ===
+
Um outro exemplo ainda mais simples é mostrar na tela uma contagem numérica:
  
Um conjunto de operações sobre variáveis, constantes e funções numéricas, e que gera um determinado resultado numérico.
+
<syntaxhighlight lang=text>
 +
inicio
 +
    inteiro contador
 +
    contador <- 0
  
Exemplos de expressões aritméticas: <syntaxhighlight lang=text>
+
    enquanto contador < 10 faz
# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
+
        escrever contador , " "
3600*horas + 60*minutos + segundos
+
        contador <- contador + 1
 
+
    fimenquanto
# Uma expressão que calcula a velocidade instantânea, segundo um MRV
+
fim
vel_inicial + aceleracao*tempo;
 
 
 
# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
 
raiz(x^2 + y^2)
 
 
</syntaxhighlight>
 
</syntaxhighlight>
  
Os resultados de expressões podem ser mostrados na tela, ou armazenados em variáveis:
+
Ao executar esse algoritmo, tem-se como resultado:
  
 
<syntaxhighlight lang=text>
 
<syntaxhighlight lang=text>
# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
+
0 1 2 3 4 5 6 7 8 9
segundos <- 3600*horas + 60*minutos + segundos
 
 
 
# Uma expressão que calcula a velocidade instantânea, segundo um MRV
 
escrever 'Velocidade no instante ', tempo, ' = ', vel_inicial + aceleracao*tempo;
 
 
 
# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
 
modulo <- raiz(x^2 + y^2)
 
 
</syntaxhighlight>
 
</syntaxhighlight>
  
Repare que uma expressão fica sempre do lado direito, quando atribuída a uma variável. A expressão é primeiro calculada, e em seguida seu resultado é armazenado na variável:
+
Seu fluxograma está mostrado a seguir:
  
<syntaxhighlight lang=text>
+
[[imagem:Flux-Contagem.png]]
segundos <- 3600*horas + 60*minutos + segundos
 
</syntaxhighlight>
 
  
==== Operadores aritméticos ====
 
  
Expressões aritméticas sao compostas por números e operadores aritméticos:
+
Repare no uso de uma variável auxiliar ''contador'' para controlar a quantidade de repetições. Essa é uma técnica comum para
 +
controlar a estrutura de repetição, e já foi usada no exemplo da média. Ela pode também ser combinada com outras condições, como mostrado abaixo:
  
Obs: para os exemplos abaixo são usadas estas variáveis: <syntaxhighlight lang=text>
+
<syntaxhighlight lang=text>
Real x, area, lado
+
inicio
inteiro dias, horas
+
    inteiro contador <- 1
 +
    caracter opcao <- "s"
 +
       
 +
    enquanto ((opcao = "s") ou (opcao = "S")) e contador < 11 faz
 +
        escrever contador, "\n"
 +
        contador <- contador + 1
 +
       
 +
        escrever "\nContinuar (S/N) ? "
 +
        ler opcao
 +
    fimenquanto
 +
   
 +
    escrever "Terminou com contador = " , contador
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
  
{| border="1" cellpadding="2"
+
Nesse exemplo se usa um laço para mostrar uma contagem de 1 a 10. Porém há a possibilidade de encerrar o algoritmo. Observe a condição para continuidade do laço: ela é verdadeira somente se ''contador < 11'' e se a variável ''opcao'' for igual a "s" ou "S".
!Operador
+
 
!Descrição
+
Finalmente, apesar de ser comum o uso de contadores para controle do laço, pode-se usar qualquer condição para essa finalidade. Então podem existir laços que não usam contadores para controlar a quantidade de repetições, como no próximo exemplo:
!Exemplo
+
 
|-
+
<syntaxhighlight lang=text>
|<nowiki>+</nowiki>||Adição|| x <- x + 1
+
inicio
|-
+
    constante texto segredo <- "secreto"
|<nowiki>-</nowiki>||Subtração|| x <- x - 1
+
    texto senha
|-
+
   
|*||Multiplicação|| x <- x*x*x
+
    ler senha
|-
+
    enquanto senha =/= segredo faz
|/||Divisão|| dias <- horas / 24
+
        escrever "Voce continua preso ! Digite a senha correta: "
|-
+
        ler senha
|%||Módulo (resto de divisão)|| horas <- horas % 24
+
    fimenquanto
|-
+
    escrever "Voce foi liberado ..."
|^||Potenciação|| area <- lado^2
+
fim
|}
+
</syntaxhighlight>
  
Precedência dos operadores (nesta ordem): ^, *, /, %, + e -
+
O fluxograma desse algoritmo segue abaixo:
  
A precedência pode ser modificada com o uso de parênteses. Ex:
+
[[imagem:Flux-Senha.png]]
  
<syntaxhighlight lang=text>
+
=== Atividades ===
escrever 1 + 2 * 3
 
escrever (1 + 2)*3
 
</syntaxhighlight>
 
  
No Portugol, existem também algumas [http://www.sj.ifsc.edu.br/%7Emsobral/SOP/soft/help/funcoes_biblioteca.html funções úteis], como a função '''''raiz''''': <syntaxhighlight lang=text>
+
# Escreva um  algoritmo que mostre a tabuada de um número fornecido pelo teclado. Esse número deve estar entre 1 e 10.
r <- raiz(x^2 + y^2)
+
# Aproveite o algoritmo anterior para mostrar a tabuada de todos os números entre 1 e 10.
 +
# Modifique o exemplo da média para que a quantidade de números a serem lidos seja previamente informada pelo teclado.
 +
# Escreva  algoritmos que mostrem as seguintes sequências de números:
 +
#* Números pares menores que 100
 +
#* Números ímpares menores que 100
 +
#* Números pares de 100 até 4 (contagem decrescente)
 +
#* Múltiplos de 3 menores que 100
 +
#* Múltiplos de 4 e múltiplos de 5 menores que 100
 +
#* Múltiplos comuns menores que 1000 de dois números fornecidos pelo teclado (ex: se forem  fornecidos 6 e 8, deve aparecer 24, 48, 72, ...)<syntaxhighlight lang=text>
 +
inicio
 +
    inteiro n <- 0 , n1 , n2, mmc
 +
   
 +
    escrever "Primeiro numero : "
 +
    ler n1
 +
    escrever "Segundo numero : "
 +
    ler n2
 +
   
 +
    // descobre o mmc de n1 e n2
 +
    se n1 > n2 entao
 +
      mmc <- n1
 +
    senao
 +
      mmc <- n2
 +
    fimse
 +
   
 +
    enquanto mmc % n1 =/= 0 OU mmc % n2 =/= 0 faz
 +
      mmc <- mmc + 1
 +
    fimenquanto
 +
   
 +
    // mostra os multiplos de mmc que sejam <= 1000
 +
    enquanto n <= 1000 faz
 +
        escrever "\n " , n
 +
        n <- n + mmc
 +
    fimenquanto
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
# Escreva um algoritmo para calcular o MMC (Mínimo Múltiplo Comum) de dois números. O MMC de dois números é um número que pode ser dividido por qualquer um deles sem deixar resto. Ex:
O resultado de expressões aritméticas depende dos tipos numéricos das variáveis e constantes: <syntaxhighlight lang=text>
+
#* MMC de 8 e 6 = 24
 +
#* MMC de 3 e 5 = 15
 +
#* MMC de 3 e 27 = 27
 +
# Escreva um  algoritmo para calcular o MDC (Máximo Divisor Comum) de dois números. O MDC de dois números é um número capaz de dividi-los sem deixar resto. Ex:
 +
#* MDC de 24 e 16 = 8
 +
#* MDC de 5 e 3 = 1
 +
#* MDC de 60 e 150 = 30<syntaxhighlight lang=text>
 
inicio
 
inicio
  real x
+
    inteiro n1 , n2, mdc
  inteiro y
+
  inteiro resultadoInteiro
+
    escrever "Primeiro numero : "
  real resultadoReal
+
    ler n1
 
+
    escrever "Segundo numero : "
  x <- 9
+
    ler n2
  y <- 9
+
 
+
    // descobre o mmc de n1 e n2
  escrever "O resultado de uma expressão aritmética depende dos tipos das variáveis e constantes\n"
+
    se n1 > n2 entao
  escrever "usadas na expressão. Se forem todas inteiras, então o resultado será inteiro.\n"
+
      mdc <- n2
  escrever "Veja este exemplo: \n"
+
    senao
  escrever "Variável inteira y=", y
+
      mdc <- n1
  escrever "\nExpressão: y/2=", y/2
+
    fimse
 
 
  escrever "\n\nNeste segundo exemplo, repete-se a mesma expressão, porém usando-se uma\n"
 
  escrever "variável real:\n"
 
  escrever "Variável real x=", x
 
  escrever "\nExpressão: x/2=", x/2
 
 
 
  x <- 4
 
  y <- 5
 
  escrever "\n\nSe as variáveis de diferentes tipos forem combinadas, o resultado da\n"
 
  escrever "expressão será real:\n"
 
  escrever "Variável real x=", x, " e inteira y=", y
 
  escrever "\nExpressão: (x+y)/2=", (x+y)/2
 
  
  escrever "\n\nNo entanto, se uma expressão tiver um resultado real, mas este for\n"
+
    enquanto mdc > 1 E NAO (n1 % mdc = 0 E n2 % mdc = 0) faz
  escrever "atribuído a uma variável inteira, então apenas a parte inteira será guardada:\n"
+
      mdc <- mdc - 1
  escrever "Variável real x=", x, " e inteira y=", y
+
    fimenquanto
  y <- (x+y)/2
 
  escrever "\nExpressão: y <- (x+y)/2 ... após executada, y=", y
 
 
    
 
    
 +
    escrever "MDC=", mdc
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
# Escreva um algoritmo que leia até 10 números do teclado, e informe ao final qual o maior e o menor deles.
 +
# Escreva um algoritmo que teste se um número informado pelo teclado é primo.
 +
# Aproveite o algoritmo do ítem anterior para mostrar todos os números primos menores que 1000.
 +
# Modifique novamente o exemplo da média para que ela funcione para um quantidade de números desconhecida de antemão. Quer dizer, o algoritmo deve ler os números para calcular a média, mas não sabe quantos números existem (e isto não pode ser informado pelo teclado).
 +
# Modifique o exemplo da senha para que o usuário tenha somente três tentativas permitidas para digitar a senha correta. Caso ao final as três senhas estejam erradas, o algoritmo deve informar que a conta foi bloqueada.
 +
# Usando o algoritmo de teste de número primo, faça um novo algoritmo que fatore um número. Ex: a fatoração do número 15 resulta em  3 e 5 (3 X 5), e a fatoração de 54 resulta em 2, 3, 3 e 3 (2 X 3 X 3 X 3).
 +
# Usando estrutura de repetição, escreva um algoritmo que converta um número de decimal para binário. Assuma um número binário com 16 bits.
 +
# Faça um algoritmo para descobrir os dois maiores números dentre 6 números lidos do teclado.
 +
# Faça um algoritmo para converter um número entre 0 e 16777215 para sua representação em hexadecimal.
 +
# Escreva o algoritmo para uma calculadora aritmética (operações +, -, * e /) que lê continuamente do teclado números e operadores. A calculadora deve ler continuamente uma sequência do tipo ''número operador_aritmético'', até que o operador informado seja ''='', quando então o resultado da conta deve ser mostrado na tela. Ex:
 +
#* 1 + 2 * 5 / 3 = 5
 +
#* 2 * 5 - 1 / 3 = 3
  
=== Atividade ===
+
== 04/05: Estruturas de repetição ==
  
Para os exercícios abaixo, desenhe o fluxograma e escreva o algoritmo no Portugol.
+
Resolução de exercícios (continuando aula anterior).
  
# Faça um algoritmo que, usando somente duas variáveis, calcule a média de quatro números.
+
== 05/05: Estruturas de repetição ==
# Escreva um algoritmo que calcule as raizes de uma equação de 2o grau.
 
# Um equipamento conta o tempo desde que foi ligado. No entanto, essa contagem é feita em segundos. Faça um algoritmo que converta o valor desse contador para horas, minutos e segundos. Ex: se o contador tiver o valor 43397, o seu programa deve mostrar: 12:03:17.
 
# Crie um conversor de decimal para binário (limite: 4 bits). Ex.: tendo o número 10 (decimal) de entrada, deve-se obter o número 1010 (binário) de saída.
 
# Escreva um algoritmo que leia o nome, sobrenome e idade de uma pessoa, e escreva na tela (no lugar dos colchetes devem aparecer os valores lidos):<syntaxhighlight lang=text>
 
Nome: [sobrenome, nome]
 
Idade: [idade] anos</syntaxhighlight>
 
# Crie um gerador automático de código Logo para desenhar polígonos:
 
#* Ler um número do usuário.
 
#* Calcular o ângulo interno.
 
#* Apresentar como resultado o código [http://pt.wikipedia.org/wiki/Logo Logo]  a ser inserido no programa '''kturtle'''. Ex.: se o usuário digitar o número 3, o código será de um triângulo equilátero, cuja soma dos ângulos internos é 180 - logo, cada ângulo será de 60 graus (no Logo, lembre-se que o ângulo a ser usado é o complemento de 180, conforme visto em aula). Se o usuário digitar 5 ao invés de 3, teremos um pentágono. Abaixo temos soluções para triângulo e pentágono:
 
reset
 
repeat 3
 
{
 
  forward 100
 
  turnright 120
 
}
 
  
reset
+
Continuando a [[PRG-2011-1-sobral#Atividades_4|resolução de exercícios ...]]
repeat 5
 
{
 
  forward 100
 
  turnright 72
 
}
 
  
 +
=== Uma pequena avaliação ===
  
 +
Resolva o seguinte problema em 15 minutos:
  
== 20/09: Estruturas de decisão ==
+
'''Contador de moedas:''' faça um programa que leia valores de moedas e some as quantidades de tipos de moedas informadas. Por exemplo, se o usuário digitar 25, 50, 25, 5, 10, 5, o programa deve informar: 2 moedas de 5 centavos, 1 moeda de 10 centavos, 2 moedas de 25 centavos, 1 moeda de 50 centavos. São aceitos apenas valores de moedas de 1, 5, 10, 25 e 50 centavos. Seu programa deve ler 10 valores de moedas, e então apresentar o resultado.
  
O exemplo abaixo está contido no livro ''Classification and Regression Trees'', de Leo Breiman, Jerome Friedman, Richard Olshen e Charles Stone, editora Chapman & Hall, 1984.
+
<syntaxhighlight lang=text>
 +
inicio
 +
  inteiro m1, m5, m10, m25, m50
 +
  inteiro moeda
 +
  inteiro n
 +
   
 +
  enquanto n < 10 faz
 +
    escrever "Moeda: "
 +
    ler moeda
 +
   
 +
    escolhe moeda
 +
      caso 1:
 +
        m1 <- m1 + 1
 +
      caso 5:
 +
        m5 <- m5 + 1
 +
      caso 10:
 +
        m10 <- m10 + 1
 +
      caso 25:
 +
        m25 <- m25 + 1
 +
      caso 50:
 +
        m50 <- m50 + 1
 +
      defeito:
 +
        escrever "Valor invalido ...\n"
 +
        n <- n - 1
 +
    fimescolhe
 +
    n <- n + 1
 +
  fimenquanto
  
Na Universidade da California, no centro Médico de San Diego, quando um paciente com ataque cardíaco é recebido, dezenove variáveis são medidas ao longo das primeiras 24 horas. Essas incluem pressão sanguínea, idade, e 17 outras variáveis ordenadas e binárias (booleanas) que sumarizam os sintomas médicos considerados importantes indicadores da condição do paciente.
+
  escrever "Moedas de 1 centavo: ", m1
 +
  escrever "\nMoedas de 5 centavos: ", m5
 +
  escrever "\nMoedas de 10 centavos: ", m10 
 +
  escrever "\nMoedas de 25 centavos: ", m25
 +
  escrever "\nMoedas de 50 centavos: ", m50
 +
fim
 +
</syntaxhighlight>
  
O objetivo  de um  estudo médico feito entre os anos 70 e 80 foi o desenvolvimento de um método para identificar pacientes de alto risco (que não sobreviverão ao menos 30 dias) com  base nos dados obtidos nas primeiras 24 horas. O diagrama abaixo mostra uma regra de classificação que foi produzida nesse estudo. A letra F  significa que não há um risco alto, e a letra G quer dizer paciente de alto risco.
+
== 11/05: Estruturas de repetição ==
  
[[imagem:Crt-decisao.png]]
 
  
Essa regra classifica os pacientes como F ou G dependendo de respostas do tipo sim/não a no máximo três perguntas. Como  seria um algoritmo que a implementasse ?
+
=== Variáveis multidimensionais ===
  
----
+
Em matemática existem matrizes e vetores, que são variáveis multidimensionais. Por exemplo, uma matriz 2 x 2 (2 linhas e 2 colunas) pode ser:
  
Algoritmos para resolver problemas como  o exemplo acima dependem de se poder expressar a '''''tomada de decisões'''''. Deve ser possível testar '''''condições''''', como  por exemplo "''a idade é maior que 62.5 anos ?''", e executar instruções dependendo do resultado do teste. Em Lógica de Programação, e em linguagens de programação em geral, isso se faz com '''''estruturas de decisão'''''.
+
<math>A = \begin{pmatrix}
 +
1 & 6 \\
 +
3 & 5
 +
\end{pmatrix}</math>
  
'''''Estruturas de decisão''''' possibilitam que se executem diferentes sequências de instruções de um programa, dependendo de uma condição a ser avaliada. Por exemplo, um jogo poderia armazenar a maior pontuação já obtida, e verificar se foi ultrapassada ao final de cada partida:
+
Vetores são matrizes unidimensionais, portanto possuem somente uma linha ou uma coluna:
  
<syntaxhighlight lang=text>
+
<math>v = \begin{pmatrix}
  Se pontuacao > recorde então
+
1 & 6 & 2 & 18 & 5
    recorde <- pontuação
+
\end{pmatrix}</math>
  FimSe
 
</syntaxhighlight>
 
  
O exemplo em Portugol acima mostra a estrutura de decisão ''Se condição entao comandos Fimse''. O equivalente em  fluxograma é:
+
Cada elemento em uma matriz ou vetor é identificado pela sua posição. Por exemplo, na posição ''1, 2'' da matriz A acima está o valor 6, e na posição ''4'' do vetor ''v'' está o valor 18. Assim, a matriz A do exemplo pode ser entendida da seguinte forma:
  
[[imagem:Seentao.png]]
+
<math>A = \begin{pmatrix}
 +
A_{11} & A_{12} \\
 +
A_{21} & A_{22}
 +
\end{pmatrix}</math>
  
Veja o próximo exemplo:
+
... e o vetor ''v'' do exemplo:
  
<syntaxhighlight lang=text>
+
<math>v = \begin{pmatrix}
  Se conceito > 70 entao
+
v_{1} & v_{2} & v_{3} & v_{4} & v_{5}
    escrever 'Voce esta aprovado'
+
\end{pmatrix}</math>
  senao
 
    escrever 'Desta vez não deu ... tente de novo !'
 
  Fimse
 
</syntaxhighlight>
 
  
O uso de ''Se condição entao comandos Senao comandos Fimse'' possibilita que se execute uma sequência de comandos se a condição for verdadeira, e outra sequência se for falsa. Seu equivalente em  fluxograma é:
+
Nas linguagens de programação em geral existe um conceito muito parecido, chamado de variáveis multidimensionais ou simplesmente ''matrizes'' (''arrays'' em inglês). Para exemplificar, no Portugol se poderiam definir a matriz ''A'' e o vetor ''v'':
  
[[imagem:Seentaosenao.png]]
+
<syntaxhighlight lang=text>
 +
inicio
 +
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
 +
  inteiro v[5] <- {1, 6, 2, 18, 5}
 +
 
 +
  escrever "Valor de A[0][0]: ", A[0][0], "\n"
 +
  escrever "Valor de A[0][1]: ", A[0][1], "\n"
 +
  escrever "Valor de A[1][0]: ", A[1][0], "\n"
 +
  escrever "Valor de A[1][1]: ", A[1][1], "\n"
  
Para fazer um bom uso de estruturas de decisão deve-se primeiro conseguir identificar as condições a serem avaliadas. Condições são escritas com expressões lógicas, e estas são compostas de operadores lógicos e relacionais aplicados a variáveis e constantes.
+
  escrever "Valor de v[0]: ", v[0], "\n"
 +
  escrever "Valor de v[1]: ", v[1], "\n"
 +
  escrever "Valor de v[2]: ", v[2], "\n"
 +
  escrever "Valor de v[3]: ", v[3], "\n"
 +
  escrever "Valor de v[4]: ", v[4], "\n"
  
=== Condições ===
+
fim
 +
</syntaxhighlight>
  
Condições representam os testes que precisam ser feitos com dados do problema. Esses testes são expressados por '''''expressões lógicas''''', cujos resultados são ''verdadeiro'' ou ''falso''. Expressões lógicas são construídas usando-se '''''operadores lógicos''''' e '''''operadores relacionais''''' para comparar e avaliar os valores de variáveis e constantes.
+
A declaração da matriz se faz como uma variável comum, porém indicando-se suas dimensões:
  
Obs: para os exemplos abaixo são usadas estas variáveis: <syntaxhighlight lang=text>
+
<syntaxhighlight lang=text>
Logico correto, multa, aprovado, barato, bombear_agua
+
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
Logico descartar, baixo, reprovado, erro, enviado, recebido
 
inteiro erros, pontuacao, preco, endereco, velocidade
 
Real faltas, nivel_agua, altura
 
 
</syntaxhighlight>
 
</syntaxhighlight>
  
==== Operadores relacionais ====
+
Veja que a matriz ''A'' foi inicializada na declaração.
  
Operadores relacionais são usados para comparar valores, e estão listados  na tabela abaixo. Cada comparação feita com esses operadores forma uma ''expressão lógica simples''. Portanto, o resultado de uma comparação é um valor lógico que pode ser ''verdadeiro'' ou ''falso''.
+
O acesso aos elementos da matriz se faz usando-se ''índices'', que funcionam como coordenadas dos elementos que se quer acessar. Os índices iniciam em ''0'':
  
{| border="1" cellpadding="2"
+
<syntaxhighlight lang=text>
!Operador
+
  # índice 0,0
!Descrição
+
  escrever "Valor de A[0][0]: ", A[0][0], "\n"
!Exemplo
+
</syntaxhighlight>
|-
 
|=||Igualdade|| correto <- (erros = 0)
 
|-
 
|>||Maior|| multa <- (velocidade > 80)
 
|-
 
|>=||Maior ou igual|| aprovado <- (pontuacao >= 70)
 
|-
 
|<||Menor|| barato <- (preco < 100)
 
|-
 
|<=||Menor ou igual|| bombear_agua <- (nivel_agua <= 0.7)
 
|-
 
|=/=||Diferente|| descartar <- (endereco =/= 12345)
 
|}
 
  
==== Operadores lógicos ====
+
O exemplo acima pode ser reescrito usando-se estruturas de repetição:
  
Operadores lógicos são usados para combinar expressões lógicas simples, criando assim ''expressões lógicas compostas'', e estão listados  na tabela abaixo.
+
<syntaxhighlight lang=text>
 +
inicio
 +
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
 +
  inteiro v[5] <- {1, 6, 2, 18, 5}
 +
  inteiro i, j
  
{| border="1" cellpadding="2"
+
  enquanto i < 2 faz
!Operador
+
    j <- 0
!Descrição
+
    enquanto j < 2 faz
!Exemplo
+
      escrever "Valor do elemento de A na posição ", i, ", ", j, ": ", A[i][j], "\n"
|-
+
      j <- j + 1
|NAO||Negação|| baixo <- NAO (altura > 1.8)
+
    fimenquanto
|-
+
    i <- i + 1
|E||Conjunção|| aprovado <- NAO (conceito = "D") E (faltas <= 0.25)
+
  fimenquanto
|-
 
|OU||Disjunção|| reprovado <- (conceito = "D") OU (faltas > 0.25)
 
|-
 
|XOU||Disjunção exclusiva|| erro <- enviado XOU recebido
 
|}
 
  
Precedência dos operadores (nesta ordem): NAO, E, OU e XOU
+
  i <- 0
 +
  enquanto i < 5 faz
 +
    escrever "valor de v na posição ", i, ": ", v[i], "\n"
 +
    i <- i + 1
 +
  fimenquanto
  
Lembre que a precedência pode ser modificada com o uso de parênteses.
+
fim
 +
</syntaxhighlight>
  
=== Atividades ===
+
Como se vê, matrizes e vetores casam perfeitamente com estruturas de repetição, e são úteis para a resolução de inúmeros problemas. Por exemplo, o problema de mostrar o nome do mês a partir do número do mês, feito anteriormente com ''escolhe ... caso'' pode ser feito assim com um vetor:
 
 
# Faça um programa que leia um número e então informe se ele é par ou ímpar.
 
# Um radar de trânsito faz a medição de velocidade de veículos e, dependendo do valor, calcula a multa a ser aplicada. Em uma determinada via esse radar foi configurado da seguinte forma:
 
#* Se a velocidade for maior que 80 km/h, a multa é de R$ 360.
 
#* Se a velocidade for maior que 60 km/h, a multa é de R$ 180.
 
#* Se a velocidade for menor ou igual a 60 km/h, não há multa.<br><br>Escreva um algoritmo que calcule a multa de acordo com a velocidade de um veículo.
 
# Faça um algoritmo que leia três números do teclado, e mostre o maior e menor números.
 
# Escreva um  algoritmo  que faça o [[SOP-2010-2-sobral#20.2F09:_Estruturas_de_decis.C3.A3o|teste de risco de paciente cardíaco]], mostrado no início da aula.
 
# Um estudo sobre sensibilidade de pessoas a temperaturas da água identificou que a maioria das pessoas considera fria a água com temperaturas abaixo de 25 graus, morna entre 25 e 30 graus, e quente acima de 30 graus. Escreva um algoritmo que mostre as palavras "fria", "morna" ou "quente" dependendo da temperatura da água que for informada.
 
# Em uma rede de computadores, o ''firewall'' restringe o acesso a Internet dependendo do horário e do tipo de usuário que faz o acesso. Os tipos de usuário abaixo têm as seguintes restrições:
 
#* Funcionario: apenas entre 0:00 e 7:30, entre 18:30 e 0:00, e entre 12:00 e 13:30
 
#* Financeiro: qualquer horario
 
#* Diretoria: qualquer horário<br>Escreva um algoritmo que informe se um acesso foi permitido, em função do horário e do tipo de usuário.
 
# Modifique o algoritmo acima para adicionar a seguinte restrição:
 
#* Funcionario: nao pode acessar sites de jornal (ex: www.rbs.com.br)
 
#* Financeiro: nao pode acessar sites de jornal durante o expediente
 
#* Diretoria: sem restrições a sites
 
# Faça um algoritmo para fazer a divisão de dois números reais. Antes de dividi-los deve ser feito um teste de validade. Caso não seja possível dividi-los, deve ser mostrada uma mensagem de erro. Se for possível, deve-se mostrar o resultado da divisão.
 
# Faça um jogo de par ou ímpar, em que o jogador aposta contra o computador. O jogador deve digitar um número entre 0 e 5 e optar entre par ou ímpar. O computador deve sortear um número também entre 0 e 5. Se a paridade da soma dos números do jogador e do computador for a mesma que o jogador optou, então ele ganha a partida, senão o computador vence.
 
# Escreva um algoritmo para identificar se há, dentre três palavras lidas do teclado, ao menos duas palavras distintas que pertençam ao conjunto {azul, preto, vermelho}. Exemplos:
 
#* Se o usuário digitar ''verde'', ''preto'', ''vermelho'', o programa deve mostrar na tela ''verdadeiro''
 
#* Se o usuário digitar ''verde'', ''preto'', ''preto'', o programa deve mostrar na tela ''falso''
 
#* Se o usuário digitar ''azul'', ''preto'', ''azul'', o programa deve mostrar na tela ''verdadeiro'' <br>''Solução vista em aula:''<syntaxhighlight lang=text>
 
// Neste algoritmo deve-se identificar se ao menos duas palavras
 
// diferentes têm os valores azul, preto ou vermelho.
 
// A variável inteira "cont" conta quantas palavras diferentes
 
// tem esses valores.
 
  
 +
<syntaxhighlight lang=text>
 
inicio
 
inicio
    texto p1 , p2 , p3
+
  texto nome_mes[12] <- {"Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho",
    inteiro cont <- 0
+
                        "Julho", "Agosto", "Setembro", "Outubro", "Novembro",
   
+
                        "Dezembro"}
    escrever "Palavra 1: "
+
   inteiro dia, mes, ano
    ler p1
 
    escrever "Palavra 2: "
 
    ler p2
 
    escrever "Palavra 3: "
 
    ler p3
 
   
 
    // ao menos uma palavra é "azul" ?
 
    se p1 = "azul" OU p2 = "azul" OU p3 = "azul" entao
 
      cont <- 1
 
    fimse
 
 
 
    // ao menos uma palavra é "preto" ?
 
    se p1 = "preto" OU p2 = "preto" OU p3 = "preto" entao
 
      cont <- cont + 1
 
    fimse
 
 
 
    // ao menos uma palavra é "vermelho" ?
 
    se p1 = "vermelho" OU p2 = "vermelho" OU p3 = "vermelho" entao
 
      cont <- cont + 1
 
    fimse
 
 
 
    // se duas ou mais palavras diferentes têm aqueles valores,
 
    // então mostra VERDADEIRO
 
    se cont > 1 entao
 
      escrever "VERDADEIRO"
 
    senao
 
      escrever "FALSO"
 
    fimse
 
       
 
fim
 
</syntaxhighlight>
 
 
 
== 21/09: estruturas de decisão ==
 
 
 
Há situações em que se precisa fazer um conjunto de comparações, como mostrado abaixo:
 
 
 
<syntaxhighlight lang=text>
 
// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
 
// dia, nome do mês, ano.
 
Inicio
 
   inteiro dia, mes, ano
 
  texto nome_mes
 
  
 
   escrever "Dia: "
 
   escrever "Dia: "
Linha 1 563: Linha 1 312:
 
   ler ano
 
   ler ano
  
   se mes = 1 entao
+
   escrever dia, " de ", nome_mes[mes - 1], " de ", ano
    nome_mes <- "Janeiro"
+
fim
  senao
+
</syntaxhighlight>
    se mes = 2 entao
+
 
      nome_mes <- "Fevereiro"
+
Outro problema comum é precisar guardar um conjunto de valores, e depois ordená-los:
    senao
+
 
      se mes = 3 entao
+
<syntaxhighlight lang=text>
        nome_mes <- "Março"
+
inicio
      senao
+
  inteiro v[10]
        se mes = 4 entao
+
  inteiro i
          nome_mes <- "Abril"
+
  inteiro quantidade
        senao
+
 
          se mes = 5 entao
+
  escrever "Quantos valores serão digitados (máximo = 10) ? "
            nome_mes <- "Maio"
+
  ler quantidade
          senao
+
 
            se mes = 6 entao
+
  i <- 0
              nome_mes <- "Junho"
+
  enquanto i < quantidade faz
            senao
+
    escrever "valor ", i, ": "
              se mes = 7 entao
+
    ler v[i]
                nome_mes <- "Julho"
+
    i <- i + 1
              senao
+
  fimenquanto
                se mes = 8 entao
+
 
                  nome_mes <- "Agosto"
+
   // ordena os valores ...
              senao
 
                  se mes = 9 entao
 
                    nome_mes <- "Setembro"
 
                  senao
 
                    se mes = 10 entao
 
                      nome_mes <- "Outubro"
 
                    senao
 
                      se mes = 11 entao
 
                        nome_mes <- "Novembro"
 
                      senao
 
                        nome_mes <- "Dezembro"
 
                      fimSe
 
                    fimSe
 
                  fimSe
 
                fimSe
 
              fimSe
 
            fimSe
 
          fimSe
 
        fimSe
 
      fimSe
 
    fimSe
 
   fimSe
 
  
   escrever dia, " de ", nome_mes, " de ", ano
+
   escrever "valores ordenados: "
 +
  i <- 0
 +
  enquanto i < quantidade
 +
    escrever v[i], " "
 +
    i <- i + 1
 +
  fimenquanto
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
  
Além de ser trabalhoso esse encadeamento de ''Se ... entao ... senao'', o algoritmo resultante fica pouco legível. Quer dizer, ele fica feio e difícil de entender.
+
=== Atividades ===
 +
 
 +
# Escreva um algoritmo que leia 5 números do teclado, e depois mostre-os em ordem inversa à que foi digitada.
 +
# Escreva um algoritmo que leia dois vetores de 5 números, e depois mostre se os vetores são iguais.
 +
# Escreva um algoritmo que leia um palavra do teclado e depois procure-a numa lista de palavras preexistente. Essa lista deve ser implementada usando um vetor.
 +
# Escreva um algoritmo que leia 5 números, e mostre-os em ordem crescente.
 +
# Modifique o algoritmo anterior para mostrá-los em ordem decrescente.
 +
# Em um  jogo de bingo devem-se contar quantos números de uma aposta foram sorteados no bingo. Faça um algoritmo que, dados os números sorteados e uma aposta, conta quantos números apostados foram sorteados.
 +
 
 +
== 12/05: Estruturas de repetição  ==
 +
 
 +
=== Para ''variavel'' de ''inicio'' ate ''fim'' passo ''incremento'' ===
  
Existe uma estrutura de decisão criada justamente para casos como esse, e que resulta em um algoritmo mais limpo e compreensível:
+
Há casos em que se deseja repetir uma parte de um algoritmo certo número de vezes. Para esses casos há uma estrutura mais prática que ''enquanto condição faz'':
  
 
<syntaxhighlight lang=text>
 
<syntaxhighlight lang=text>
// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
+
inicio
// dia, nome do mês, ano.
+
   inteiro i
Inicio
+
   inteiro v[5]
   inteiro dia, mes, ano
 
   texto nome_mes
 
  
   escrever "Dia: "
+
   escrever "Digite 5 números: \n"
   ler dia
+
 
   escrever "Mes: "
+
   para i de 0 ate 4 passo 1
   ler mes
+
    escrever "Numero ", i, ": "
  escrever "Ano: "
+
    ler v[i]
   ler ano
+
  proximo
 +
 
 +
   escrever "\nOs numeros em ordem inversa são:\n\n"
 +
 
 +
   para i de 4 ate 0 passo -1
 +
    escrever "Numero ", i, ": ", v[i], "\n"
 +
   proximo
  
  Escolhe mes
+
fim
    caso 1:
+
</syntaxhighlight>
      nome_mes <- "Janeiro"
 
    caso 2:
 
      nome_mes <- "Fevereiro"
 
    caso 3:
 
      nome_mes <- "Março"
 
    caso 4:
 
      nome_mes <- "Abril"
 
    caso 5:
 
      nome_mes <- "Maio"
 
    caso 6:
 
      nome_mes <- "Junho"
 
    caso 7:
 
      nome_mes <- "Julho"
 
    caso 8:
 
      nome_mes <- "Agosto"
 
    caso 9:
 
      nome_mes <- "Setembro"
 
    caso 10:
 
      nome_mes <- "Outubro"
 
    caso 11:
 
      nome_mes <- "Novembro"
 
    Defeito:
 
      nome_mes <- "Dezembro"
 
  fimEscolhe
 
  
  escrever dia, " de ", nome_mes, " de ", ano
+
=== Atividades ===
fim
 
</syntaxhighlight>
 
 
 
A estrutura de decisão ''escolhe ... caso'' tem uma certa flexibilidade. No exemplo abaixo, mostra-se a possibilidade de testar mais de um valor no mesmo ''caso'':
 
  
<syntaxhighlight lang=text>
+
# Escreva o algoritmo para comparar dois vetores, porém usando a estrutura ''para''.
inicio   
+
# Escreva um algoritmo para procurar se um vetor menor (com menos valores) está contido em um vetor maior. Caso afirmativo, mostre a partir de que posição ele aparece. Assuma vetores de números inteiros.
  caracter sexo   
+
# Modifique o algoritmo acima para mostrar todas as ocorrências do vetor menor dentro do maior.
 +
# Modifique o algoritmo do anterior para remover todas as ocorrências do vetor menor dentro do maior.
 +
# Resolva o problema do bingo usando a estrutura ''para''
  
  escrever "Qual o seu sexo (f/m):"   
+
== 18/05: Revisão ==
  ler sexo   
 
  
  escrever "Olá "   
+
...exercícios
  
  escolhe sexo       
+
Resolver a avaliação de 2010-2:
    caso "m", "M" :            
+
 
      escrever "senhor"       
+
1) '''Contador de moedas:''' faça um programa que leia valores de moedas e some as quantidades de tipos de moedas informadas. Por exemplo, se o usuário digitar 25, 50, 25, 5, 10, 5, o programa deve informar: ''2 moedas de 5 centavos, 1 moeda de 10 centavos, 2 moedas de 25 centavos, 1 moeda de 50 centavos''. São aceitos apenas  valores de moedas de 1, 5, 10, 25 e 50 centavos.  Seu programa deve ler 10 valores de moedas, e então apresentar o resultado.
    caso "f","F" :            
 
      escrever "senhorita"       
 
    defeito :           
 
      escrever "Sexo indefinido"   
 
  fimescolhe   
 
  
  escrever ", benvindo ao portugol"
+
2) '''Calculadora de troco:''' escreva um programa que calcule o troco a ser devolvido a um cliente. O troco deve ser composto de notas de 50, 20, 10, 5, 2 ou 1 real, e também moedas de 1, 5, 10, 25 ou 50 centavos. Dê preferência ao uso de notas e moedas de maior valor. Por exemplo, se o troco a ser devolvido for ''R$ 17,75'', seu programa deve informar ''1 nota de R$ 10, 1 nota de R$ 5, 1 nota de R$ 2, 1 moeda de R$ 0,50, 1 moeda de R$ 0,25''.
fim
 
</syntaxhighlight>
 
  
=== Atividades ===
+
3) '''Modificação de senha:''' Para modificar uma senha deve-se primeiro fornecer a senha
 +
atual, e então a nova senha.  É necessário escrever duas vezes a nova
 +
senha para confirmá-la. Escreva um  algoritmo  que modifique uma senha
 +
predefinida (no seu programa ela pode estar guardada em uma variável), seguindo essa regra de mudança de senha.
  
# Faça um algoritmo que converta um número de 1 a 7 para o respectivo dia da semana (ex: 1 = domingo, 2 = 2a feira, e assim por diante).
+
4) '''Nota de uma prova olímpica:''' um atleta de saltos ornamentais tem seu salto avaliado por 6 juízes. Após descartar a maior e a menor nota, calcula-se a média aritmética das notas dadas. Faça um programa que calcule a nota de um atleta.
# Faça uma calculadora com as quatro operações aritméticas. Sua calculadora deve ler (nesta ordem) o primeiro número, a operação aritmética (que deve ser informada usando o símbolo da respectiva operação: +, -, * ou /), e depois o segundo número. Ao final, seu algoritmo deve mostrar o resultado, ou uma mensagem de erro se a operação não for possível de realizar (ex: divisão por zero).
 
# A previsão do tempo na costa brasileira pode ser feita de forma aproximada usando-se um barômetro e um termômetro. Uma estimativa com boa chance de acerto se baseia na tabela abaixo: <br>[[imagem:Previsao-barometro.png|800px]]<br><br>Faça um algoritmo que forneça uma estimativa da previsão do tempo, usando essa tabela.
 
# Faça um algoritmo que mostre qual o último dia de um determinado mês informado pelo teclado. Caso seja informado o mês 2 (fevereiro), seu algoritmo deve identificar se é [http://pt.wikipedia.org/wiki/Bissexto ano bissexto] (assim o mês tem 29 dias), ou não (mês tem 28 dias). Obs: anos bissextos são dados pelas regras (segundo o [http://pt.wikipedia.org/wiki/Calend%C3%A1rio_Gregoriano calendário Gregoriano]):
 
## De 400 em 400 anos é ano bissexto.
 
## De 100 em 100 anos não é ano bissexto.
 
## De 4 em 4 anos é ano bissexto.<br>''Solução vista em aula: (sem considerar anos bissextos):'' <syntaxhighlight lang=text>
 
inicio
 
      texto mes
 
     
 
      escrever "Mes: "
 
      ler mes
 
     
 
      escolhe mes
 
        caso "janeiro","março", "maio", "julho", "agosto", "outubro", "dezembro":
 
          escrever 31
 
        caso "abril", "junho", "setembro", "novembro":
 
          escrever 30
 
        caso "fevereiro":
 
          escrever 28
 
        defeito:
 
          escrever "mes desconhecido !"
 
      fimEscolhe
 
fim
 
</syntaxhighlight>
 
  
== 27/09: Estruturas de decisão ==
+
... mais algumas questões interessantes:
  
Na aula se fez a resolução de exercícios propostos.
+
5) '''Filtro:''' Um programa possui 10 números inteiros em um vetor. No entanto, para a próxima etapa de processamento apenas os números entre 3 e 8 são válidos. Escreva um algoritmo que remova os números inválidos do vetor. Ao final o vetor deve conter somente os números válidos.
* jogo de par ou ímpar <syntaxhighlight lang=text>
+
 
inicio
+
6) '''Tabuleiro:''' Escreva um programa que mostre o conteúdo de uma matriz 4 x 4 na tela. Essa matriz contém valores inteiros que podem ser positivos ou negativos. Ao mostrar cada valor da matriz, deve-se seguir a seguinte regra:
  inteiro numeroComputador
+
* Valor negativo: deve ser mostrado um espaço em branco
  inteiro numeroUsuario
+
* Valor entre 1 e 8: mostrar o próprio valor
  inteiro soma
+
* Valor acima de 8: mostrar o caractere '''H'''
  inteiro resto
+
 
  texto opcaoUsuario
+
Além disso, a matriz deve ser mostrada com uma grade. Por exemplo, a matriz com conteúdo:
 
+
 
  // sorteia um numero entre 0 e 10 para o computador
+
<math>A = \begin{pmatrix}
  // "aleatorio()" é uma função que gera um número real entre 0 e 1
+
1 & 6 & 10 & -1 \\
  numeroComputador <- aleatorio()*10
+
3 & 5 & -1 & -1 \\
 
+
10 & 4 & -1 & 5 \\
  escrever "Escolha par ou impar: "
+
3 & -1 & 1 & -1 \\
  ler opcaoUsuario
+
\end{pmatrix}</math>
 +
 
 +
... que em Portugol seria declarada assim:
  
  escrever "Escolha seu numero: "
+
<syntaxhighlight lang=text>
  ler numeroUsuario
+
inteiro A <- {{1, 6, 10, -1}, {3, 5, -1, -1},
 
+
              {10, 4, -1, 5}, {3, -1, 1, -1}}
 
 
  escrever "Numero do computador: ", numeroComputador
 
  escrever "\nNumero do usuario: ", numeroUsuario
 
 
 
  // Para fazer a verificacao do vencedor, precisa antes descobrir se
 
  // o resultado é par ou ímpar. O resultado se obtém somando os números do
 
  // computador e do usuário, e é guardado na variável "soma". Já a variável
 
  // "resto" contém 0 se o resultado for par, ou 1 se for ímpar.
 
  soma <- numeroUsuario + numeroComputador
 
  resto <- soma % 2
 
 
 
  // Verifica quem venceu: se o resultado for par e o usuário escolheu "par", ou se
 
  // o resultado for ímpar e o usuário escolheu ímpar, então o usuário venceu. Caso
 
  // contrário o computador é o vencedor.
 
  // Repare como se expressa a condição para descobrir o vencedor: ela usa
 
  // várias condições simples combinadas com operadores lógicos E e OU.
 
  se (resto = 0) E (opcaoUsuario = "par") OU (resto = 1) E (opcaoUsuario = "impar") entao
 
    escrever "\n\nUsuario venceu :-)"
 
  senao
 
    escrever "\n\nComputador venceu :-b"
 
  fimSe
 
 
 
fim
 
 
</syntaxhighlight>
 
</syntaxhighlight>
* leitura de três palavras e teste se ano menos duas pertencem a um conjunto: <syntaxhighlight lang=text>
 
// Neste algoritmo deve-se identificar se ao menos duas palavras
 
// diferentes têm os valores azul, preto ou vermelho.
 
// A variável inteira "cont" conta quantas palavras diferentes
 
// tem esses valores.
 
  
inicio
+
... deveria ser mostrada da seguinte forma:
    texto p1 , p2 , p3
 
    inteiro cont <- 0
 
   
 
    escrever "Palavra 1: "
 
    ler p1
 
    escrever "Palavra 2: "
 
    ler p2
 
    escrever "Palavra 3: "
 
    ler p3
 
   
 
    // ao menos uma palavra é "azul" ?
 
    se p1 = "azul" OU p2 = "azul" OU p3 = "azul" entao
 
      cont <- 1
 
    fimse
 
  
    // ao menos uma palavra é "preto" ?
+
<syntaxhighlight lang=text>
    se p1 = "preto" OU p2 = "preto" OU p3 = "preto" entao
 
      cont <- cont + 1
 
    fimse
 
  
    // ao menos uma palavra é "vermelho" ?
+
..| 1 | 2 | 3 | 4 |
    se p1 = "vermelho" OU p2 = "vermelho" OU p3 = "vermelho" entao
+
-------------------
      cont <- cont + 1
+
A | 1 | 6 | H |  |
    fimse
+
-------------------
 +
B | 3 | 5 |  |  |
 +
-------------------
 +
C | H | 4 |  | 5 |
 +
-------------------
 +
D | 3 |  | 1 |  |
 +
-------------------
 +
</syntaxhighlight>
  
    // se duas ou mais palavras diferentes têm aqueles valores,
+
Quer dizer, as colunas devem ser identificadas por seus números, e as linhas devem ser identificadas pelas letras '''A''', '''B''','''C''', '''D'''.
    // então mostra VERDADEIRO
+
 
    se cont > 1 entao
+
= Linguagem C: Projeto final =
      escrever "VERDADEIRO"
 
    senao
 
      escrever "FALSO"
 
    fimse
 
       
 
fim
 
</syntaxhighlight>
 
* teste de ano bissexto, e seu uso para mostrar quantos dias tem cada mês de um determinado ano. <syntaxhighlight lang=text>
 
inicio
 
      texto mes
 
      inteiro ano
 
     
 
      escrever "Mes: "
 
      ler mes
 
     
 
      escolhe mes
 
        caso "janeiro","março", "maio", "julho", "agosto", "outubro", "dezembro":
 
          escrever 31
 
        caso "abril", "junho", "setembro", "novembro":
 
          escrever 30
 
        caso "fevereiro":
 
            escrever "Ano: "
 
            ler ano
 
           
 
            se ano % 400 = 0 ou ano % 4 = 0 e ano % 100 =/= 0 entao
 
                escrever 29
 
            senao
 
                escrever 28
 
            fimse
 
        defeito:
 
          escrever "mes desconhecido !"
 
      fimEscolhe
 
fim
 
</syntaxhighlight>
 
  
== 28/09: Estruturas de repetição ==
+
== 19/05: Introdução à linguagem C ==
  
Alguns algoritmos vistos anteriormente possuem sequências repetitivas de instruções. Por exemplo, o algoritmo da média de quatro avaliações:
+
* Baseado no [[Guia Básico de C]].
 +
* Para provocar: [http://norvig.com/21-days.html Teach Yourself Programming in Ten Years]
 +
* Apostila adotada: [http://www.ead.cpdee.ufmg.br/cursos/C/c.html Curso de Linguagem C - Engenharia Elétrica -  UFMG]
 +
* Ver [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula15.pdf transparências]
 +
* Alguns textos escritos por mim sobre  tópicos específicos da linguagem  C:
 +
** [[SOP-strings|Strings]]
 +
** [[SOP-funcoes|Funções]]
 +
** [[SOP-stdio|Entrada e saída padrão]]
 +
** [[SOP-arquivos|Arquivos]]
 +
** [[SOP-struct|Estruturas de dados]]
  
<syntaxhighlight lang=text>
+
Obs: durante as aulas usaremos o [http://www.netbeans.org NetBeans], um Ambiente Integrado de Desenvolvimento (''IDE - Integrated Development Environment''). Um IDE é um programa que auxilia o programador. Ele integra um  editor de programas, um gerenciador de projetos, compilador e um depurador (''debugger'').
Inicio
 
  real m1, m2, m3, m4
 
  real media
 
  
  escrever "Avaliação 1:"
+
'''''Para instalar o Netbeans:'''''
  Ler m1
+
# Faça o [http://netbeans.c3sl.ufpr.br/6.8/bundles/netbeans-6.8-ml-cpp-linux.sh download do instalador]. Salve-o em algum subdiretório.
  escrever "Avaliação 2:"
+
# Abrindo um terminal, entre no subdiretório onde está o instalador e execute esse comando: <syntaxhighlight lang=bash>
  Ler m2
+
bash netbeans-6.8-ml-cpp-linux.sh
  escrever "Avaliação 3:"
+
</syntaxhighlight>
  Ler m3
+
#* Caso o instalador falhe porque não encontrou a JVM (Máquina Virtual Java), instale-a com esse comando: <syntaxhighlight lang=bash>
  escrever "Avaliação 4:"
+
sudo apt-get install -y sun-java6-jdk
  Ler m4
+
</syntaxhighlight>
 +
# Aceite as opções de instalação sugeridas pelo instalador.
 +
# Ao final da instalação, o Netbeans 6.8 estará acessível pelo menu ''Aplicativos -> Programação''.
 +
# Sempre que for escrever um novo programa com o Netbeans, crie um novo projeto (ver menu Arquivo->Novo Projeto). Selecione um projeto do tipo "Aplicativo C/C++".
 +
 
 +
=== Compilando o primeiro programa ===
  
  media <- (m1 + m2 + m3 + m4) / 4
+
Obs: para compilar um programa você deve salvá-lo em um arquivo com extensão '''.c'''. Por exemplo, ao editar o exemplo abaixo, você pode gravá-lo no arquivo ''hello.c''. Em seguida você deve compilá-lo usando o seguinte comando:
  
  escrever "Média: ", media
+
<syntaxhighlight lang=bash>
Fim
+
gcc hello.c -o hello
 
</syntaxhighlight>
 
</syntaxhighlight>
  
O algoritmo acima repete quatro vezes a sequência que lê uma nota do teclado. Porém há uma forma de expressar a repetição de sequências de instruções, usando-se o que se chama de ''estrutura de repetição''. A estrutura de repetição '''''enquanto''''' ''condição'' '''''faz''''' repete todas as instruções enquanto a condição for verdadeira. A condição é escrita como uma expressão lógica, da mesma forma que na estrutura de decisão '''''se''''' ''condição'' '''''então ... senão'''''. Veja como fica o algoritmo acima ao ser reescrito para usar essa estrutura de repetição:
+
Para executá-lo, faça assim:
  
<syntaxhighlight lang=text>
+
<syntaxhighlight lang=bash>
Inicio
+
./hello
  constante inteiro NUMEROS <- 4
+
</syntaxhighlight>
  real m
 
  real media <- 0
 
  inteiro contador <- 0
 
  
  enquanto contador < NUMEROS faz 
+
O arquivo que contém o programa em linguagem C se chama ''arquivo de código-fonte''. O arquivo que contém o programa compilado se chama ''arquivo de código-objeto''.
    escrever "Avaliação ", contador, ":"
+
 
    ler m
+
 
    media <- media + m
+
* O clássico ''Hello World!''
    contador <- contador + 1
+
<syntaxhighlight lang=c>
  fimEnquanto
+
#include <stdio.h>
  
  escrever "Média: ", media/NUMEROS
+
int main(int argc, char *argv[])
Fim
+
{
 +
printf("Alô mundo!\n");
 +
}
 
</syntaxhighlight>
 
</syntaxhighlight>
  
Esse algoritmo funciona somente para médias de quatro números. Porém calcular a média de qualquer quantidade de números usa a mesma lógica: ler os números, somá-los e dividir o total pela quantidade de números lidos. Por exemplo, para fazer com que ele calcule a média de 7 números é necessário escrevê-lo assim:
+
* Mostrando mensagens de na tela: [http://manpages.ubuntu.com/manpages/karmic/en/man3/puts.3.html puts] e [http://manpages.ubuntu.com/manpages/karmic/en/man3/printf.3.html printf]: <syntaxhighlight lang=c>
 +
#include <stdio.h>
  
<syntaxhighlight lang=text>
+
int main() {
Inicio
+
   int n;
  constante inteiro NUMEROS <- 7
 
  real m
 
  real media
 
   inteiro contador <- 0
 
  
   enquanto contador < NUMEROS faz 
+
   n = 5;
    escrever "Avaliação ", contador, ":"
+
 
    ler m
+
  puts("Demonstração de puts e printf");
    media <- media + m
+
 
    contador <- contador + 1
+
  printf("Valor de n: %d\n", n);
   fimEnquanto
+
 
 +
  n = n + 1;
 +
 
 +
   printf("Novo valor de n: %d\n", n);
  
   escrever "Média: ", media/NUMEROS
+
   return 0;
Fim
+
}
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
* Lendo dados do teclado: [http://manpages.ubuntu.com/manpages/karmic/en/man3/scanf.3.html scanf]<syntaxhighlight lang=c>
 +
#include <stdio.h>
  
Note que o algoritmo praticamente não foi modificado, pois somente se alterou o valor da constante NUMEROS.
+
int main() {
 +
  int n;
  
Esses dois algoritmos podem também ser visualizados como fluxogramas, como mostrado na figura abaixo:
+
  printf("Digite um número inteiro: ");
  
[[imagem:Ex-repeticao.png]]
+
  scanf("%d", &n);
  
A estrutura de repetição junto com a sequência de instruções a ser repetida é comumente chamada de '''laço''. No fluxograma acima o laço está destacado em vermelho, e no algoritmo em Portugol o laço aparece em:
+
  printf("Valor digitado foi %d\n", n);
  
<syntaxhighlight lang=text>
+
   return 0;
   enquanto contador < NUMEROS faz 
+
}
    escrever "Avaliação ", contador, ":"
 
    ler m
 
    media <- media + m
 
    contador <- contador + 1
 
  fimEnquanto
 
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
** Outras formas de ler do teclado: [http://manpages.ubuntu.com/manpages/karmic/man3/fgets.3.html getchar],  [http://manpages.ubuntu.com/manpages/karmic/en/man3/fgets.3.html fgets]
  
----
+
==== Tipos de dados ====
 +
{| border="1" cellpadding="2"
 +
!Portugol
 +
!C
 +
!Exemplo
 +
|-
 +
|inteiro||int|| int x;
 +
|-
 +
|caracter||char|| char letra;
 +
|-
 +
|real||float ou double|| float pi = 3.1416;
 +
|-
 +
|texto||char * ou vetor de char || char * mensagem = "Hello world";<br>char palavra[16];
 +
|-
 +
|logico||qualquer tipo (valor 0 = Falso, qualquer outro valor = Verdadeiro)||int ok = 1;<br>char teste = 0;
 +
|}
  
Um outro exemplo ainda mais simples é mostrar na tela uma contagem numérica:
 
  
<syntaxhighlight lang=text>
+
==== Comandos/funções e Estruturas de controle ====
inicio
+
{| border="1" cellpadding="2"
    inteiro contador
+
!Portugol
    contador <- 0
+
!C
 +
!Exemplo
 +
|-
 +
|<syntaxhighlight lang=text>inteiro n, m
  
    enquanto contador < 10 faz
+
escrever "Ola, mundo!\n"
        escrever contador , " "
+
escrever "valor de n: ", n, "\n"
        contador <- contador + 1
+
escrever "valor de n: ", n, ", e valor de x: ", x, "\n"
    fimenquanto
+
</syntaxhighlight>||<syntaxhighlight lang=c>int n, m;
fim
 
</syntaxhighlight>
 
  
Ao executar esse algoritmo, tem-se como resultado:
+
printf("Ola, mundo!\n");
 +
printf("valor de n: %d\n", n);
 +
printf("valor de n: %d, e valor de x: %d\n", n, x);</syntaxhighlight>||
 +
|-
 +
|<syntaxhighlight lang=text>inteiro n
  
<syntaxhighlight lang=text>
+
ler n</syntaxhighlight>||<syntaxhighlight lang=c>int n;
0 1 2 3 4 5 6 7 8 9
 
</syntaxhighlight>
 
  
Seu fluxograma está mostrado a seguir:
+
scanf("%d", &n);</syntaxhighlight>||
 
+
|-
[[imagem:Flux-Contagem.png]]
+
|<syntaxhighlight lang=text>se condição então
 
+
//comandos
 
+
fimse</syntaxhighlight> ||<syntaxhighlight lang=c>if (condição) {
Repare no uso de uma variável auxiliar ''contador'' para controlar a quantidade de repetições. Essa é uma técnica comum para
+
//comandos
controlar a estrutura de repetição, e já foi usada no exemplo da média. Ela pode também ser combinada com outras condições, como mostrado abaixo:
+
}</syntaxhighlight>||
 
+
|-
<syntaxhighlight lang=text>
+
|<syntaxhighlight lang=text>se condição então
inicio
+
  //comandos
    inteiro contador <- 1
+
senão
    caracter opcao <- "s"
+
  //comandos
       
+
fimse</syntaxhighlight> ||<syntaxhighlight lang=c>if (condição) {
    enquanto ((opcao = "s") ou (opcao = "S")) e contador < 11 faz
+
//comandos
        escrever contador, "\n"
+
} else {
        contador <- contador + 1
+
//comandos
       
+
}</syntaxhighlight> ||
        escrever "\nContinuar (S/N) ? "
+
|-
        ler opcao
+
|<syntaxhighlight lang=text>escolhe expressão
    fimenquanto
+
  caso valor1: 
   
+
    //comandos
    escrever "Terminou com contador = " , contador
+
  caso valor2:
fim
+
    //comandos
</syntaxhighlight>
+
  defeito:
 
+
    //comandos
Nesse exemplo se usa um laço para mostrar uma contagem de 1 a 10. Porém há a possibilidade de encerrar o algoritmo. Observe a condição para continuidade do laço: ela é verdadeira somente se ''contador < 11'' e se a variável ''opcao'' for igual a "s" ou "S".
+
fimescolhe</syntaxhighlight>||<syntaxhighlight lang=c>switch (expressão) {
 
+
  case valor1:
Finalmente, apesar de ser comum o uso de contadores para controle do laço, pode-se usar qualquer condição para essa finalidade. Então podem existir laços que não usam contadores para controlar a quantidade de repetições, como no próximo exemplo:
+
     //comandos
 
+
  case valor2:
<syntaxhighlight lang=text>
+
     //comandos
inicio
+
  default:
     constante texto segredo <- "secreto"
+
     //comandos
     texto senha
+
}</syntaxhighlight>||
   
+
|-
     ler senha
+
|<syntaxhighlight lang=text>enquanto condição faz
    enquanto senha =/= segredo faz
+
//comandos
        escrever "Voce continua preso ! Digite a senha correta: "
+
fimenquanto</syntaxhighlight>||<syntaxhighlight lang=c>while (condição) {
        ler senha
+
  //comandos
    fimenquanto
+
}</syntaxhighlight>||
    escrever "Voce foi liberado ..."
+
|-
fim
+
|<syntaxhighlight lang=text>para variavel de inicio ate fim passo incremento
</syntaxhighlight>
+
//comandos
 
+
proximo</syntaxhighlight>||<syntaxhighlight lang=c>for (variavel=inicio; variavel <= fim; variavel++) {
O fluxograma desse algoritmo segue abaixo:
+
//comandos
 
+
}</syntaxhighlight>||
[[imagem:Flux-Senha.png]]
+
|}
  
 
=== Atividades ===
 
=== Atividades ===
  
# Escreva um  algoritmo que mostre a tabuada de um número fornecido pelo teclado. Esse número deve estar entre 1 e 10.
+
* Traduza para C os seguintes algoritmos Portugol:  
# Aproveite o algoritmo anterior para mostrar a tabuada de todos os números entre 1 e 10.
+
*# <syntaxhighlight lang=text>
# Modifique o exemplo da média para que a quantidade de números a serem lidos seja previamente informada pelo teclado.
+
Inicio
# Escreva  algoritmos que mostrem as seguintes sequências de números:
+
  inteiro x, y
#* Números pares menores que 100
+
#* Números ímpares menores que 100
+
  Escrever "Digite um numero: ",
#* Números pares de 100 até 4 (contagem decrescente)
+
  Ler x
#* Múltiplos de 3 menores que 100
+
  Escrever "Digite outro numero: ",
#* Múltiplos de 4 e múltiplos de 5 menores que 100
+
  Ler y
#* Múltiplos comuns menores que 1000 de dois números fornecidos pelo teclado (ex: se forem  fornecidos 6 e 8, deve aparecer 24, 48, 72, ...)
+
  Escrever "Soma = ", x+y
# Modifique novamente o exemplo da média para que ela funcione para um quantidade de números desconhecida de antemão. Quer dizer, o algoritmo deve ler os números para calcular a média, mas não sabe quantos números existem (e isto não pode ser informado pelo teclado).
+
Fim
# Modifique o exemplo da senha para que o usuário tenha somente três tentativas permitidas para digitar a senha correta. Caso ao final as três senhas estejam erradas, o algoritmo deve informar que a conta foi bloqueada.
+
</syntaxhighlight>
# Escreva um algoritmo que leia até 10 números do teclado, e informe ao final qual o maior e o menor deles.
+
*# <syntaxhighlight lang=text>
# Escreva um algoritmo que teste se um número informado pelo teclado é primo.
+
Inicio
# Aproveite o algoritmo do ítem anterior para mostrar todos os números primos menores que 1000.
+
  inteiro x, y, z
# Usando o algoritmo de teste de número primo, faça um novo algoritmo que fatore um número. Ex: a fatoração do número 15 resulta em  3 e 5 (3 X 5), e a fatoração de 54 resulta em 2, 3, 3 e 3 (2 X 3 X 3 X 3).
+
# Usando estrutura de repetição, escreva um algoritmo que converta um número de decimal para binário. Assuma um número binário com 16 bits.
+
  Escrever "Digite um numero: ",
# Escreva um algoritmo para calcular o MMC (Mínimo Múltiplo Comum) de dois números. O MMC de dois números é um número que pode ser dividido por qualquer um deles sem deixar resto. Ex:
+
  Ler x
#* MMC de 8 e 6 = 24
+
  Escrever "Digite outro numero: ",
#* MMC de 3 e 5 = 15
+
  Ler y
#* MMC de 3 e 27 = 27
+
  z <- x + y
# Escreva um  algoritmo para calcular o MDC (Máximo Divisor Comum) de dois números. O MDC de dois números é um número capaz de dividi-los sem deixar resto. Ex:
+
  Escrever "Soma = ", z
#* MDC de 24 e 16 = 8
+
Fim
#* MDC de 5 e 3 = 1
+
</syntaxhighlight>
#* MDC de 60 e 150 = 30
+
*# <syntaxhighlight lang=text>
# Faça um algoritmo para descobrir os dois maiores números dentre 6 números lidos do teclado.
+
Inicio
# Faça um algoritmo para converter um número entre 0 e 16777215 para sua representação em hexadecimal.
+
  Inteiro n1, n2, n3, r
# Escreva o algoritmo para uma calculadora aritmética (operações +, -, * e /) que lê continuamente do teclado números e operadores. A calculadora deve ler continuamente uma sequência do tipo ''número operador_aritmético'', até que o operador informado seja ''='', quando então o resultado da conta deve ser mostrado na tela. Ex:
+
#* 1 + 2 * 5 / 3 = 5
+
  Escrever "Primeiro numero: "
#* 2 * 5 - 1 / 3 = 3
+
  ler n1
 
+
  Escrever "Segundo numero: "
== 04/10: Estruturas de repetição ==
+
  ler n2
 
+
  Escrever "Terceiro numero: "
Resolução de exercícios
+
  ler n3
 
+
== 05/10: Estruturas de repetição ==
+
  r <- (n1 + n2 + n3) /3
 
+
   
Resolução de exercícios
+
  Escrever "Media=", r
 
+
Fim
== 18/10: Estruturas de repetição ==
+
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
=== Para ''variavel'' de ''inicio'' ate ''fim'' passo ''incremento'' ===
+
Inicio
 
+
   inteiro anos, dias
Há casos em que se deseja repetir uma parte de um algoritmo certo número de vezes. Para esses casos há uma estrutura mais prática que ''enquanto condição faz'':
+
 
+
   Escrever "Quantos anos se passaram ? "
<syntaxhighlight lang=text>
+
   Ler anos
inicio
+
   dias <- anos * 365
   inteiro i
+
   Escrever "... então se passaram ", dias, " dias"
  inteiro v[5]
+
Fim
 
 
   escrever "Digite 5 números: \n"
 
 
 
   para i de 0 ate 4 passo 1
 
    escrever "Numero ", i, ": "
 
    ler v[i]
 
   proximo
 
 
 
   escrever "\nOs numeros em ordem inversa são:\n\n"
 
 
 
  para i de 4 ate 0 passo -1
 
    escrever "Numero ", i, ": ", v[i], "\n"
 
  proximo
 
 
 
fim
 
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
=== Atividades ===
+
Inicio
 
+
  constante inteiro diasPorAno <- 365
# Escreva o algoritmo para comparar dois vetores, porém usando a estrutura ''para''.
+
  inteiro anos, dias
# Escreva um algoritmo para procurar se um vetor menor (com menos valores) está contido em um vetor maior. Caso afirmativo, mostre a partir de que posição ele aparece. Assuma vetores de números inteiros.  
+
# Modifique o algoritmo acima para mostrar todas as ocorrências do vetor menor dentro do maior.
+
  Escrever "Quantos anos se passaram ? "
# Modifique o algoritmo do anterior para remover todas as ocorrências do vetor menor dentro do maior.
+
  Ler anos
# Resolva o problema do bingo usando a estrutura ''para''
+
  dias <- anos * diasPorAno
 
+
  Escrever "... então se passaram ", dias, " dias"
== 19/10: Estruturas de repetição ==
+
Fim
 
+
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
=== Variáveis multidimensionais ===
+
inicio
 
+
  real massa
Em matemática existem matrizes e vetores, que são variáveis multidimensionais. Por exemplo, uma matriz 3 x 3 (3 linhas e 3 colunas) pode ser:
+
  real forca
 
+
  real aceleracao
<math>A = \begin{pmatrix}
+
  constante real g <- 9.8
1 & 6 \\
+
  constante real ac <- 0.01
3 & 5
+
 
\end{pmatrix}</math>
+
  escrever "força: "
 
+
  ler forca
Vetores são matrizes unidimensionais, portanto possuem somente uma linha ou uma coluna:
+
 
 
+
  escrever "massa: "
<math>v = \begin{pmatrix}
+
  ler massa
1 & 6 & 2 & 18 & 5
+
 
\end{pmatrix}</math>
+
  aceleracao <- forca/massa - ac*massa*g
 
+
 
Cada elemento em uma matriz ou vetor é identificado pela sua posição. Por exemplo, na posição ''1, 2'' da matriz A acima está o valor 6, e na posição ''4'' do vetor ''v'' está o valor 18. Assim, a matriz A do exemplo pode ser entendida da seguinte forma:
+
  escrever "aceleracao=", aceleracao, " m/s2"
 
+
 
<math>A = \begin{pmatrix}
+
fim
A_{11} & A_{12} \\
+
</syntaxhighlight>
A_{21} & A_{22}
+
*# <syntaxhighlight lang=text>
\end{pmatrix}</math>
 
 
 
... e o vetor ''v'' do exemplo:
 
 
 
<math>v = \begin{pmatrix}
 
v_{1} & v_{2} & v_{3} & v_{4} & v_{5}
 
\end{pmatrix}</math>
 
 
 
Nas linguagens de programação em geral existe um conceito muito parecido, chamado de variáveis multidimensionais ou simplesmente ''matrizes'' (''arrays'' em inglẽs). Para exemplificar, no Portugol se poderiam definir a matriz ''A'' e o vetor ''v'':
 
 
 
<syntaxhighlight lang=text>
 
 
inicio
 
inicio
   inteiro A[2][2] <- {{1, 6}, {3, 5}}
+
   real a, b, c
   inteiro M[3][2]
+
   real delta
   inteiro v[5] <- {1, 6, 2, 18, 5}
+
   real x1, x2
 
+
 
   escrever "Valor de A[0][0]: ", A[0][0], "\n"
+
   escrever "Forneça os coeficientes da equação de 2o grau (formato: ax^2 + bx + c):\n"
   escrever "Valor de A[0][1]: ", A[0][1], "\n"
+
   escrever "a="
   escrever "Valor de A[1][0]: ", A[1][0], "\n"
+
  ler a
   escrever "Valor de A[1][1]: ", A[1][1], "\n"
+
   escrever "b="
 
+
  ler b
   escrever "Valor de v[0]: ", v[0], "\n"
+
   escrever "c="
   escrever "Valor de v[1]: ", v[1], "\n"
+
  ler c
   escrever "Valor de v[2]: ", v[2], "\n"
+
    
   escrever "Valor de v[3]: ", v[3], "\n"
+
   delta <- b^2 - 4*a*c
   escrever "Valor de v[4]: ", v[4], "\n"
+
   x1 <- (-b + delta^0.5)/(2*a)
 
+
  x2 <- (-b - delta^0.5)/(2*a)
 +
 
 +
   escrever "1a raiz=", x1
 +
   escrever "\n2a raiz=", x2 
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
A declaração da matriz se faz como uma variável comum, porém indicando-se suas dimensões:
+
inicio
 
+
      real x1, x2, x3, x4, x5
<syntaxhighlight lang=text>
+
      real media
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
+
      real desvioPadrao
  inteiro M[3][2];
+
      constante inteiro N <- 5
 +
     
 +
      // Ler os cinco valores pelo teclado
 +
      ler x1
 +
      ler x2
 +
      ler x3
 +
      ler x4
 +
      ler x5
 +
     
 +
      // Calcular a media
 +
      media <- (x1 + x2 + x3 + x4 + x5) / N
 +
     
 +
      // Calcular o desvio padrao
 +
      desvioPadrao <- (x1 - media)^2 + (x2 - media)^2 + (x3 - media)^2 + (x4 - media)^2 + (x5 - media)^2
 +
      desvioPadrao <- (desvioPadrao / (N-1))^0.5
 +
     
 +
      escrever "Media=", media
 +
      escrever "\nDesvio padrao=", desvioPadrao
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
Veja que a matriz ''M'' foi declarada sem valores iniciais, porém a matriz ''A'' foi inicializada na declaração.
+
inicio
 
+
      real renda
O acesso aos elementos da matriz se faz usando-se ''índices'', que funcionam como coordenadas dos elementos que se quer acessar. Os índices iniciam em ''0'':
+
      real irpf
 
+
     
<syntaxhighlight lang=text>
+
      escrever "Informe sua renda: "
  # índice 0,0
+
      ler renda
  escrever "Valor de A[0][0]: ", A[0][0], "\n"
+
     
 +
      se renda < 1372.81 entao
 +
        irpf <- 0
 +
      senao
 +
        se renda < 2743.25 entao
 +
          irpf <- (renda - 205.92)*0.15
 +
        senao
 +
          irpf <- (renda - 548.82)*0.275
 +
        fimSe
 +
      fimSe
 +
     
 +
      escrever "Imposto devido: ", irpf 
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
*# <syntaxhighlight lang=text>
 +
inicio
 +
    inteiro dividendo
 +
    inteiro divisor
 +
    inteiro resto , quociente
  
O exemplo acima pode ser reescrito usando-se estruturas de repetição:
+
    escrever "Dividendo: "
 
+
    ler dividendo
<syntaxhighlight lang=text>
+
    escrever "Divisor: "
 +
    ler divisor
 +
 
 +
    se divisor = 0 entao
 +
        escrever "Nao pode dividir: divisor = 0 !!!"
 +
    senao
 +
        resto <- dividendo % divisor
 +
        se resto = 0 entao
 +
            quociente <- dividendo / divisor
 +
            escrever dividendo , " / " , divisor , " = " , quociente
 +
        senao
 +
            escrever "Nao pode fazer divisao inteira"
 +
            escrever " (resto = " , resto , ")"
 +
        fimse
 +
    fimse
 +
fim
 +
</syntaxhighlight>
 +
*# <syntaxhighlight lang=text>
 
inicio
 
inicio
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
+
    inteiro termometro , barometro , tempo
  inteiro v[5] <- {1, 6, 2, 18, 5}
 
  inteiro i, j
 
  
  enquanto i < 2 faz
+
     escrever "Qual a condição do termômetro: \n"
    j <- 0
+
    escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
     enquanto j < 2 faz
+
     ler termometro
      escrever "Valor do elemento de A na posição ", i, ", ", j, ": ", A[i][j], "\n"
 
      j <- j + 1
 
     fimenquanto
 
    i <- i + 1
 
  fimenquanto
 
  
  i <- 0
+
     escrever "Informe a condição do barômetro: \n"
  enquanto i < 5 faz
+
    escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
     escrever "valor de v na posição ", i, ": ", v[i], "\n"
+
     ler barometro
     i <- i + 1
 
  fimenquanto
 
  
fim
+
    tempo <- termometro*10 + barometro
</syntaxhighlight>
 
  
Como se vê, matrizes e vetores casam perfeitamente com estruturas de repetição, e são úteis para a resolução de inúmeros problemas. Por exemplo, o problema de mostrar o nome do mês a partir do número do mês, feito anteriormente com ''escolhe ... caso'' pode ser feito assim com um vetor:
+
    escolhe tempo
 
+
        caso 11:
<syntaxhighlight lang=text>
+
            escrever "Chuvas abundantes e ventos de sul a sudoeste fortes"
inicio
+
        caso 12:
  texto nome_mes[12] <- {"Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho",
+
            escrever "Chuva Provavel , ventos de sul a sudeste"
                        "Julho", "Agosto", "Setembro", "Outubro", "Novembro",
+
        caso 13:
                        "Dezembro"}
+
            escrever "Tempos Bons , ventos de sul a sudeste"
  inteiro dia, mes, ano
+
        caso 21:
 
+
            escrever "Frente quente com chuva provavel"
  escrever "Dia: "
+
        caso 22:
  ler dia
+
            escrever "Tempo Incerto , ventos variaveis"
  escrever "Mes: "
+
        caso 23:
  ler mes
+
            escrever "Tempos Bons , ventos do leste frescos"
  escrever "Ano: "
+
        caso 31:
  ler ano
+
            escrever " Tempo instavel , aproximaçao de frente"
 
+
        caso 32:
  escrever dia, " de ", nome_mes[mes - 1], " de ", ano
+
            escrever "Tempo Mudando para bom , ventos de leste"
 +
        caso 33:
 +
            escrever "Tempos Bons , ventos quentes e secos"
 +
        defeito:
 +
            escrever "Utilize somente os algorismos de 1 a 3 para indicar as condiçoes do equipamentos"
 +
    fimescolhe
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
Outro problema comum é precisar guardar um conjunto de valores, e depois ordená-los:
+
inicio
 
+
    // coeficientes do polinomio  [ax^2 + bx + c = 0 ]
<syntaxhighlight lang=text>
+
    real a , b , c
inicio
+
   
  inteiro v[10]
+
    escrever "Forneça os coeficientes da equação de 2o grau:\n"
  inteiro i
+
    escrever "a="
  inteiro quantidade
+
    ler a
 
+
    escrever "b="
  escrever "Quantos valores serão digitados (máximo = 10) ? "
+
    ler b
  ler quantidade
+
     escrever "c="
 
+
     ler c
  i <- 0
+
   
  enquanto i < quantidade faz
+
    // equação do tipo [ bx + c = 0 ]
     escrever "valor ", i, ": "
+
     se a = 0 entao
     ler v[i]
+
        escrever " não é uma equação de 2o grau !!!"
     i <- i + 1
+
    senao
  fimenquanto
+
        // calcular o delta => interior da raiz
 
+
        real delta
  // ordena os valores ...
+
       
 
+
        delta <- b ^ 2 - 4 * a * c
  escrever "valores ordenados: "
+
       
  i <- 0
+
        // não existem raizes  reais de números negativos
  enquanto i < quantidade
+
        se delta < 0 entao
    escrever v[i], " "
+
            escrever " não tem raizes reais"
    i <- i + 1
+
        senao
  fimenquanto
+
            // -----------  raiz dupla  ----------------
fim
+
            se delta = 0 entao
</syntaxhighlight>
+
                real x1
 
+
                x1 <- -b / 2 * a
=== Atividades ===
+
                escrever "\nraiz dupla : " , x1
 
+
            senao
# Escreva um algoritmo que leia 5 números do teclado, e depois mostre-os em ordem inversa à que foi digitada.
+
                // - ---------- duas raizes ---------------
# Escreva um algoritmo que leia dois vetores de 5 números, e depois mostre se os vetores são iguais. <br><br>''Solução vista em aula:''<syntaxhighlight lang=text>
+
                real x1 , x2
 +
                x1 <- ( -b + raiz ( delta ) ) / 2 * a
 +
                x2 <- ( -b - raiz ( delta ) ) / 2 * a
 +
                escrever "\nraiz x1 : " , x1
 +
                escrever "\nraiz x2 : " , x2
 +
            fimse//raiz dupla
 +
        fimse// delta >0
 +
    fimse// a <> 0
 +
fim
 +
</syntaxhighlight>
 +
*# <syntaxhighlight lang=text>
 
inicio
 
inicio
  inteiro v1[5]
+
    inteiro num, bit
  inteiro v2[5]
+
    inteiro base
  inteiro n
+
   
 +
    escrever "Este algoritmo converte um número inteiro para sua representação binária.\n"
 +
    Escrever "O número deve estar entre 0 e 131071, e a representação binária\n"
 +
    escrever "terá 17 bits\n\n"
 +
    escrever "Digite o numero a ser convertido para binário: "
 +
    ler num
  
  // Le os valores do primeiro vetor
+
     base <- 65536
  para n de 0 ate 4 passo 1
+
    enquanto base > 0 faz
    escrever "v1[", n, "]: "
+
        bit <- ( num / base ) % 2
    ler v1[n]
+
        escrever bit, " "  
  proximo
+
        base <- base / 2
 
+
     fimenquanto
  // Le os valores do segundo vetor
 
  para n de 0 ate 4 passo 1
 
    escrever "v2[", n, "]: "
 
     ler v1[n]
 
  proximo
 
 
 
  // compara os dois vetores
 
  n <- 0
 
  enquanto n < 5 E v1[n] = v2[n] faz
 
    n <- n + 1
 
  fimenquanto
 
 
 
  // mostra o resultado da comparação
 
  se n = 5 entao
 
    escrever "Vetores iguais"
 
  senao
 
     escrever "Vetores diferem na posicao ", n
 
  fimse
 
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
# Escreva um algoritmo que leia um palavra do teclado e depois procure-a numa lista de palavras preexistente. Essa lista deve ser implementada usando um vetor.
+
inicio
# Escreva um algoritmo que leia 5 números, e mostre-os em ordem crescente.
+
    inteiro n , fat
# Modifique o algoritmo anterior para mostrá-los em ordem decrescente.
+
    ler n
# Em um  jogo de bingo devem-se contar quantos números de uma aposta foram sorteados no bingo. Faça um algoritmo que, dados os números sorteados e uma aposta, conta quantos números apostados foram sorteados.
+
    fat <- 1
 
+
    enquanto n > 1 faz
== 25/10: 2a avaliação ==
+
      fat <- fat * n
 
+
      n <- n - 1       
Avaliação em aula, no laboratório de Redes. Resolver usando o Portugol. Compactar os arquivos com os algoritmos e entregar para o professor.
+
    fimenquanto
 
+
   
Obs: para compactar:
+
    escrever "Fatorial=" , fat
 
+
fim
<syntaxhighlight lang=bash>
 
tar czf prova2-seu_nome.tgz *.alg
 
 
</syntaxhighlight>
 
</syntaxhighlight>
 
+
*# <syntaxhighlight lang=text>
Você precisará [[SOP-2010-2-sobral#Guia_r.C3.A1pido_de_instala.C3.A7.C3.A3o_e_utiliza.C3.A7.C3.A3o_do_Portugol|instalar o Portugol]] em seu computador.
+
inicio
 
+
    inteiro valor , menor , contador
1) '''Contador de moedas:''' faça um programa que leia valores de moedas e some as quantidades de tipos de moedas informadas. Por exemplo, se o usuário digitar 25, 50, 25, 5, 10, 5, o programa deve informar: ''2 moedas de 5 centavos, 1 moeda de 10 centavos, 2 moedas de 25 centavos, 1 moeda de 50 centavos''. São aceitos apenas  valores de moedas de 1, 5, 10, 25 e 50 centavos.  Seu programa deve ler 10 valores de moedas, e então apresentar o resultado.
+
    ler menor
 
+
     contador <- 9
2) '''Calculadora de troco:''' escreva um programa que calcule o troco a ser devolvido a um cliente. O troco deve ser composto de notas de 50, 20, 10, 5, 2 ou 1 real, e também moedas de 1, 5, 10, 25 ou 50 centavos. Dê preferência ao uso de notas e moedas de maior valor. Por exemplo, se o troco a ser devolvido for ''R$ 17,75'', seu programa deve informar ''1 nota de R$ 10, 1 nota de R$ 5, 1 nota de R$ 2, 1 moeda de R$ 0,50, 1 moeda de R$ 0,25''.<syntaxhighlight lang=text>Inicio
+
     enquanto contador > 0 faz
  inteiro notas[6] <- {50,20,10,5,2,1}
+
        ler valor
  inteiro moedas[5] <- {50,25,10,5,1}
+
        se valor < menor entao
  inteiro reais, centavos
+
            menor <- valor
  inteiro i, aux
+
        fimse
 
+
        contador <- contador - 1
  escrever "Reais: "
+
     fimenquanto
  ler reais
+
     escrever "menor valor = ", menor
  escrever "Centavos: "
 
  ler centavos
 
 
 
  para i de 0 ate 5 passo 1
 
     aux <- reais / notas[i]
 
     se aux > 0 entao
 
      escrever aux, " notas de R$ ", notas[i], "\n"
 
      reais <- reais % notas[i]
 
    fimse
 
  proximo
 
 
 
  para i de 0 ate 4 passo 1
 
     aux <- centavos / moedas[i]
 
     se aux > 0 entao
 
      escrever aux, " moedas de R$ 0,", moedas[i], "\n"
 
      centavos <- centavos % moedas[i]
 
    fimse
 
  proximo
 
 
 
 
fim
 
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
*# <syntaxhighlight lang=text>
 +
inicio
 +
    inteiro numero , n , resto <- 1
 +
    escrever "Forneçaa um número: "
 +
    ler numero
 +
    n <- 2
 +
    enquanto n <= numero / 2 e resto =/= 0 faz
 +
        resto <- numero % n
 +
        escrever "n=" , n , ": resto=" , resto , "\n"
 +
        n <- n + 1
 +
    fimenquanto
 +
   
 +
    se resto = 0 entao
 +
        escrever "nao primo: divisivel por " , n - 1
 +
    senao
 +
        escrever "primo"
 +
    fimse
 +
fim
 +
</syntaxhighlight>
 +
*# <syntaxhighlight lang=text>
 +
// Le 5 valores e identifica os dois maiores.
 +
inicio
 +
    inteiro valor , maior1 , maior2 , contador
 +
   
 +
    ler valor
 +
    maior1 <- valor
 +
   
 +
    ler valor
 +
    se valor > maior1 entao
 +
      maior2 <- maior1
 +
      maior1 <- valor
 +
    senao
 +
      maior2 <- valor
 +
    fimse
 +
 +
    contador <- 3
 +
    enquanto contador > 0 faz
 +
        ler valor
 +
        se valor > maior1 entao
 +
            maior2 <- maior1
 +
            maior1 <- valor
 +
        senao
 +
            se valor > maior2 entao
 +
                maior2 <- valor
 +
            fimse
 +
        fimse
 +
        contador <- contador - 1
 +
    fimenquanto
 +
 +
    escrever "maior valor = " , maior1
 +
    escrever "\nsegundo maior valor = " , maior2
 +
fim
 +
</syntaxhighlight>
 +
 +
== 25/05: Avaliação ==
  
3) '''Modificação de senha:''' Para modificar uma senha deve-se primeiro fornecer a senha
+
No laboratório ...
atual, e então a nova senha. É necessário escrever duas vezes a nova
 
senha para confirmá-la. Escreva um  algoritmo  que modifique uma senha
 
predefinida (no seu programa ela pode estar guardada em uma variável), seguindo essa regra de mudança de senha.
 
 
 
4) '''Nota de uma prova olímpica:''' um atleta de saltos ornamentais tem seu salto avaliado por 6 juízes. Após descartar a maior e a menor nota, calcula-se a média aritmética das notas dadas. Faça um programa que calcule a nota de um  atleta.
 
 
 
=== Conceitos da 2a avaliação ===
 
 
 
 
 
''Legenda:''
 
*'''T:''' Atendeu a questão totalmente
 
*'''P:''' Atendeu parcialmente
 
*'''N:''' Não atendeu
 
*''Sufixo +:'' apresentou uma boa solução
 
*''Sufixo -:'' apresentou uma solução que não escolheu estruturas de controle mais adequadas
 
 
 
 
 
{| border="1" cellpadding="2"
 
!Aluno
 
!Questão 1
 
!Questão 2
 
!Questão 3
 
!Questão 4
 
!Conceito
 
|-
 
|Fernando || N || N || T- ''(testes redundantes da senha atual)''|| T || C
 
|-
 
|Gabriel || || || || || D (ausente)
 
|-
 
|Levi || || || || || D (ausente)
 
|-
 
|Lindalvo || T ''(faltou testar se a moeda é válida)'' || N || T || T || A
 
|-
 
|Marcos || T- ''(faltou testar se a moeda é válida, algoritmo muito extenso ...)''|| N || T || N || C
 
|-
 
|Marine || T+ || T || T || N || A
 
|-
 
|Mario Allan || T || N || T ''(algoritmo nunca termina !)''|| N || C
 
|-
 
|Mário André || T ''(faltou testar se a moeda é válida)'' || T || T+ || N || A
 
|-
 
|Mário Sérgio || N || N || N || N || D
 
|-
 
|Maykon || T- ''(erro no  teste de moeda válida)''|| N || T || N ''(não identificou maior e menor notas)''|| C
 
|-
 
|Nathany || N || N || T || T- ''(não identifica a menor nota)''|| C
 
|-
 
|Pamella || || || || || D (ausente)
 
|-
 
|Paulo Igor || T- ''(não mostra moedas de 1 centavo, lê 11 moedas)''|| N || N || N || D
 
|-
 
|Ricardo || N || N || P ''(não confere a senha atual)''|| N || D
 
|-
 
|Roberto || || || || || D (ausente)
 
|-
 
|Thiago Martins || || || || || D (ausente)
 
|-
 
|Thiago Ramos || N || N || N || N || D
 
|-
 
|Vinicius Kachniacz || N || N || N || N || D
 
|-
 
|Vinicius Rodrigues || N || N || N ''(não confere as senhas)'' || N ''(não exclui maior e menor notas)''|| D
 
|-
 
|}
 
 
 
= Linguagem C: Projeto final =
 
  
== 26/10:  Introdução à linguagem C ==
+
* [http://tele.sj.ifsc.edu.br/~msobral/prog/prova.html Prova]
  
* Baseado no [[Guia Básico de C]].
+
=== Prova resolvida ===
* Para provocar: [http://norvig.com/21-days.html Teach Yourself Programming in Ten Years]
+
 
* Apostila adotada: [http://www.ead.cpdee.ufmg.br/cursos/C/c.html Curso de Linguagem C - Engenharia Elétrica -  UFMG]
+
1) Um programa em uma estação meteorológica converte os valores lidos de um higrômetro para um indicativo de umidade do ar. A tabela de conversão segue abaixo:  
* Ver [http://www.sj.ifsc.edu.br/~msobral/SOP/slides/aula15.pdf transparências]
 
* Alguns textos escritos por mim sobre  tópicos específicos da linguagem  C:
 
** [[SOP-strings|Strings]]
 
** [[SOP-funcoes|Funções]]
 
** [[SOP-stdio|Entrada e saída padrão]]
 
** [[SOP-arquivos|Arquivos]]
 
** [[SOP-struct|Estruturas de dados]]
 
  
Obs: durante as aulas usaremos o [http://www.netbeans.org NetBeans], um Ambiente Integrado de Desenvolvimento (''IDE - Integrated Development Environment''). Um IDE é um programa que auxilia o programador. Ele integra um editor de programas, um gerenciador de projetos, compilador e um depurador (''debugger'').  
+
{| border="1" cellpadding="2"
 +
!Valor do higrômetro
 +
!Indicativo de umidade
 +
|-
 +
|0 a 0.3 || muito baixa
 +
|-
 +
|0.3 a 0.5 || baixa
 +
|-  
 +
|0.5 a 0.8 || normal
 +
  |-
 +
|0.8 a 1 || alta
 +
|-
 +
|1 || precipitação
 +
|}
  
'''''Para instalar o Netbeans:'''''
+
Escreva esse programa, que deve simular a leitura do higrômetro pelo teclado e mostrar o indicativo de umidade na tela.  
# Faça o [http://netbeans.c3sl.ufpr.br/6.8/bundles/netbeans-6.8-ml-cpp-linux.sh download do instalador]. Salve-o em algum subdiretório.
 
# Abrindo um terminal, entre no subdiretório onde está o instalador e execute esse comando: <syntaxhighlight lang=bash>
 
bash netbeans-6.8-ml-cpp-linux.sh
 
</syntaxhighlight>
 
#* Caso o instalador falhe porque não encontrou a JVM (Máquina Virtual Java), instale-a com esse comando: <syntaxhighlight lang=bash>
 
sudo apt-get install -y sun-java6-jdk
 
</syntaxhighlight>
 
# Aceite as opções de instalação sugeridas pelo instalador.
 
# Ao final da instalação, o Netbeans 6.8 estará acessível pelo menu ''Aplicativos -> Programação''.
 
# Sempre que for escrever um novo programa com o Netbeans, crie um novo projeto (ver menu Arquivo->Novo Projeto). Selecione um projeto do tipo "Aplicativo C/C++".
 
  
=== Compilando o primeiro programa ===
+
<syntaxhighlight lang=text>
* O clássico ''Hello World!''
+
inicio
<syntaxhighlight lang=c n>
+
    real h
#include <stdio.h>
+
   
 +
    escrever " digite um valor de 0 a 1:  " ,
 +
    ler h
 +
   
 +
    se h = 1 entao
 +
        escrever "chuva" ,
 +
    senao
 +
        se h <= 0.3 entao
 +
            escrever "umidade muito baixa" ,
 +
        senao
 +
            se h <= 0.5 entao
 +
                escrever "umidade baixa" ,
 +
            senao
 +
                se h <= 0.8 entao
 +
                    escrever "umidade normal"
 +
                senao
 +
                    se h < 1 entao
 +
                        escrever "umidade alta"
 +
                       
 +
                    fimse
 +
                fimse
 +
            fimse
 +
        fimse
 +
    fimse
 +
fim
 +
</syntaxhighlight>
  
int main(int argc, char *argv[])
+
2) Um programa possui o seguinte vetor de inteiros guardado em memória:
{
 
printf("Alô mundo!\n");
 
}
 
</syntaxhighlight>
 
  
* Mostrando mensagens de na tela: [http://manpages.ubuntu.com/manpages/karmic/en/man3/puts.3.html puts] e [http://manpages.ubuntu.com/manpages/karmic/en/man3/printf.3.html printf]: <syntaxhighlight lang=c n>
+
{10,12,7,4,5,9,2,3,1,3,4,2,2,9}
#include <stdio.h>
 
  
int main() {
+
Esse programa deve ler uma sequência de 4 números pelo teclado, e procurá-la dentro do vetor que está em memória. Se a sequência for encontrada, a posição em que ela inicia dentro do vetor deve ser mostrada.
  int n;
 
  
   n = 5;
+
<syntaxhighlight lang=text>
 +
inicio
 +
   inteiro numeros[14] <- {10,12,7,4,5,9,2,3,1,3,4,2,2,9}
 +
  inteiro v[4]
 +
  inteiro i, j
 
    
 
    
   puts("Demonstração de puts e printf");
+
   Escrever "Digite 4 numeros inteiros:"
 
+
  para i de 0 ate 3 passo 1
   printf("Valor de n: %d\n", n);
+
    ler v[i]
 
+
  proximo
  n = n + 1;
+
 
 
+
   para j de 0 ate 10 passo 1
  printf("Novo valor de n: %d\n", n);
+
    i <- 0
 
+
    enquanto i < 4 E v[i] = numeros[j+i] faz
   return 0;
+
      i <- i + 1
}
+
    fimenquanto
 +
    se i = 4 entao
 +
      escrever "Encontrou na posicao ", j
 +
      fim
 +
    fimse
 +
  proximo
 +
   escrever "Nao encontrou em nenhuma posicao"
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
* Lendo dados do teclado: [http://manpages.ubuntu.com/manpages/karmic/en/man3/scanf.3.html scanf]<syntaxhighlight lang=c n>
 
#include <stdio.h>
 
 
int main() {
 
  int n;
 
  
  printf("Digite um número inteiro: ");
+
3) Um número primo é um número divisível somente por 1 ou por si mesmo. Quer dizer, se tentar dividi-lo por qualquer valor maior que 1 e menor que esse número, o resto será diferente de zero. Exemplos de números primos são 2, 3, 5, 7 e 11. Escreva um programa que leia um número pelo teclado e informe se ele é primo.
  
  scanf("%d", &n);
+
<syntaxhighlight lang=text>
 +
inicio
 +
    inteiro numero , n
  
  printf("Valor digitado foi %d\n", n);
+
    escrever "Forneça um número: "
 +
    ler numero
  
   return 0;
+
    n <- 2    
}
+
    enquanto n <= numero / 2 E resto =/= 0 faz     
 +
        resto <- numero % n
 +
        n <- n + 1
 +
    fimenquanto
 +
   
 +
    se resto = 0 entao
 +
        escrever "nao primo: divisivel por " , n - 1
 +
    senao
 +
        escrever "primo"
 +
    fimse
 +
fim
 
</syntaxhighlight>
 
</syntaxhighlight>
** Outras formas de ler do teclado: [http://manpages.ubuntu.com/manpages/karmic/man3/fgets.3.html getchar],  [http://manpages.ubuntu.com/manpages/karmic/en/man3/fgets.3.html fgets]
 
  
==== Tipos de dados ====
+
... uma versão com uma pequena melhoria:
{| border="1" cellpadding="2"
+
 
!Portugol
+
<syntaxhighlight lang=text>
!C
+
inicio
!Exemplo
+
    inteiro numero, n, resto
|-
 
|inteiro||int|| int x;
 
|-
 
|caracter||char|| char letra;
 
|-
 
|real||float ou double|| float pi = 3.1416;
 
|-
 
|texto||char * ou vetor de char || char * mensagem = "Hello world";<br>char palavra[16];
 
|-
 
|logico||qualquer tipo (valor 0 = Falso, qualquer outro valor = Verdadeiro)||int ok = 1;<br>char teste = 0;
 
|}
 
  
 +
    escrever "Forneça um número: "
 +
    ler numero
  
==== Comandos/funções e Estruturas de controle ====
+
   
{| border="1" cellpadding="2"
+
    resto <- numero % 2
!Portugol
+
    se resto =/= 0 entao
!C
+
      n <- 1
!Exemplo
+
      enquanto n <= numero / 2 E resto =/= 0 faz     
|-
+
          n <- n + 2
|<syntaxhighlight lang=text>inteiro n, m
+
          resto <- numero % n
 +
      fimenquanto
 +
     
 +
      se resto = 0 entao
 +
          escrever "nao primo: divisivel por " , n
 +
      senao
 +
          escrever "primo"
 +
      fimse
 +
    senao
 +
      escrever "nao primo: divisivel por 2"
 +
    fimse
 +
fim
 +
</syntaxhighlight>
  
escrever "Ola, mundo!\n"
+
4) Jogo de Pedra, papel e tesoura: nesse jogo cada jogador faz sua escolha, e vence aquele que escolher um objeto que seja capaz de vencer o outro:
escrever "valor de n: ", n, "\n"
 
escrever "valor de n: ", n, ", e valor de x: ", x, "\n"
 
</syntaxhighlight>||<syntaxhighlight lang=c>int n, m;
 
  
printf("Ola, mundo!\n");
+
<syntaxhighlight lang=text>
printf("valor de n: %d\n", n);
+
* Tesoura corta papel
printf("valor de n: %d, e valor de x: %d\n", n, x);</syntaxhighlight>||
+
* Pedra quebra tesoura
|-
+
* Papel embrulha a pedra
|<syntaxhighlight lang=text>inteiro n
+
</syntaxhighlight>
  
ler n</syntaxhighlight>||<syntaxhighlight lang=c>int n;
+
Faça um algoritmo que pergunte ao usuário o objeto escolhido, e em seguida faça o computador escolher o seu próprio objeto (dica: use o gerador de números aleatórios visto no jogo do par ou ímpar). Ao final o algoritmo deve informar o vencedor.
  
scanf("%d", &n);</syntaxhighlight>||
+
<syntaxhighlight lang=text>
|-
+
inicio
|<syntaxhighlight lang=text>se condição então
+
    inteiro lancecomputador , lancejogador
//comandos
+
    logico jogadorvenceu
fimse</syntaxhighlight> ||<syntaxhighlight lang=c>if (condição) {
+
   
//comandos
+
    escrever "Escolha papel (1) , pedra (2) ou tesoura (3): "
}</syntaxhighlight>||
+
    ler lancejogador
 +
    se lancejogador > 3 ou lancejogador < 1 entao
 +
        escrever "Lance inválido !!! (deve digitar um número entre 1 e 3)\n\n"
 +
    senao
 +
        lancecomputador <- 1 + 2 * aleatorio ( )
 +
        escrever "Computador escolheu: " , lancecomputador , "\n"
 +
        se lancejogador = lancecomputador entao
 +
            escrever "Empate !\n\n"
 +
        senao
 +
          jogadorvenceu <- falso
 +
          escolhe lancejogador
 +
              caso 1:
 +
                  se lancecomputador = 2 entao
 +
                      jogadorvenceu <- verdadeiro
 +
                  fimse
 +
              caso 2:
 +
                  se lancecomputador = 3 entao
 +
                      jogadorvenceu <- verdadeiro
 +
                  fimse
 +
              caso 3:
 +
                  se lancecomputador = 1 entao
 +
                      jogadorvenceu <- verdadeiro
 +
                  fimse
 +
          fimescolhe
 +
          escrever "Vencedor: "
 +
          se jogadorvenceu entao
 +
              escrever "jogador\n\n"
 +
          senao
 +
              escrever "computador\n\n"
 +
          fimse
 +
      fimse
 +
    fimse
 +
fim
 +
</syntaxhighlight>
 +
 
 +
... uma outra solução:
 +
 
 +
<code>
 +
inicio
 +
    inteiro lancecomputador , lancejogador
 +
    logico jogadorvenceu
 +
    texto vencedor , opcao
 +
   
 +
    escrever "Jogo de Papel , Pedra e Tesoura\n\n"
 +
    escrever "Escolha papel (1) , pedra (2) ou tesoura (3): "
 +
    ler lancejogador
 +
    se lancejogador > 3 ou lancejogador < 1 entao
 +
        escrever "Lance inválido !!! (deve digitar um número entre 1 e 3)\n\n"
 +
    senao
 +
        lancecomputador <- 1 + 2 * aleatorio ( )
 +
        escrever "Computador escolheu: " , lancecomputador , "\n"
 +
        se lancejogador = lancecomputador entao
 +
            escrever "Empate !\n\n"
 +
        senao
 +
            escrever "Vencedor: "
 +
            // repare em como se determina o vencedor nesta versao do jogo: a condicao abaixo
 +
            // é verdadeira sempre que o lance do jogador estiver uma unidade atrás do computador:
 +
            //
 +
            // Jogador    Computador
 +
            //    1            2
 +
            //    2            3
 +
            //   3            1 (nesta última preciso considerar que a contagem é circular)
 +
               
 +
            se (lanceJogador%3 + 1) = lanceComputador entao
 +
                escrever "jogador\n\n"
 +
            senao
 +
                escrever "computador\n\n"
 +
            fimse
 +
        fimse
 +
    fimse
 +
fim
 +
</syntaxhighlight>
 +
 
 +
5) Um programa somente pode ser usado por usuários autorizados. Os usuários e seuas senhas estão armazenados em uma tabela dentro do próprio programa. Escreva o algoritmo desse programa que pede o usuário e senha pelo teclado, consulta a tabela de usuários e depois informa se o acesso foi concedido ou negado. Os usuários e respectivas senhas estão mostrados abaixo:
 +
 
 +
{| border="1" cellpadding="2"
 +
!Usuário
 +
!Senha
 +
|-
 +
|joao || tainha
 +
|-
 +
|maria || pirao
 +
|-
 +
|maneca || onibus
 +
|-
 +
|bilica || bilro
 
  |-
 
  |-
  |<syntaxhighlight lang=text>se condição então
+
  |}
   //comandos
+
 
senão
+
Obs: essa tabela pode ser modificada futuramente, então o algoritmo deve conseguir trabalhar com quaisquer usuários e senhas que ali sejam armazenados. Isso é, mesmo que essa tabela tenha seu conteúdo alterado, o algoritmo não pode ser modificado.
   //comandos
+
 
fimse</syntaxhighlight> ||<syntaxhighlight lang=c>if (condição) {
+
<syntaxhighlight lang=text>
//comandos
+
inicio
} else {
+
   texto usuarios <- {"joao", "maria", "maneca", "bilica"}
//comandos
+
   texto senhas <- {"tainha", "pirao", "onibus", "bilro"}
}</syntaxhighlight> ||
+
  texto login, umaSenha
|-  
+
  inteiro i
|<syntaxhighlight lang=text>escolhe expressão
+
  logico OK <- falso
   caso valor1:
+
 
    //comandos
+
   escrever "Usuario:"
   caso valor2:
+
  ler login
    //comandos
+
 
   defeito:
+
   escrever "Senha:"
     //comandos
+
  ler umaSenha
fimescolhe</syntaxhighlight>||<syntaxhighlight lang=c>switch (expressão) {
+
 
   case valor1:
+
   enquanto i < 4 E nao OK faz
     //comandos
+
     ok <- usuarios[i] = login
   case valor2:
+
    i <- i + 1
     //comandos
+
  fimenquanto
   default:
+
 
    //comandos
+
  se ok entao
}</syntaxhighlight>||
+
    ok <- senhas[i] = umaSenha
 +
  fimse
 +
 
 +
   se ok entao
 +
     escrever "acesso concedido"
 +
   senao
 +
     escrever "acesso negado"
 +
  fimse
 +
    
 +
fim
 +
</syntaxhighlight>
 +
 
 +
=== Conceitos ===
 +
 
 +
''Legenda:''
 +
* '''T:''' questão correta
 +
* '''P:''' questão parcialmente correta
 +
* '''N:''' questão incorreta ou em branco
 +
 
 +
{| border="1" cellpadding="2"
 +
!Aluno
 +
!Conceito
 +
!Questão 1
 +
!Questão 2
 +
!Questão 3
 +
!Questão 4
 +
!Questão 5
 
  |-
 
  |-
  |<syntaxhighlight lang=text>enquanto condição faz
+
  |Alfredo || D || T (mas não da forma mais eficiente) || N || N || N || N
//comandos
 
fimenquanto</syntaxhighlight>||<syntaxhighlight lang=c>while (condição) {
 
  //comandos
 
}</syntaxhighlight>||
 
 
  |-
 
  |-
  |<syntaxhighlight lang=text>para variavel de inicio ate fim passo incremento
+
  |Ângelo || D || T (mas não da forma mais eficiente) || N || P || N || N
//comandos
 
proximo</syntaxhighlight>||<syntaxhighlight lang=c>for (variavel=inicio; variavel <= fim; variavel++) {
 
//comandos
 
}</syntaxhighlight>||
 
|}
 
 
 
=== Atividades ===
 
 
 
* Traduza para C os seguintes algoritmos Portugol:
 
*# <syntaxhighlight lang=text>
 
Inicio
 
  inteiro x, y
 
 
  Escrever "Digite um numero: ",
 
  Ler x
 
  Escrever "Digite outro numero: ",
 
  Ler y
 
  Escrever "Soma = ", x+y
 
Fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
Inicio
 
  inteiro x, y, z
 
 
  Escrever "Digite um numero: ",
 
  Ler x
 
  Escrever "Digite outro numero: ",
 
  Ler y
 
  z <- x + y
 
  Escrever "Soma = ", z
 
Fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
Inicio
 
  Inteiro n1, n2, n3, r
 
 
  Escrever "Primeiro numero: "
 
  ler n1
 
  Escrever "Segundo numero: "
 
  ler n2
 
  Escrever "Terceiro numero: "
 
  ler n3
 
 
  r <- (n1 + n2 + n3) /3
 
 
  Escrever "Media=", r
 
Fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
Inicio
 
  inteiro anos, dias
 
 
  Escrever "Quantos anos se passaram ? "
 
  Ler anos
 
  dias <- anos * 365
 
  Escrever "... então se passaram ", dias, " dias"
 
Fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
Inicio
 
  constante inteiro diasPorAno <- 365
 
  inteiro anos, dias
 
 
  Escrever "Quantos anos se passaram ? "
 
  Ler anos
 
  dias <- anos * diasPorAno
 
  Escrever "... então se passaram ", dias, " dias"
 
Fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
  real massa
 
  real forca
 
  real aceleracao
 
  constante real g <- 9.8
 
  constante real ac <- 0.01
 
 
 
  escrever "força: "
 
  ler forca
 
 
 
  escrever "massa: "
 
  ler massa
 
 
 
  aceleracao <- forca/massa - ac*massa*g
 
 
 
  escrever "aceleracao=", aceleracao, " m/s2"
 
 
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
  real a, b, c
 
  real delta
 
  real x1, x2
 
 
 
  escrever "Forneça os coeficientes da equação de 2o grau (formato: ax^2 + bx + c):\n"
 
  escrever "a="
 
  ler a
 
  escrever "b="
 
  ler b
 
  escrever "c="
 
  ler c
 
 
 
  delta <- b^2 - 4*a*c
 
  x1 <- (-b + delta^0.5)/(2*a)
 
  x2 <- (-b - delta^0.5)/(2*a)
 
 
 
  escrever "1a raiz=", x1
 
  escrever "\n2a raiz=", x2 
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
      real x1, x2, x3, x4, x5
 
      real media
 
      real desvioPadrao
 
      constante inteiro N <- 5
 
     
 
      // Ler os cinco valores pelo teclado
 
      ler x1
 
      ler x2
 
      ler x3
 
      ler x4
 
      ler x5
 
     
 
      // Calcular a media
 
      media <- (x1 + x2 + x3 + x4 + x5) / N
 
     
 
      // Calcular o desvio padrao
 
      desvioPadrao <- (x1 - media)^2 + (x2 - media)^2 + (x3 - media)^2 + (x4 - media)^2 + (x5 - media)^2
 
      desvioPadrao <- (desvioPadrao / (N-1))^0.5
 
     
 
      escrever "Media=", media
 
      escrever "\nDesvio padrao=", desvioPadrao
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
      real renda
 
      real irpf
 
     
 
      escrever "Informe sua renda: "
 
      ler renda
 
     
 
      se renda < 1372.81 entao
 
        irpf <- 0
 
      senao
 
        se renda < 2743.25 entao
 
          irpf <- (renda - 205.92)*0.15
 
        senao
 
          irpf <- (renda - 548.82)*0.275
 
        fimSe
 
      fimSe
 
     
 
      escrever "Imposto devido: ", irpf 
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro dividendo
 
    inteiro divisor
 
    inteiro resto , quociente
 
 
 
    escrever "Dividendo: "
 
    ler dividendo
 
    escrever "Divisor: "
 
    ler divisor
 
 
 
    se divisor = 0 entao
 
        escrever "Nao pode dividir: divisor = 0 !!!"
 
    senao
 
        resto <- dividendo % divisor
 
        se resto = 0 entao
 
            quociente <- dividendo / divisor
 
            escrever dividendo , " / " , divisor , " = " , quociente
 
        senao
 
            escrever "Nao pode fazer divisao inteira"
 
            escrever " (resto = " , resto , ")"
 
        fimse
 
    fimse
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro termometro , barometro , tempo
 
 
 
    escrever "Qual a condição do termômetro: \n"
 
    escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
 
    ler termometro
 
 
 
    escrever "Informe a condição do barômetro: \n"
 
    escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
 
    ler barometro
 
 
 
    tempo <- termometro*10 + barometro
 
 
 
    escolhe tempo
 
        caso 11:
 
            escrever "Chuvas abundantes e ventos de sul a sudoeste fortes"
 
        caso 12:
 
            escrever "Chuva Provavel , ventos de sul a sudeste"
 
        caso 13:
 
            escrever "Tempos Bons , ventos de sul a sudeste"
 
        caso 21:
 
            escrever "Frente quente com chuva provavel"
 
        caso 22:
 
            escrever "Tempo Incerto , ventos variaveis"
 
        caso 23:
 
            escrever "Tempos Bons , ventos do leste frescos"
 
        caso 31:
 
            escrever " Tempo instavel , aproximaçao de frente"
 
        caso 32:
 
            escrever "Tempo Mudando para bom , ventos de leste"
 
        caso 33:
 
            escrever "Tempos Bons , ventos quentes e secos"
 
        defeito:
 
            escrever "Utilize somente os algorismos de 1 a 3 para indicar as condiçoes do equipamentos"
 
    fimescolhe
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    // coeficientes do polinomio  [ax^2 + bx + c = 0 ]
 
    real a , b , c
 
   
 
    escrever "Forneça os coeficientes da equação de 2o grau:\n"
 
    escrever "a="
 
    ler a
 
    escrever "b="
 
    ler b
 
    escrever "c="
 
    ler c
 
   
 
    // equação do tipo [ bx + c = 0 ]
 
    se a = 0 entao
 
        escrever " não é uma equação de 2o grau !!!"
 
    senao
 
        // calcular o delta => interior da raiz
 
        real delta
 
       
 
        delta <- b ^ 2 - 4 * a * c
 
       
 
        // não existem raizes  reais de números negativos
 
        se delta < 0 entao
 
            escrever " não tem raizes reais"
 
        senao
 
            // -----------  raiz dupla  ----------------
 
            se delta = 0 entao
 
                real x1
 
                x1 <- -b / 2 * a
 
                escrever "\nraiz dupla : " , x1
 
            senao
 
                // - ---------- duas raizes ---------------
 
                real x1 , x2
 
                x1 <- ( -b + raiz ( delta ) ) / 2 * a
 
                x2 <- ( -b - raiz ( delta ) ) / 2 * a
 
                escrever "\nraiz x1 : " , x1
 
                escrever "\nraiz x2 : " , x2
 
            fimse//raiz dupla
 
        fimse// delta >0
 
    fimse// a <> 0
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro num, bit
 
    inteiro base
 
   
 
    escrever "Este algoritmo converte um número inteiro para sua representação binária.\n"
 
    Escrever "O número deve estar entre 0 e 131071, e a representação binária\n"
 
    escrever "terá 17 bits\n\n"
 
    escrever "Digite o numero a ser convertido para binário: "
 
    ler num
 
 
 
    base <- 65536
 
    enquanto base > 0 faz
 
        bit <- ( num / base ) % 2
 
        escrever bit, " "
 
        base <- base / 2
 
    fimenquanto
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro n , fat
 
    ler n
 
    fat <- 1
 
    enquanto n > 1 faz
 
      fat <- fat * n
 
      n <- n - 1       
 
    fimenquanto
 
   
 
    escrever "Fatorial=" , fat
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro valor , menor , contador
 
    ler menor
 
    contador <- 9
 
    enquanto contador > 0 faz
 
        ler valor
 
        se valor < menor entao
 
            menor <- valor
 
        fimse
 
        contador <- contador - 1
 
    fimenquanto
 
    escrever "menor valor = ", menor
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
inicio
 
    inteiro numero , n , resto <- 1
 
    escrever "Forneçaa um número: "
 
    ler numero
 
    n <- 2
 
    enquanto n <= numero / 2 e resto =/= 0 faz
 
        resto <- numero % n
 
        escrever "n=" , n , ": resto=" , resto , "\n"
 
        n <- n + 1
 
    fimenquanto
 
   
 
    se resto = 0 entao
 
        escrever "nao primo: divisivel por " , n - 1
 
    senao
 
        escrever "primo"
 
    fimse
 
fim
 
</syntaxhighlight>
 
*# <syntaxhighlight lang=text>
 
// Le 5 valores e identifica os dois maiores.
 
inicio
 
    inteiro valor , maior1 , maior2 , contador
 
   
 
    ler valor
 
    maior1 <- valor
 
   
 
    ler valor
 
    se valor > maior1 entao
 
      maior2 <- maior1
 
      maior1 <- valor
 
    senao
 
      maior2 <- valor
 
    fimse
 
 
 
    contador <- 3
 
    enquanto contador > 0 faz
 
        ler valor
 
        se valor > maior1 entao
 
            maior2 <- maior1
 
            maior1 <- valor
 
        senao
 
            se valor > maior2 entao
 
                maior2 <- valor
 
            fimse
 
        fimse
 
        contador <- contador - 1
 
    fimenquanto
 
 
 
    escrever "maior valor = " , maior1
 
    escrever "\nsegundo maior valor = " , maior2
 
fim
 
</syntaxhighlight>
 
 
 
== 08/11: Começo do projeto ==
 
 
 
Para referência: [http://www.ead.cpdee.ufmg.br/cursos/C/c.html apostila online sobre linguagem C].
 
 
 
=== O jogo: Campo Minado ===
 
 
 
O [http://pt.wikipedia.org/wiki/Campo_minado ''Campo Minado''] é um jogo clássico disponível em diferentes sistemas operacionais. Basicamente devem-se procurar as minas existentes em um campo quadriculado, perdendo-se o jogo se ''pisar''  em uma mina, e vencendo se descobrir todas as minas. O jogo  oferece pistas sobre a localização das minas, e usando-se um  pouco de lógica podem-se encontrá-las. Uma  versão desse jogo para Linux aparece na figura a seguir .
 
 
 
[[imagem:Minas-linux.png]]
 
 
 
A cada jogada deve-se escolher uma casa do tabuleiro, optando-se por ''pisá-la'' (revelando-a) ou marcá-la como uma provável mina. Ao ''pisar'' numa casa, três ações podem ocorrer:
 
# A casa contém uma mina, o que revela  todas as demais minas e encerra o jogo com derrota.
 
# A casa é vizinha a uma ou mais minas, e assim em seu lugar aparece o número de minas em casas vizinhas.
 
# Nenhuma mina existe em casas vizinhas, então a casa aparece vazia. Além disso, todas as casas vizinhas que também  não possuam minas são reveladas.
 
 
 
Ao final do jogo, a pontuação é dada pelo tempo que se levou para se descobrirem as minas. O jogo contabiliza uma lista de melhores jogadas, contendo o nome do jogador e o tempo de jogo. Melhores jogadas são aquelas com menores tempos de jogo.
 
 
 
A versão do jogo a ser desenvolvida será baseada em texto ''(mas quem quiser fazer uma versão gráfica fique à vontade !)''. Basicamente  deve ser apresentado o tabuleiro, mostrando-se apenas as casas já reveladas e o tempo de jogo decorrido, e um ''prompt'' pedindo as coordenadas da próxima casa assim como o tipo de ação (se ''pisar'' na casa ou marcar a provável existência de uma mina).
 
 
 
=== Etapas de desenvolvimento ===
 
 
 
* Proposta de código que contempla:
 
** Bibliotecas e definições
 
** Declaração de constantes e variáveis
 
*** Uso de variáveis com mesmo nome e diferentes escopos
 
*** Vetor e matriz
 
** Operadores lógicos e matemáticos
 
** Expressões
 
*** E/S
 
*** Estruturas de decisão e repetição
 
** Funções
 
*** Passagem de parâmetro por valor e por referência
 
** Acesso a uma matriz através de um vetor linear (função <tt>iniciaJogo</tt>)
 
** Ponteiros
 
 
 
=== O laço principal do jogo ... ===
 
 
 
O jogo funciona com um laço principal, que lê as coordenadas da pŕoxima casa e a ação a ser feita, e atualiza o tabuleiro apropriadamente:
 
 
 
[[imagem:Fluxo-Campo-Minado.png|400px]]
 
 
 
Note que nesse laço devem-se contabilizar quantas minas fora descobertas. Esse valor é necessário para decidir se o jogo já acabou.
 
 
 
=== O tabuleiro ===
 
 
 
Representar o tabuleiro: etapas X a Y.
 
 
 
O tabuleiro do campo minado deve ser representado em formato de texto conforme a figura abaixo. Nesta figura, o tabuleiro está completamente revelado.
 
 
 
 
 
[[imagem:Tabuleiro-Minado.png]]
 
 
 
 
 
As colunas são identificadas por letras maiúsculas, e as linhas por números. Cada casa do tabuleiro pode conter uma mina ou um número que conta quantas minas existem em casas vizinhas. Casas já reveladas devem mostrar essa contagem de minas vizinhas, sendo que se não houver  minas a casa deve ser mostrada vazia (preenchida com espaço). Casa em  que se marcou a existência de uma mina deve mostrar a letra '''M'''. Por fim, casas ainda não reveladas devem ser preenchidas com o caractere '''?'''. Um tabuleiro parcialmente revelado apareceria como a figura abaixo.
 
 
 
 
 
[[imagem:Tabuleiro-Minado2.png]]
 
 
 
 
 
==== Etapa 1: Modelando  o tabuleiro ====
 
 
 
Conteúdos abordados:
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c300.html declaração de variáveis]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c940.html saída de dados na tela]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c500.html Matrizes]
 
 
 
Deve-se nessa etapa escolher uma forma de representar o tabuleiro dentro do programa. Devem-se pensar também  em como representar as casas do campo minado:
 
* Como definir que uma casa possui uma mina ?
 
* Como definir que uma casa está  vazia ? Nesse caso, como informar a quantidade de minas em casas vizinhas ?
 
* Como definir se uma casa já foi pisada ?
 
 
 
Ponto de partida para mostrar um tabuleiro na tela. O tabuleiro foi modelado como uma matriz de inteiros bidimensional.
 
 
 
<syntaxhighlight lang=c>
 
#include <stdio.h>
 
 
 
int main(int argc, char** argv) {
 
    int tabuleiro[5][5] = { {0,0,1,1,0}, {1,1,2,0,0}, {10,1,0,0,0},
 
                          {1,1,1,0,0}, {0,0,0,0,10} };
 
    int i, j;
 
 
 
    //i indica a linha, j a coluna
 
    i = 0;
 
    while (i < 5) {
 
        // O while abaixo mostra todas as colunas da linha "i"
 
        j = 0;
 
        while (j < 5) {
 
            printf("%d ", tabuleiro[i][j]);
 
            j++;
 
        }
 
 
 
        printf("\n");
 
        i++;
 
    }
 
 
 
    return (EXIT_SUCCESS);
 
}
 
</syntaxhighlight>
 
 
 
Cada casa dessa matriz contém um número inteiro. Para representar o estado de cada casa do tabuleiro sugeriu-se um código numérico:
 
* ''Casa com mina:'' algum número > 8
 
* ''Casa sem mina:'' um número que conte quantas minas existem em casas vizinhas. Tal número estará entre 0 e 8.
 
 
 
 
 
 
 
== 09/11: O tabuleiro ==
 
 
 
Conteúdos abordados:
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c300.html declaração de variáveis]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c530.html matrizes de números]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c940.html saída de dados na tela]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c410.html estrutura de decisão: ''if'']
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c440.html estrutura de repetição: ''while'']
 
 
 
=== Desenhando o tabuleiro completo ===
 
 
 
Dando continuidade à visualização do tabuleiro, agora deve-se desenhá-lo por completo na tela. Quer dizer, mostrar um tabuleiro com 10 linhas e 10 colunas, seguindo a abordagem da [[SOP-2010-2-sobral#08.2F11:_Come.C3.A7o_do_projeto|aula anterior]].
 
 
 
Para mostrar o tabuleiro, deve-se explorar o uso da função [[SOP-stdio|printf]], que serve para mostrar dados na tela.
 
 
 
=== Zerando o tabuleiro ===
 
 
 
Fluxograma do algoritmo que zera o tabuleiro:
 
 
 
[[imagem:Zera-tabuleiro.png|300px]]
 
 
 
 
 
... e esse fluxograma traduzido para linguagem C:
 
 
 
<syntaxhighlight lang=c>
 
    // Zerar todas as posições do tabuleiro !
 
    i = 0;
 
    while ( i < LINHAS) {
 
        j = 0;
 
        while (j < COLUNAS) {
 
            tabuleiro[i][j] = 0;
 
            j++;
 
        }
 
        i++;
 
    }
 
</syntaxhighlight>
 
 
 
=== Colocando as minas ===
 
 
 
No tabuleiro devem ser colocadas 10 minas aleatoriamente. Isso significa que para cada uma dessas minas, devem-se gerar aleatoriamente  suas coordenadas no tabuleiro. A geração de números aleatórios pode ser feita usando-se a função [http://manpages.ubuntu.com/manpages/lucid/man3/random.3.html random()]. Essa função gera um número inteiro entre 0 e 2147483647. Para usá-la no tabuleiro, pode-se fazer o seguinte:
 
 
 
<syntaxhighlight lang=c>
 
#include <stdlib.h>
 
 
 
int main() {
 
  int x, y;
 
 
 
  // gera dois números aleatórios entre 0 e 9, e guarda-os nas variáveis x e y
 
  x = random() % 10;
 
  y = random() % 10;
 
 
 
}
 
</syntaxhighlight>
 
 
 
A versão atual do jogo está assim:
 
 
 
<syntaxhighlight lang=c>
 
#include <stdio.h>
 
#include <stdlib.h>
 
 
 
#define MINAS 10
 
#define LINHAS 10
 
#define COLUNAS 10
 
 
 
#define TEM_MINA 10
 
 
 
/*
 
*
 
*/
 
int main(int argc, char** argv) {
 
    int tabuleiro[LINHAS][COLUNAS];
 
    int i, j;
 
 
 
    // Zerar todas as posições do tabuleiro !
 
    i = 0;
 
    while ( i < LINHAS) {
 
        j = 0;
 
        while (j < COLUNAS) {
 
            tabuleiro[i][j] = 0;
 
            j++;
 
        }
 
        i++;
 
    }
 
   
 
    // Colocar as minas aqui !!!
 
 
 
    //Agora mostrar o tabuleiro que foi gerado
 
 
 
    //i indica a linha, j a coluna
 
    i = 0;
 
    while (i < LINHAS) {
 
        // O while abaixo mostra todas as colunas da linha "i"
 
        j = 0;
 
        while (j < COLUNAS) {
 
            if (tabuleiro[i][j] == TEM_MINA) {
 
                printf("* ");
 
            } else {
 
                if (tabuleiro[i][j] == 0) {
 
                    printf("  ");
 
                } else {
 
                    if ((tabuleiro[i][j] >= 1) && (tabuleiro[i][j] <= 8)) {
 
                        printf("%d ", tabuleiro[i][j]);
 
                    }
 
                }
 
            }
 
            j++;
 
        }
 
 
 
        printf("\n");
 
        i++;
 
    }
 
 
 
    return (EXIT_SUCCESS);
 
}
 
</syntaxhighlight>
 
 
 
=== Meta desta semana ===
 
 
 
Nesta semana  o seu programa deve ser capaz de:
 
# Representar o tabuleiro com uma matriz de inteiros bidimensional.
 
# Mostrar o tabuleiro na tela, indicando o que existe em cada casa. O tabuleiro deve ser apresentado  ''enfeitado'', o que significa que deve mostrar uma grade para separar as casas e a identificação de linhas e colunas. Linhas  devem ser identificadas numericamente, e colunas devem ser identificadas por letras (ex: primeira coluna=A, segunda coluna=B, ...).
 
# Colocar minas aleatoriamente no tabuleiro.
 
 
 
O seu programa cumprindo as metas acima '''deve ser apresentado ao professor na próxima aula (16/11), já sendo parte da avaliação desta unidade'''.
 
 
 
== 16/11: As minas e o tabuleiro ==
 
 
 
Conteúdos abordados:
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c300.html declaração de variáveis]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c940.html saída de dados na tela]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c410.html estrutura de decisão: ''if'']
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c440.html estrutura de repetição: ''while'']
 
 
 
=== Dúvidas sobre random() ===
 
 
 
<syntaxhighlight lang=c>
 
#include <stdio.h>
 
#include <stdlib.h>
 
#include <sys/time.h>
 
 
 
int main() {
 
  int x,y,z;
 
  time_t t;
 
 
 
  // Lê o valor do relógio
 
  t = time(NULL);
 
 
 
  // Usa o valor do relógio como semente do gerador de números
 
  // pseudo-aleatórios.
 
  srandom(t);
 
 
 
  // Sorteia três números pseudo-aleatórios
 
  x = random() % 10;
 
  y = random() % 10;
 
  z = random() % 10;
 
 
 
  printf("x=%d, y=%d, z=%d\n", x, y, z);
 
}
 
</syntaxhighlight>
 
=== Funções ===
 
 
 
A função ''mostra_tabuleiro'' corresponde ao algoritmo para desenhar o tabuleiro na tela. O tabuleiro em si é representado pela variável ''tabuleiro'', que tem escopo global (visível em todas as funções do programa). Neste momento é bom ler um pouco mais sobre funções na linguagem C:
 
* [[SOP-funcoes|Texto introdutório sobre funções na linguagem C]]
 
 
 
 
 
 
 
=== Para fixar ===
 
 
 
# Escreva um  programa para ler 10 números do teclado, e em seguida mostrar o maior e o menor número.
 
# Reescreva o programa anterior, mas criando as funções "maior" e "menor", que retornam o maior e menor número de um vetor de tamanho arbitrário.
 
# Faça um  programa  que desenhe a seguinte figura na tela (use uma função para desenhar cada linha): <syntaxhighlight lang=text>
 
    '
 
    *
 
  ***
 
  *****
 
*******
 
*********
 
*******
 
  *****
 
  ***
 
    *
 
</syntaxhighlight>
 
# Até que ponto se poderiam usar funções para mostrar o tabuleiro ? Imagine que mostrar o tabuleiro fosse uma tarefa composta por 2 subtarefas:
 
## ''Mostrar o estado de uma casa do tabuleiro''
 
## ''Mostrar o tabuleiro completo:'' mostrar todas as casas (usa subtarefa anterior).<br>Note que essas subtarefas correspondem à sequência do projeto até o momento. Um bom exercício seria modelar a tarefa de mostrar o tabuleiro usando uma função para cada subtarefa escrita acima.
 
 
 
== 22/11: Contando minas vizinhas ==
 
 
 
=== A contagem de minas ===
 
 
 
Para ajudar a visualizar o algoritmo de contagem de minas vizinhas, segue o fluxograma abaixo. As coordenadas ''(x,y)'' correspondem à casa cujas minas vizinhas se desejam contar. Veja sua semelhança com o [[SOP-2010-2-sobral#Zerando_o_tabuleiro|algoritmo para zerar o tabuleiro]].
 
 
 
[[imagem:Conta-minas.png|300px]]
 
 
 
A sugestão é por o algoritmo de contagem em uma [[SOP-funcoes|função]], e dentro do programa chamá-lo para cada casa do tabuleiro:
 
 
 
<syntaxhighlight lang=c>
 
#include <stdio.h>
 
#include <stdlib.h>
 
 
 
#define MINAS 10
 
#define LINHAS 10
 
#define COLUNAS 10
 
 
 
#define TEM_MINA 10
 
 
 
// O tabuleiro como variável global: necessário para poder ser usado e modificado por outras funções
 
int tabuleiro[LINHAS][COLUNAS];
 
 
 
 
 
// Função para contar as minas vizinhas da casa x, y
 
void conta(int x, int y) {
 
 
 
 
 
}
 
 
 
/*
 
*
 
*/
 
int main(int argc, char** argv) {
 
    int i, j;
 
 
 
    // Zerar todas as posições do tabuleiro !
 
    i = 0;
 
    while ( i < LINHAS) {
 
        j = 0;
 
        while (j < COLUNAS) {
 
            tabuleiro[i][j] = 0;
 
            j++;
 
        }
 
        i++;
 
    }
 
   
 
    // Colocar as minas aqui !!!
 
 
 
    // contar as minas vizinhas de cada casa do tabuleiro
 
    i = 0;
 
    while ( i < LINHAS) {
 
        j = 0;
 
        while (j < COLUNAS) {
 
            // chama a função de contagem de minas, para contar as minas
 
            // vizinhas da casa com coordenadas i,j
 
            conta(i, j);
 
            j++;
 
        }
 
        i++;
 
    }
 
 
 
    //Agora mostrar o tabuleiro que foi gerado
 
</syntaxhighlight>
 
 
 
Obs: cuidado com o seguinte ao fazer o algoritmo de contagem de minas:
 
* Casas na borda do tabuleiro possuem menos casas vizinhas (casas fora do tabuleiro devem ser ignoradas).
 
* O algoritmo não pode ser aplicado em casas que contêm minas. Quer dizer, não faz sentido calcular as minas vizinhas de uma casa que tem uma mina.
 
 
 
Levando em conta as restrições acima, o algoritmo deve ficar parecido com o seguinte:
 
 
 
[[imagem:Conta-minas2.png|320px]]
 
 
 
== 23/11: Pensando no jogo completo !  ==
 
 
 
=== Topicos sobre linguagem C ===
 
 
 
==== Estrutura de repetição ''for'' ====
 
 
 
* Mais detalhes na [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c430.html apostila da UFMG]
 
 
 
A estrutura de repetição ''for'' costuma ser mais usada por programadores do que ''while''. Ao contrário de sua contraparte no Portugol (''para variavel de inicio ate fim ...''), a estrutura ''for'' possui grande flexibilidade para controle das repetições de um laço. Quer dizer, qualquer laço implementado com ''while'' pode ser também implementado com ''for''. O diferencial do ''for'' está na possibilidade de concentrar todas as instruções necessárias para controlar o laço.
 
 
 
A estrutura ''for'' pode ser declarada da seguinte forma:
 
 
 
<syntaxhighlight lang=c>
 
for (inicialização; condição; atualização) {
 
//comandos
 
}
 
</syntaxhighlight>
 
 
 
... sendo:
 
* '''inicialização:'''  instruções a serem executadas no início do laço (antes da primeira repetição)
 
* '''condição:''' condição (expressão lógica) para a continuidade do laço. Deve ser verdadeira para que se execute mais uma repetição.
 
* '''atualização:''' instruções a serem executadas ao final de cada repetição
 
 
 
Por exemplo, para mostrar os números de 1 a 10 na tela:
 
 
 
<syntaxhighlight lang=c>
 
  int x;
 
 
 
  for (x=1; x < 11; x++) {
 
    printf("x=%d\n", x);
 
  }
 
</syntaxhighlight>
 
 
 
Veja no exemplo acima que tudo o que se precisa saber sobre o controle do laço está concentrado na declaração do ''for'':
 
 
 
<syntaxhighlight lang=c>
 
  for (x=1; x < 11; x++) {
 
</syntaxhighlight>
 
 
 
* ''inicialização (x=1):'' o laço inicia atribuindo o valor ''1'' à variável ''x''
 
* ''condição (x < 11):'' o laço continua enquanto o valor da variável ''x'' for menor que ''11''
 
* ''atualização (x++):'' ao final de cada repetição, o valor da variável ''x'' é incrementado
 
 
Muitas possibilidades existem para o uso de ''for''. Veja este outro exemplo:
 
 
 
<syntaxhighlight lang=c>
 
  char c;
 
 
 
  for (c = 'A'; c < 'K'; c++) {
 
    printf("%c ", c);
 
  }
 
</syntaxhighlight>
 
 
 
 
 
O programa do Campo Minado com o uso de ''for'' ao invés de ''while'' fica assim:
 
 
 
<syntaxhighlight lang=c>
 
#include <stdio.h>
 
#include <stdlib.h>
 
 
 
#define MINAS 10
 
#define LINHAS 10
 
#define COLUNAS 10
 
 
 
/*
 
*
 
*/
 
int tabuleiro[LINHAS][COLUNAS];
 
 
 
void conta(int x, int y) {
 
 
 
}
 
 
 
int main(int argc, char** argv) {
 
    int i, j;
 
 
 
    // zerar todas as posições do tabuleiro !
 
    for (i=0; i < LINHAS; i++) {
 
        for (j=0; j < COLUNAS; j++) {
 
            tabuleiro[i][j] = 0;
 
        }
 
    }
 
 
 
    // Colocar as minas aqui !!!
 
 
 
    // Contar as minas vizinhas
 
    for (i=0; i < LINHAS; i++) {
 
        for (j=0; j < COLUNAS; j++) {
 
            conta(i,j);
 
        }
 
    }
 
 
 
    //Agora mostrar o tabuleiro que foi gerado
 
 
 
    //i indica a linha, j a coluna
 
    for (i=0; i < LINHAS; i++) {
 
        // O while abaixo mostra todas as colunas da linha "i"
 
      for (j=0; i < COLUNAS; j++) {
 
            if (tabuleiro[i][j] == 10) {
 
                printf("* ");
 
            } else {
 
                if (tabuleiro[i][j] == 0) {
 
                    printf("  ");
 
                } else {
 
                    if ((tabuleiro[i][j] >= 1) && (tabuleiro[i][j] <= 8)) {
 
                        printf("%d ", tabuleiro[i][j]);
 
                    }
 
                }
 
            }
 
        }
 
        printf("\n");
 
    }
 
 
 
    return (EXIT_SUCCESS);
 
}
 
</syntaxhighlight>
 
 
 
==== Condições e operadores lógicos ====
 
 
 
* Maiores detalhes na [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c360.html apostila da UFMG]
 
 
 
Obs: cada condição em linguagem C deve ser envolvida por parênteses.
 
 
 
Necessários para expressar condições a serem usadas principalmente com ''if'' e ''while''.
 
 
 
{| border="1" cellpadding="2"
 
!Portugol
 
!C
 
!Exemplo
 
|-
 
|E||&&|| <syntaxhighlight lang=c>
 
if ((x >= 0) && (x <= 9)) {
 
  printf("coordenada x é válida\n");
 
}</syntaxhighlight>
 
|-
 
|OU||<nowiki>||</nowiki>||<syntaxhighlight lang=c>if ((x < 0) || (x > 9)) {
 
  printf("coordenada x é inválida\n");
 
}</syntaxhighlight>
 
|-
 
|NAO||!||<syntaxhighlight lang=c>if (!((x < 0) || (x > 9))) {
 
  printf("coordenada x é válida\n");
 
}</syntaxhighlight>
 
|-
 
|}
 
 
 
=== Uma versão alternativa para a contagem de minas ===
 
 
 
A versão abaixo foi criada pela Marine e Nathany. Apesar de parecida com o [[SOP-2010-2-sobral#A_contagem_de_minas|algoritmo proposto pelo professor]], possui uma diferença importante que a torna mais eficiente.
 
 
 
[[imagem:Conta-minas-nathany-marine.png|320px]]
 
 
 
== 29/11: Pensando no jogo completo ... ==
 
 
 
* Continuando [[SOP-funcoes|funções]]: variáveis locais e globais, escopo de variáveis.
 
 
 
=== Escondendo as casas não pisadas ===
 
 
 
No momento você deve já conseguir mostrar todo o tabuleiro, porém no jogo apenas as casas já pisadas devem ser reveladas. Pense em como modelar o seu tabuleiro de forma a esconder as casas ainda não pisadas.
 
 
 
Opções:
 
# '''''Criar uma matriz de controle:''''' cada casa dessa nova matriz informa se a casa correspondente do tabuleiro está revelada. Por exemplo, se a casa (1,2) na matriz de controle tiver o valor 0 (zero), então a casa (1,2) do tabuleiro está escondida, e se tiver o valor 1, a casa (1,2) do tabuleiro está revelada.
 
# '''''Explorar os códigos numéricos das casas:''''' os códigos numéricos das casas do tabuleiro poderiam ser estendidos para informar se uma casa está escondida ou revelada. Por exemplo, se o código de uma casa for menor do que 100, ela está escondida, caso contrário está revelada. Assim, se casa (1,2) tiver o valor 3, significa que possui 3 minas vizinhas porém ainda está escondida, e se tiver o valor 103 significa que possui 3 minas vizinhas e já está revelada.
 
 
 
== 30/11: Pensando no jogo completo ... ==
 
 
 
* Continuando [[SOP-funcoes|funções]]: variáveis locais e globais, escopo de variáveis.
 
* [[SOP-stdio|Entrada de dados pelo teclado]]
 
 
 
=== Organizando o programa ===
 
 
 
Nesse momento seu programa deve estar ficando comprido, e já começando a se mostrar difícil de ler. Para torná-lo organizado e compreensível, deve ser reorganizado usando [[SOP-funcoes|funções]]:
 
 
 
<syntaxhighlight lang=c>
 
void inicia_tabuleiro() {
 
}
 
 
 
void mostra_tabuleiro(int tudo) {
 
}
 
 
 
int main() {
 
  // cria o tabuleiro, pondo as minas e contando as minas vizinhas de
 
  // cada casa
 
  inicia_tabuleiro();
 
 
 
  // Mostra o tabuleiro que foi criado
 
  // O argumento da função serve para forçar a mostrar todas as casas
 
  // (mesmo as escondidas), ou mostrar somente as reveladas.
 
  // Se for 1, mostra todas as casas.
 
  // Se for 0, mostra somente as reveladas.
 
  mostra_tabuleiro(1);
 
}
 
</syntaxhighlight>
 
 
 
=== Retomando o laço principal do jogo ... ===
 
 
 
Como apresentado na [[SOP-2010-2-sobral#O_la.C3.A7o_principal_do_jogo_...|aula de 08/11]], o jogo funciona com um laço principal, que lê as coordenadas da casa a ser pisada ou marcada e atualiza o tabuleiro apropriadamente:
 
 
 
Note que nesse laço devem-se contabilizar quantas minas foram realmente descobertas - o que não corresponde necessariamente a quantas minas foram  marcadas ... A quantidade de minas descobertas é necessária para decidir se o jogo já acabou.
 
 
 
[[imagem:Fluxo-Campo-Minado.png|400px]]
 
 
 
A leitura das coordenadas deve ser feita usando a função [[SOP-stdio#Explorando_scanf_para_ler_dados_do_teclado|scanf]].
 
 
 
=== Pisando e marcando casas ===
 
 
 
O jogador pode pisar em uma casa. Ao pisar em uma casa com mina, o jogador perde e o jogo acaba. Se pisar em uma casa com minas vizinhas, o jogo revela a contagem de minas vizinhas daquela casa. Mas se a casa pisada não tiver minas vizinhas, o jogo deve revelar todas as casas vizinhas na mesma condição, até o limite de casas que possuem minas vizinhas. Na figura abaixo, pisou-se na casa que está sob o ponteiro do ''mouse''. Note que todas as casas ao redor também foram reveladas, até alcançarem-se casas que têm minas vizinhas.
 
 
 
[[imagem:Minas-pisadas.png]]
 
 
 
Pense em como ''abrir'' as casas vazias em torno da casa pisada ...
 
 
 
O jogador pode também marcar uma casa, indicando que acredita ter ali uma mina. Essa casa continua escondida, porém deve ser mostrado que possui uma bandeira, como mostrado na figura abaixo. Se o jogador descobrir todas as minas, o jogo termina com sua vitória. Como se podem representar essas bandeiras no tabuleiro ?
 
 
 
[[imagem:Casas-marcadas.png]]
 
 
 
== 06/12: Pensando no jogo completo ... ==
 
 
 
Uma vez criado o [[SOP-2010-2-sobral#Retomando_o_la.C3.A7o_principal_do_jogo_...|laço principal do jogo]], em que se lêem coordenadas de casas a serem pisadas até que o jogo termine, faltam alguns refinamentos. São eles:
 
# Ao pisar numa casa que não possui minas vizinhas, devem-se abrir todas as casas adjacentes até encontrar casa que possuam minas vizinhas.
 
# Devem-se contabilizar as minas descobertas (casas marcadas), pois se todas as minas tiverem sido marcadas o jogo termina com vitória do jogador.
 
 
 
=== Abrindo casas que não possuem minas vizinhas ===
 
 
 
Ao se pisar em uma casa que não possui minas vizinhas, devem-se abrir recursivamente as casas adjacentes até encontrar casa com minas vizinhas. As figuras a seguir mostram como seria a abertura das casas (em vermelho a casa a ser aberta, e em hachurado as casas adjacentes):
 
 
 
[[imagem:Abre-1.png]]<br><br>
 
''Início: abrindo a casa pisada e testando as casas adjacentes''
 
 
 
 
 
 
 
[[imagem:Abre-2.png|300px]]<br>
 
''Aplicando a abertura de casa à cada casa adjacente''
 
 
 
 
 
 
 
[[imagem:Abre-3.png|300px]]<br>
 
''Novamente aplicando a abertura de casa às casas adjacentes ...''
 
 
 
 
 
O algoritmo que faz essa abertura de casas pode ser expresso da seguinte forma (note a recorrência, quando o algoritmo chama si próprio):
 
 
 
 
 
[[imagem:Abre-casas.png|275px]]
 
 
 
=== Mostrando o tempo decorrido ===
 
 
 
Para mostrar o tempo de jogo decorrido, devem-se contabilizar quantos segundos passaram desde que o jogo começou. Assim, deve-se ler o relógio do computador no início do jogo, e a cada vez que se precisar mostrar o tempo de jogo. Para ler o relógio pode-se usar a função [http://manpages.ubuntu.com/manpages/lucid/en/man2/time.2.html time] (já usada no [[SOP-2010-2-sobral#D.C3.BAvidas_sobre_random.28.29|algoritmo para colocar minas aleatoriamente]]). A contagem de quantos segundos se passaram se resume portanto a ler o relógio e subtrair o valor do relógio lido no início do programa, como no exemplo abaixo:
 
 
 
<syntaxhighlight lang=c>
 
#include <time.h>
 
#include <stdio.h>
 
 
 
int main() {
 
  time_t inicio;
 
  time_t agora;
 
 
 
  inicio = time(NULL);
 
 
 
  // faz alguma coisa
 
 
 
  agora = time(NULL);
 
 
 
  printf("Já passaram %d segundos desde o inicio deste programa ...\n", agora - inicio); 
 
}
 
</syntaxhighlight>
 
 
 
Além de contar o tempo de jogo decorrido, é necessário mostrá-lo em um formato claro para o jogador. Isso quer dizer, mostrar o tempo na forma ''"horas:minutos:segundos"'' ao invés de simplesmente a contagem de segundos. Por exemplo, se já transcorreram 143 segundos, deve-se mostrar ''"00:02:23"''. Para converter a contagem de segundos para esse formato você pode aproveitar o algoritmo de conversão de tempo proposto em um [[SOP-2010-2-sobral#Atividade_5|exercício de Lógica de Programação]]. Para formatar o horário como indicado, você pode conferir a [[SOP-stdio|explicação sobre printf]].
 
 
 
=== Tópico sobre linguagem C: caracteres e strings ===
 
 
 
Caracteres (tipo de dados '''''char''''') seguem a codificação [http://pt.wikipedia.org/wiki/ASCII ASCII].
 
 
 
''Strings'' são o equivalente ao tipo ''Texto'' do Portugol.
 
 
 
* [[SOP-strings|Strings]]: um texto introdutório
 
* Ver também:  [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/c500.html capítulo sobre ''strings'' no curso da UFMG].
 
 
 
==== Para fixar ====
 
 
 
# Faça um  programa que mostre a tabela ASCII: cada caractere com seu correspondente código numérico.
 
# Escreva um  programa que leia separadamente o nome e o sobrenome de uma pessoa, e em seguida mostre-os da seguinte forma na tela: ''sobrenome, nome''.
 
# Escreva um programa para resolver a questão 2 da [[SOP-2010-1-sobral#12.2F05:_2a_avalia.C3.A7.C3.A3o|2a avaliação]].
 
# Escreva um programa que leia uma palavra do teclado, e em seguida mostre-a em ordem invertida.
 
# Faça um programa que leia uma palavra do teclado, e em seguida converta-a para maiúsculas.
 
# Faça um  programa para ler uma frase do teclado, e contar quantas palavras nela existem.
 
 
 
== 07/12: Pensando no jogo completo ==
 
 
 
'''ATENÇÃO: preparar para apresentar o projeto na próxima 2a feira (13/12)'''
 
 
 
 
 
=== Tópicos em linguagem C: estruturas de dados (criação de novos tpos de dados) ===
 
 
 
* [[SOP-struct|Introdução a estruturas de dados em C]]
 
* [http://www.ead.cpdee.ufmg.br/cursos/C/aulas/cb00.html Capítulo sobre estruturas de dados no curso da UFMG]
 
 
 
== 13/12: Últimos ajustes ... ==
 
 
 
== 14/12: Apresentação ==
 
 
 
 
 
Na próxima semana, na 3a feira (14/12), vocês deverão apresentar os projetos sobre o campo minado. A apresentação será individual, em que entrevistarei cada aluno sobre sua participação no trabalho. Assim, se na entrevista for demonstrado que o aluno teve participação ativa no desenvolvimento do programa e domina o entendimento de como ele funciona, então receberá o conceito dado ao trabalho. Se não mostrar que entende como ele funciona, então ficará com conceito D e terá que fazer a prova de recuperação.
 
 
 
Para avaliar o trabalho, irei conferir se:
 
 
 
# O jogo funciona como especificado (ver [[SOP-2010-2-sobral#Retomando_o_la.C3.A7o_principal_do_jogo_...|laço principal do jogo na Wiki]]):
 
## O tabuleiro é criado e inicializado corretamente
 
## O tabuleiro é apresentado na tela conforme especificado na wiki (linhas como números e colunas como letras, grade, casas escondidas, reveladas ou marcadas com bandeiras)
 
## O programa lê as coordenadas de casas fornecidas pelo jogador, e executa corretamente a ação solicitada (pisar ou marcar), além de conferir se as coordenadas são válidas
 
## O jogo termina se uma mina  for pisada, e nesse caso revelando todas as casas do tabuleiro
 
## O jogo termina se o jogador descobrir todas as minas: se as únicas casas não reveladas forem exatamente as casas com minas que podem estar marcadas ou não com bandeiras)
 
## Se ao pisar numa casa sem minas vizinhas, as casas adjacentes são reveladas recursivamente até encontrar casas que tenham minas vizinhas.
 
# O programa está organizado e estruturado com funções
 
 
 
 
 
Ítens extras, que são opcionais e aumentarão o conceito final da disciplina se forem feitos, são:
 
 
 
* Mostrar o tempo de jogo decorrido sempre que for ler novas coordenadas do jogador ([[SOP-2010-2-sobral#Mostrando_o_tempo_decorrido|ver na wiki dicas de como fazer]])
 
* Ter níveis de dificuldade do jogo:
 
** ''Fácil:'' tabuleiro 10 x 10 com 10 minas
 
** ''Médio:'' tabuleiro 15 x 15 com 15 minas
 
** ''Difícil:'' tabuleiro 20 x 20 com 30 minas
 
 
 
A prova de recuperação será dia 20/12 (2a feira), e terá ao menos quatro questões sobre programação em linguagem C a serem resolvidas em laboratório. Essa prova terá que ser resolvida iniciando às 7:30 e terminando no máximo às 10:30 h.
 
 
 
== Conceitos do projeto ==
 
 
 
{| border="1" cellpadding="2"
 
!Aluno
 
!Conceito
 
 
  |-
 
  |-
  |Fernando || C
+
  |Beatriz || D || N || N || P (primos somente menores que 12)|| N ||N
 
  |-
 
  |-
  |Gabriel || D*
+
  |Bruno da Silva || D|| N (erros de sintaxe) || N || T || N (copiado de um exemplo do professor) || N
 
  |-
 
  |-
  |Levi || D*
+
  |Bruno || D || N || N || N || N || N
 
  |-
 
  |-
  |Lindalvo || C
+
  |Caroline || D || T || N || N || N || N
 
  |-
 
  |-
  |Marcos || A
+
  |Danilo || B ||T (mas não da forma mais eficiente) || T (mas não da forma mais eficiente)|| N (idêntico ao do Gabriel)|| T || T (mas não da forma mais eficiente)
 
  |-
 
  |-
  |Marine || D
+
  |Dayze || D || N || N || N || N || N
 
  |-
 
  |-
  |Mario Allan || A
+
  |Fabiana || C || T (mas não da forma mais eficiente)|| P || N || T ||N
 
  |-
 
  |-
  |Mário André || A
+
  |Felipe || C || T || N || N || P (computador deveria jogar) || P (funciona só pra 4 contas)
 
  |-
 
  |-
  |Mário Sérgio || D
+
  |Gabriel || D || T (mas não da forma mais eficiente) || N || N (idêntico ao do Danilo)|| N|| P (funciona só pra 4 contas)
 
  |-
 
  |-
  |Maykon || B
+
  |Grazielle || D || N || N || N || N || N
 
  |-
 
  |-
  |Nathany || D
+
  |Jayson || D || N || N || N || N || P (funciona só pra 4 contas)
 
  |-
 
  |-
  |Paulo Igor || D*
+
  |Juliano || C|| T (mas não da forma mais eficiente)||N||N ||T|| P (funciona só pra 4 contas)
 
  |-
 
  |-
  |Ricardo || D
+
  |Leandro || D|| T (mas não da forma mais eficiente)|| N|| N|| N||N
 
  |-
 
  |-
  |Roberto || D*
+
  |Leonardo || D||P (condições trocadas) || N|| N|| P (não indica o vencedor)||N
 
  |-
 
  |-
|Thiago Martins || D*
+
|}
|-
 
|Thiago Ramos || D*
 
|-
 
|Vinicius Kachniacz || D*
 
|-
 
|Vinicius Rodrigues || D
 
|-
 
|}
 
  
''Obs: D* = não apresentou o projeto (desistente)''
+
== 26/05: Começo do projeto ==
  
== 20/12: Recuperação ==
+
Para referência: [http://www.ead.cpdee.ufmg.br/cursos/C/c.html apostila online sobre linguagem C].
 
 
A prova de recuperação terá ao menos quatro questões sobre programação em linguagem C a serem resolvidas em laboratório. Essa prova terá que ser resolvida iniciando às 7:30 e terminando no máximo às 10:30 h. Essa prova será a mesma para quem precisar recuperar Lógica de Programação.
 
 
 
*[http://www.sj.ifsc.edu.br/~msobral/SOP/avaliacoes/rec-2010-2.pdf Prova de recuperação]
 

Edição atual tal como às 20h45min de 6 de julho de 2011

Introdução à Programação - 2011-1

Informações gerais

Professor: Marcelo Maia Sobral
Email: msobral@gmail.com
Skype: mm-sobral
Lista de email (forum): prg-ifsc@googlegroups.com
Atendimento paralelo: 4a feira 10:00 - 11:30 h, 5a feira 8:00 - 9:30 h e 13:30 - 15:30 h (no Laboratório de Desenvolvimento de Tele)
Reavaliação (recuperação): no final do semestre
IMPORTANTE: o direito de recuperar uma avaliação em que se faltou somente existe mediante justificativa reconhecida pela coordenação. Assim, deve-se protocolar a justificativa no prazo de 48 horas, contando da data e horário da avaliação, e aguardar o parecer da coordenação. O não cumprimento desse procedimento implica a impossibilidade de fazer a recuperação, e assim a reprovação na disciplina.

Softwares

Será usado como plataforma de estudo o sistema operacional Ubuntu Linux 10.04 LTS ou posterior. Para obtê-lo há essas opções:

  1. Trazer um CD-R virgem para que eu faça a cópia aqui no IFSC
  2. Fazer o download por conta própria (aprox. 700 MB)
  3. Usar uma máquina virtual do VirtualBox já preparada por mim (menos recomendado, pois o Linux roda mais lento)
    • Trazer um DVD-R ou pendrive com ao menos 4 GB livres.
    • Instalar o VirtualBox em seu computador para executar a máquina virtual

ATENÇÃO: é muito importante que se providencie o quanto antes a instalação do Ubuntu em seu computador. Sem ele você não poderá fazer os exercícios sugeridos, o que atrapalhará seu aproveitamento na disciplina ! O bom andamento do estudo depende muito de não deixar acumular o conteúdo e os exercícios. Acostume a criar uma rotina de estudo, procurando resolver os problemas apresentados e procurando o professor (ou contatando-o por email) para tirar dúvidas.

Listas de exercícios

Referências adicionais

Aulas

16/03: Introdução

Apresentação da disciplina: plano de ensino, avaliação, abordagem pedagógica.

Assunto: Lógica de programação

Referências auxiliares:


Um primeiro algoritmo


Videos sobre algoritmos:

Problemas exemplo

Problema dos três recipientes

Há três recipientes com tamanhos distintos: um com 8 litros, outro com 5 litros e o terceiro com 3 litros. O recipiente com 8 litros está completamente cheio. Deseja-se colocar 4 litros em dois recipientes. Considere que os recipientes não são graduados.

Problema da travessia

Um barqueiro precisa levar um saco de milho, uma galinha e uma raposa para o outro lado do rio. Porém o barco somente é capaz de levar uma coisa de cada vez (além do barqueiro). Qual a sequência de travessias necessário para atravessar o milho, a galinha e a raposa ?

Torres de Hanoi

Há três hastes. Uma das hastes serve de suporte para três discos de tamanhos diferentes. Um disco menor sempre é colocado sobre um disco maior. A figura abaixo ilustra as hastes e os discos:

Hanoi.png

Desejam-se mover os três discos para a haste da direita. Porém só pode se mover um disco por vez, e um disco maior nunca pode ficar sobre um disco menor.

Operação possível: Move disco para haste

Qual a sequência de operações para mover os discos de uma haste para outra ?

Leitura adicional

17/03: Algoritmos

Para criar algoritmos, deve-se primeiro entender o que são instruções e sequências lógicas;

O jogo LightBot mostra de uma forma divertida como criar pequenos algoritmos. Nesse jogo o robô deve se movimentar de acordo com uma sequência de instruções elementares definidas pelo jogador. Apesar de sua simplicidade, ele pode se tornar bastante desafiador ! Até que fase desse jogo você consegue chegar ?

Lightbot.png

Desenhando figuras geométricas

  1. Scratch é um software educacional para ajudar no ensino de matemática, geometria e introdução à programação. Com ele pequenos programas podem ser escritos de foram visual, com instruções representadas por blocos que se encaixam como Lego. Ele possibilita fazer desenhos facilmente, seguindo um programa com instruções de desenho. Use-o para desenhar estas figuras:
    • triângulo equilátero
    • triângulo isósceles
    • quadrado
    • hexágono
    • quadrado com vértices interligados
    • Um círculo

Scratch.png


  • Agora tente algo mais avançado:
    • Desenhe um triângulo equilátero, porém perguntando na tela qual o tamanho do lado.
    • repita o desenho do triângulo, porém fazendo-o lentamente: espere 1 segundo após desenhar cada lado do triângulo.
    • Faça com que um pequeno traço se movimente de um lado a outro do palco.
    • Faça com que um pequeno traço se movimente em círculo indefinidamente.
    • Faça com que uma bola role pela horizontal. Insira quatro raios na bola para ajudar no efeito visual de movimento.

Atividade para casa

Faça as figuras geométricas indicadas na aula !

23/03: Começando os primeiros algoritmos ...

Algoritmo é uma sequência lógica de instruções, a qual possui um início e um fim.

Sequência lógica é uma sequência correta de instruções capaz de gerar o resultado esperado.


A escrita ou projeto de algoritmos se faz com:

  • Estruturas de controle: comandos que modificam uma sequência de instruções
  • Estruturas de dados: dados (valores) usados e modificados durante a execução do algoritmo.


Para introduzir esses conceitos, nada melhor que um pequeno projeto ...


Projeto: um jogo muito simples

Você conhece o jogo clássico Space Invaders ?

Spaceinvaders.jpg


Pros padrões atuais esse é um jogo muito simples. Você conseguiria reproduzi-lo, mesmo que simplificado, no Scratch ? Ele poderia começar assim:

Jogo1-scratch.gif


  1. O robô se movimenta na horizontal, comandado pelas teclas seta pra direita e seta pra esquerda.
  2. O helicóptero se movimenta de um lado a outro constantemente, refletindo nas bordas.
  3. Ao teclar espaço, um tiro sai do robô em direção ao topo do palco
  4. Se o tiro acertar o helicóptero, o jogo termina.


O que você precisa para fazer esse jogo ?

  1. Movimento do helicóptero.
  2. Movimento do robô comandado pelo teclado.
  3. Acionamento do tiro.
  4. Detecção se o tirou acertou o helicóptero.

Mãos à obra !

24/03: Introduzindo algoritmos: variáveis e expressões

Variáveis são usadas em algoritmos para guardar dados. Dados podem ser:

  • Números inteiros ou reais
  • Letras ou textos
  • Valores lógicos (booleanos)
  • ... e outros que vocês descobrirão futuramente !

Uma variável possui um identificador, que nada mais é que seu nome. No Scratch as variáveis são criadas e podem ser modificadas com os comandos categorizados em Variáveis (comandos laranjas):


Scratch-var1.png


Variáveis podem ser globais ou locais:

  • Globais: são visíveis e podem ser usadas por qualquer objeto do Scratch.
  • Locais: são visíveis e podem ser usadas somente pelo objeto do Scratch ao qual está vinculada.

Variáveis podem ser usadas em qualquer comando que precise de um valor numérico, como por exemplo o comando para mover um certo número de passos. No exemplo abaixo, um programa faz com que um objeto desenhe um triângulo no palco, sendo que o comprimento do lado do triângulo é informado pelo usuário.

Scratch-var2.png


Variáveis podem ser usadas em expressões aritméticas, que realizam algum cálculo com valores numéricos. No Scratch expressões são criadas combinando-se os blocos contidos em Operadores(blocos verdes-claros):


Scratch-operadores1.png


Usando esses blocos podem-se calcular novos valores usando variáveis ou constantes. Por exemplo, um pequeno programa que desenha um polígono qualquer precisa calcular o ângulo de quina da figura dependendo de sua quantidade de lados:


Scratch-operadores2.png


Usando esses conceitos sobre variáveis e expressões, faça as atividades a seguir.

Pong

Outro jogo clássico chamado Pong mostrava uma bola que refletia em uma parede, voltando e devendo ser interceptada por uma pequena base (que funcionava como raquete) comandada pelo jogador. Se conseguir interceptá-la, a bola é refletida de novo em direção à parede oposta, continuando assim o jogo. Se falhar, o jogo termina. Esse é considerado o primeiro videogame lucrativo a ter sido criado ...

Pong1.jpg


Tente fazer uma versão do Pong usando o Scratch, mas com duas pequenas modificações. Há apenas uma base, que fica no canto inferior do palco. O placar conta quantas vezes a bola tocou o chão (i.e. o jogador falhou em interceptá-la), e se chegar a 0 o jogo termina. A segunda modificação é uma aceleração gradual da bola, que deve deve aumentar de velocidade a cada 5 segundos de jogo.


Para que serão necessárias variáveis nesse jogo ?

O problema da reflexão da bola na barra

Em aula vimos que para fazer a reflexão da bola com a barra deve-se calcular o ângulo de reflexão. No caso da barra estar na horizontal, esse ângulo deve ser calculado assim:

No Scratch isso deve ser feito combinando-se o comando "Aponte para direção" e uma expressão para calcular a nova direção. A direção atual do objeto está contida em sua variável local direção (predefinida nos comandos de Movimentos):


Scratch-reflexao.png


Uma outra versão do Pong

Em uma outra versão ainda encontrada, no canto superior há um conjunto de alvos que devem ser acertados pela bola. Quando se acertarem todos os alvos o jogo termina.


Pong2.jpg

Para casa: um jogo de sinuca

Para exercitar o uso de variáveis vamos criar também um jogo de sinuca. Nesse jogo deve-se usar a bola branca para encaçapar uma outra bola colorida. As bolas se movimentam de forma natural, desacelerando gradualmente devido ao atrito com a mesa. Quando as bolas colidem, suas velocidades e direções se modificam, dependendo de seus valores anteriores e do ângulo de colisão. Quando as bolas batem nos cantos das mesas, devem ser refletidas. Como as bolas predem velocidade gradualmente, dependendo de suas velocidades iniciais e das colisões em algum momento irão fatalmente parar. As caçapas ficam nos dois cantos superiores do palco. Você notará que sem o uso de variáveis, não será possível criar esse jogo, ainda mais se quiser acrescentar mais alguns detalhes.

As tacadas se fazem com um taco representado por uma linha reta, cuja ponta sempre está colada à bola branca. A direção do taco pode ser controlada pelo mouse ou por teclas direita e esquerda. Para fazer uma tacada, deve-se pressionar a tecla seta pra baixo durante um tempo, e então soltá-la. Quanto mais tempo se pressionar essa tecla mais forte será a tacada, e assim mais veloz a bola branca vai disparar. Assim, para realizar tacadas com pouca velocidade deve-se pressionar a tecla seta pra baixo por pouco tempo, e para tacadas mais fortes, em que a bola branca dispara com grande velocidade, deve-se pressionar essa tecla por um tempo maior. Pode-se usar algum objeto e colocá-lo na lateral do palco para informar qual a potência da tacada a ser realizada.

Alguns dados serão necessários para calcular as movimentações das bolas:

  • Peso de cada bola: 100 g
  • Diâmetro das bolas: 8 cm
  • Desaceleração das bolas devido ao atrito com a mesa:
  • Velocidade mínima de tacada: 0 m/s
  • Velocidade máxima de tacada: 4 m/s

30/03: Algoritmos: Usando variáveis, expressões e estruturas de decisão

Estruturas de decisão: blocos de controle usados quando devem-se executar comandos somente se uma determinada condição for verdadeira. Por exemplo, no jogo Pong a bola deve ser refletida se tocar na barra controlada pelo jogador:

Prg-se3.png

Esse tipo de estrutura de decisão se chama Se condição então faça algo. Como indica sua descrição, uma ou mais instruções serão executadas caso a condição seja verdadeira. No exemplo acima, a condição é tocando em barra, e as instruções executados caso isso seja verdade são aponte para a direção 180-direção.


No exemplo do tiro ao helicóptero, inspirado no jogo Space invaders, o jogo deveria terminar quando o tiro acertasse o alvo. Assim, as instruções para término do jogo precisavam estar dentro de uma estrutura Se condição então ..., como mostrado abaixo:


Prg-se1.png


Para ilustrar o uso de Se condição então ..., e o teste de condições em geral, hoje serão feitos dois projetos:

  1. A continuação do jogo Pong, em que a bola deve acertar alvos no topo da tela. Além disso, se a bola tocar no chão mais que 3 vezes o jogador perde.
  2. Um cronômetro a ser visualizado no palco.

31/03: Algoritmos: ainda estruturas de decisão e também de repetição

Por diversas vezes criaram-se programas no Scratch em que o comportamento de um objeto corresponde a repetir uma sequência de comandos. Por exemplo, cada alvo no topo da tela no jogo Pong deve sumir quando for tocado pela bola:

Prg-repeticao1.gif

Assim, ele deve sempre testar se a bola o está tocando, e quando isso acontecer ele deve desaparecer e contabilizar que há uma alvo a menos para ser acertado. Como esse teste deve ser feito repetidamente até que a bola o acerte, torna-se necessário usar uma estrutura de repetição. No caso do Scratch, usou-se o comando Sempre, que faz com que todos os comandos nele contidos sejam executados indefinidamente (no caso, até que a bola toque o alvo). Além dessa parte do programa, o teste se todos os alvos foram acertados também implica a repetição de uma sequência de comandos.

No jogo, a bola deve se movimentar pelo palco, refletindo nas paredes e sendo rebatida pela barra até que todos os alvos tenham sido acertados. Portanto, os comandos que implicam controlar seu movimento devem se repetir seguidamente até que essa condição de término seja alcançada. O comportamento da bola pode desta forma ser descrito simplificadamente pelos comandos mostrados abaixo, em que se usa a variável bananas para indicar quantos alvos faltam para serem acertados. A repetição foi obtida com o comando repita até bananas=0, e dessa forma a repetição se interrompe quando a variável bananas tiver o valor 0.

Prg-repeticao2.gif

Um terceiro exemplo diz respeito à criação de um cronômetro. Uma variável pode ser decrementada uma vez por segundo, até que o cronômetro zere. Um exemplo de como fazer isso segue abaixo:

Prg-repeticao4.png


Como se vê, estruturas de repetição são muito úteis para repetir sequências de comandos. Existem estruturas de repetição incondicionais, que repetem indefinidamente (como o comando sempre do Scratch), e condicionais, que repetem enquanto determinada condição for verdadeira ou até que determinada condição se verifique (ex: comando repita até). Com elas se evita que programas sejam desnecessariamente longos ... na verdade sem elas seriam inviáveis muitos tipos de programas !

Atividade

  1. Adicione o cronômetro na tela do seu jogo Pong. O cronômetro deve ser composto de dois dígitos.
  2. O Pong poderia ficar mais difícil se os alvos fossem móveis e fugissem da bola. Assim, quando sentisse a bola se aproximando um alvo poderia acelerar ou reduzir sua velocidade, visando evitar uma colisão. Modifique então o Pong para que os alvos movam na horizontal de uma lado para o outro, e evitem a colisão com a bola.

06/04: Portugol

Trabalho: deve-se entregar o jogo Pong com o cronômetro, conforme especificado na última aula sobre Scratch.

'Prazo de entrega: 14/04/2011


Os algoritmos que iremos criar serão compostos de:

  • Estruturas de controle: estruturas para controlar a sequência de execução (realização) de algoritmos:
    • Decisão: para realizar testes de forma a decidir que sequências de instruções devem ser executadas
    • Repetição: para repetir sequências de instruções
  • Estruturas de dados: estruturas para representar os dados a serem processados por algoritmos
    • Variáveis e constantes

Portugol

As aulas de Lógica de Programação usarão um software de auxílio ao ensino de algoritmos chamado Portugol, desenvolvido na Escola Superior de Engenharia do Instituto Politécnico de Tomar, em Portugal.

Guia rápido de instalação e utilização do Portugol

Abaixo segue uma breve ajuda de como obtê-lo, instalá-lo e usá-lo. Esse guia assume que você esteja usando o Ubuntu Linux 9.04 ou superior.

  1. Faça o download do Portugol.
  2. Descompacte-o com o seguinte comando:
    tar xzf portugol23.tar.gz
    
  3. Repare que existe agora um subdiretório portugol no diretório onde você o descompactou. Execute o Portugol com o seguinte comando:
    java -jar portugol/Portugol.jar
    
    Obs: você precisará ter Java instalado. Caso não o tenha, execute o comando:
    sudo apt-get install openjdk-6-jre
    
  4. Copie esse arquivo para poder ver fluxogramas coloridos, e grave-o no memso diretório onde está o Portugol.
  5. Veja a ajuda do Portugol, e use-a sempre que tiver dúvidas !


A tela inicial do Portugol segue abaixo, junto com um programa demonstrativo.

Editor-Portugol.png

Exemplos de programas iniciais em Portugol:

  1. Lendo um número e mostrando-o na tela em seguida:
    Inicio
      inteiro x
    
      Escrever "Digite um numero: ",
      Ler X
      Escrever "Numero digitado: ", x
    Fim
    
  2. Lendo dois números, somando-os e mostrando o resultado na tela:
    Inicio
      inteiro x, y
    
      Escrever "Digite um numero: ",
      Ler x
      Escrever "Digite outro numero: ",
      Ler y
      Escrever "Soma = ", x+y
    Fim
    
    O programa abaixo é equivalente:
    Inicio
      inteiro x, y, z
    
      Escrever "Digite um numero: ",
      Ler x
      Escrever "Digite outro numero: ",
      Ler y
      z <- x + y
      Escrever "Soma = ", z
    Fim
    

Atividades

  1. Média de três números: escreva um programa para calcular a média de três números.
  2. Média balanceada de três números: escreva um programa para calcular a média balanceada de três números.
  3. Sequência de Fibonacci: em matemática corresponde aos números:
    1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, ...
    
    ... que pode ser descrita pela relação de recorrência



    ... com valores iniciais e .

    Numerosas formas na natureza apresentam essa sequência, como neste girassol (cujas flores se dispõem em uma espiral):

    Sunflower.jpg Espiral fibonacci.png FibonacciBlocks.png

    Usando o Portugol escreva um programa que mostre os 10 primeiros números dessa sequência.
  4. Distância até o horizonte: escreva um programa que calcule a distância dos seus olhos até o horizonte. Assuma que a Terra é perfeitamente esférica, e que seu raio tem 6.378 km. Considere que você esteja no nível do mar (seus pés tocando a água do mar ;-), e que o horizonte esteja num mar perfeitamente liso. Dica: faça um diagrama desse problema para visualizar sua geometria, e use trigonometria para resolvê-lo.

13/04: Algoritmos

Variáveis e constantes

  • Variável: capaz de guardar um dado a ser usado no algoritmo. Pode ser entendida como uma caixa, onde se coloca um dado e se pode consultá-lo quantas vezes for necessário. O dado pode ser modificado (substituído por outro). Exemplo em Portugol:
    Inicio
      inteiro anos, dias
    
      Escrever "Quantos anos se passaram ? "
      Ler anos
      dias <- anos * 365
      Escrever "... então se passaram ", dias, " dias"
    Fim
    
    Nesse exemplo há duas variáveis: dias e anos
  • Constante: semelhante à variável, porém o dado armazenado não pode ser modificado. Exemplo em Portugol:
    Inicio
      constante inteiro diasPorAno <- 365
      inteiro anos, dias
    
      Escrever "Quantos anos se passaram ? "
      Ler anos
      dias <- anos * diasPorAno
      Escrever "... então se passaram ", dias, " dias"
    Fim
    
    Nesse exemplo há uma constante: diasPorAno


Variáveis e constantes devem ser declaradas antes de serem usadas (algumas poucas linguagens, como Python e Perl, não exigem isto). A declaração consiste do tipo e identificador da variável. O tipo corresponde ao tipo de valor que pode ser guardado, e o identificador é o nome da variável. No exemplo abaixo:

  constante inteiro diasPorAno <- 365
  inteiro anos, dias
Fim

No exemplo acima, há duas variáveis do tipo inteiro, e seus identificadores são dias e anos. O tipo inteiro indica que essas variáveis podem guardar somente números inteiros.

Tipos de variáveis e constantes no Portugol:

Tipo Descrição Exemplo
Inteiro Número inteiro entre -2 147 483 648 e 2 147 483 647 Inteiro x <- 10
Real Número real entre -1.7 E 308 e 1.7 E 308 Real y <- 10.789
Lógico Valor booleano, com valores "Verdadeiro" e "Falso" Logico ok <- Falso
Caracter Um caractere da tabela ASCII Caracter letra <- "A"
Texto Uma sequência de caracteres (ou string) Texto palavra <- "um teste"

A declaração de constantes é semelhante à de variáveis, bastanto prefixá-las com a palavra-chave constante.

Expressões aritméticas

Um conjunto de operações sobre variáveis, constantes e funções numéricas, e que gera um determinado resultado numérico.

Exemplos de expressões aritméticas:

# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
3600*horas + 60*minutos + segundos

# Uma expressão que calcula a velocidade instantânea, segundo um MRV
vel_inicial + aceleracao*tempo;

# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
raiz(x^2 + y^2)

Os resultados de expressões podem ser mostrados na tela, ou armazenados em variáveis:

# Uma expressão que calcula quantos segundos existem em um horário do tipo horas, minutos e segundos
segundos <- 3600*horas + 60*minutos + segundos

# Uma expressão que calcula a velocidade instantânea, segundo um MRV
escrever 'Velocidade no instante ', tempo, ' = ', vel_inicial + aceleracao*tempo;

# Uma expressão que calcula o módulo de um vetor bidimensional, que possui coordenadas x e y
modulo <- raiz(x^2 + y^2)

Repare que uma expressão fica sempre do lado direito, quando atribuída a uma variável. A expressão é primeiro calculada, e em seguida seu resultado é armazenado na variável:

segundos <- 3600*horas + 60*minutos + segundos

Operadores aritméticos

Expressões aritméticas sao compostas por números e operadores aritméticos:

Obs: para os exemplos abaixo são usadas estas variáveis:

Real x, area, lado
inteiro dias, horas
Operador Descrição Exemplo
+ Adição x <- x + 1
- Subtração x <- x - 1
* Multiplicação x <- x*x*x
/ Divisão dias <- horas / 24
% Módulo (resto de divisão) horas <- horas % 24
^ Potenciação area <- lado^2

Precedência dos operadores (nesta ordem): ^, *, /, %, + e -

A precedência pode ser modificada com o uso de parênteses. Ex:

escrever 1 + 2 * 3
escrever (1 + 2)*3

No Portugol, existem também algumas funções úteis, como a função raiz:

r <- raiz(x^2 + y^2)

O resultado de expressões aritméticas depende dos tipos numéricos das variáveis e constantes:

inicio
  real x
  inteiro y
  inteiro resultadoInteiro
  real resultadoReal
 
  x <- 9
  y <- 9
 
  escrever "O resultado de uma expressão aritmética depende dos tipos das variáveis e constantes\n"
  escrever "usadas na expressão. Se forem todas inteiras, então o resultado será inteiro.\n"
  escrever "Veja este exemplo: \n"
  escrever "Variável inteira y=", y
  escrever "\nExpressão: y/2=", y/2
 
  escrever "\n\nNeste segundo exemplo, repete-se a mesma expressão, porém usando-se uma\n"
  escrever "variável real:\n"
  escrever "Variável real x=", x
  escrever "\nExpressão: x/2=", x/2
 
  x <- 4
  y <- 5
  escrever "\n\nSe as variáveis de diferentes tipos forem combinadas, o resultado da\n"
  escrever "expressão será real:\n"
  escrever "Variável real x=", x, " e inteira y=", y
  escrever "\nExpressão: (x+y)/2=", (x+y)/2
 
  escrever "\n\nNo entanto, se uma expressão tiver um resultado real, mas este for\n"
  escrever "atribuído a uma variável inteira, então apenas a parte inteira será guardada:\n"
  escrever "Variável real x=", x, " e inteira y=", y
  y <- (x+y)/2
  escrever "\nExpressão: y <- (x+y)/2 ... após executada, y=", y
 
fim

Atividade

Para os exercícios abaixo, escreva o algoritmo no Portugol.

  1. Faça um algoritmo que, usando somente duas variáveis, calcule a média de quatro números.
  2. Escreva um algoritmo que calcule as raizes de uma equação de 2o grau.
  3. Um equipamento conta o tempo desde que foi ligado. No entanto, essa contagem é feita em segundos. Faça um algoritmo que converta o valor desse contador para horas, minutos e segundos. Ex: se o contador tiver o valor 43397, o seu programa deve mostrar: 12:03:17.
    inicio
      inteiro h,m,s
    
      escrever "Quantos segundos mostrou o equipamento ? "
      ler s
      
      h <- s /3600
      s <- s % 3600   // s <- s - h*3600
      m <- s /60
      s <- s % 60 // s <- s - m*60
      
      escrever h, ":", m, ":", s
      
    fim
    

14/04: Variáveis e expressões

Continuação ...

Exercícios

  1. Crie um conversor de decimal para binário (limite: 4 bits). Ex.: tendo o número 10 (decimal) de entrada, deve-se obter o número 1010 (binário) de saída.
  2. Escreva um algoritmo que leia o nome, sobrenome e idade de uma pessoa, e escreva na tela (no lugar dos colchetes devem aparecer os valores lidos):
     Nome: [sobrenome, nome]
     Idade: [idade] anos
    
  3. A função exponencial pode ser calculada por uma série de Taylor:

    Faça um algoritmo para calcular o valor desta função.

16/04: Estruturas de decisão

O exemplo abaixo está contido no livro Classification and Regression Trees, de Leo Breiman, Jerome Friedman, Richard Olshen e Charles Stone, editora Chapman & Hall, 1984.

Na Universidade da California, no centro Médico de San Diego, quando um paciente com ataque cardíaco é recebido, dezenove variáveis são medidas ao longo das primeiras 24 horas. Essas incluem pressão sanguínea, idade, e 17 outras variáveis ordenadas e binárias (booleanas) que sumarizam os sintomas médicos considerados importantes indicadores da condição do paciente.

O objetivo de um estudo médico feito entre os anos 70 e 80 foi o desenvolvimento de um método para identificar pacientes de alto risco (que não sobreviverão ao menos 30 dias) com base nos dados obtidos nas primeiras 24 horas. O diagrama abaixo mostra uma regra de classificação que foi produzida nesse estudo. A letra F significa que não há um risco alto, e a letra G quer dizer paciente de alto risco.

Crt-decisao.png

Essa regra classifica os pacientes como F ou G dependendo de respostas do tipo sim/não a no máximo três perguntas. Como seria um algoritmo que a implementasse ?


Algoritmos para resolver problemas como o exemplo acima dependem de se poder expressar a tomada de decisões. Deve ser possível testar condições, como por exemplo "a idade é maior que 62.5 anos ?", e executar instruções dependendo do resultado do teste. Em Lógica de Programação, e em linguagens de programação em geral, isso se faz com estruturas de decisão.

Estruturas de decisão possibilitam que se executem diferentes sequências de instruções de um programa, dependendo de uma condição a ser avaliada. Por exemplo, um jogo poderia armazenar a maior pontuação já obtida, e verificar se foi ultrapassada ao final de cada partida:

  Se pontuacao > recorde então 
    recorde <- pontuação
  FimSe

O exemplo em Portugol acima mostra a estrutura de decisão Se condição entao comandos Fimse. O equivalente em fluxograma é:

Seentao.png

Veja o próximo exemplo:

  Se conceito > 70 entao
    escrever 'Voce esta aprovado'
  senao
    escrever 'Desta vez não deu ... tente de novo !'
  Fimse

O uso de Se condição entao comandos Senao comandos Fimse possibilita que se execute uma sequência de comandos se a condição for verdadeira, e outra sequência se for falsa. Seu equivalente em fluxograma é:

Seentaosenao.png

Para fazer um bom uso de estruturas de decisão deve-se primeiro conseguir identificar as condições a serem avaliadas. Condições são escritas com expressões lógicas, e estas são compostas de operadores lógicos e relacionais aplicados a variáveis e constantes.

Condições

Condições representam os testes que precisam ser feitos com dados do problema. Esses testes são expressados por expressões lógicas, cujos resultados são verdadeiro ou falso. Expressões lógicas são construídas usando-se operadores lógicos e operadores relacionais para comparar e avaliar os valores de variáveis e constantes.

Obs: para os exemplos abaixo são usadas estas variáveis:

Logico correto, multa, aprovado, barato, bombear_agua
Logico descartar, baixo, reprovado, erro, enviado, recebido
inteiro erros, pontuacao, preco, endereco, velocidade
Real faltas, nivel_agua, altura

Operadores relacionais

Operadores relacionais são usados para comparar valores, e estão listados na tabela abaixo. Cada comparação feita com esses operadores forma uma expressão lógica simples. Portanto, o resultado de uma comparação é um valor lógico que pode ser verdadeiro ou falso.

Operador Descrição Exemplo
= Igualdade correto <- (erros = 0)
> Maior multa <- (velocidade > 80)
>= Maior ou igual aprovado <- (pontuacao >= 70)
< Menor barato <- (preco < 100)
<= Menor ou igual bombear_agua <- (nivel_agua <= 0.7)
=/= Diferente descartar <- (endereco =/= 12345)

Operadores lógicos

Operadores lógicos são usados para combinar expressões lógicas simples, criando assim expressões lógicas compostas, e estão listados na tabela abaixo.

Operador Descrição Exemplo
NAO Negação baixo <- NAO (altura > 1.8)
E Conjunção aprovado <- NAO (conceito = "D") E (faltas <= 0.25)
OU Disjunção reprovado <- (conceito = "D") OU (faltas > 0.25)
XOU Disjunção exclusiva erro <- enviado XOU recebido

Precedência dos operadores (nesta ordem): NAO, E, OU e XOU

Lembre que a precedência pode ser modificada com o uso de parênteses.

Atividades

  1. Faça um programa que leia um número e então informe se ele é par ou ímpar.
  2. Um radar de trânsito faz a medição de velocidade de veículos e, dependendo do valor, calcula a multa a ser aplicada. Em uma determinada via esse radar foi configurado da seguinte forma:
    • Se a velocidade for maior que 80 km/h, a multa é de R$ 360.
    • Se a velocidade for maior que 60 km/h, a multa é de R$ 180.
    • Se a velocidade for menor ou igual a 60 km/h, não há multa.

      Escreva um algoritmo que calcule a multa de acordo com a velocidade de um veículo.
  3. Faça um algoritmo que leia três números do teclado, e mostre o maior e menor números.
  4. Escreva um algoritmo que faça o teste de risco de paciente cardíaco, mostrado no início da aula.
  5. Um estudo sobre sensibilidade de pessoas a temperaturas da água identificou que a maioria das pessoas considera fria a água com temperaturas abaixo de 25 graus, morna entre 25 e 30 graus, e quente acima de 30 graus. Escreva um algoritmo que mostre as palavras "fria", "morna" ou "quente" dependendo da temperatura da água que for informada.
  6. Em uma rede de computadores, o firewall restringe o acesso a Internet dependendo do horário e do tipo de usuário que faz o acesso. Os tipos de usuário abaixo têm as seguintes restrições:
    • Funcionario: apenas entre 0:00 e 7:30, entre 18:30 e 0:00, e entre 12:00 e 13:30
    • Financeiro: qualquer horario
    • Diretoria: qualquer horário
      Escreva um algoritmo que informe se um acesso foi permitido, em função do horário e do tipo de usuário.
  7. Modifique o algoritmo acima para adicionar a seguinte restrição:
    • Funcionario: nao pode acessar sites de jornal (ex: www.rbs.com.br)
    • Financeiro: nao pode acessar sites de jornal durante o expediente
    • Diretoria: sem restrições a sites
  8. Faça um algoritmo para fazer a divisão de dois números reais. Antes de dividi-los deve ser feito um teste de validade. Caso não seja possível dividi-los, deve ser mostrada uma mensagem de erro. Se for possível, deve-se mostrar o resultado da divisão.
  9. Faça um jogo de par ou ímpar, em que o jogador aposta contra o computador. O jogador deve digitar um número entre 0 e 5 e optar entre par ou ímpar. O computador deve sortear um número também entre 0 e 5. Se a paridade da soma dos números do jogador e do computador for a mesma que o jogador optou, então ele ganha a partida, senão o computador vence.
  10. Escreva um algoritmo para identificar se há, dentre três palavras lidas do teclado, ao menos duas palavras distintas que pertençam ao conjunto {azul, preto, vermelho}. Exemplos:
    • Se o usuário digitar verde, preto, vermelho, o programa deve mostrar na tela verdadeiro
    • Se o usuário digitar verde, preto, preto, o programa deve mostrar na tela falso
    • Se o usuário digitar azul, preto, azul, o programa deve mostrar na tela verdadeiro
inicio
  texto cor1, cor2, cor3
  inteiro cont <- 0
    
  ler cor1
  ler cor2
  ler cor3
  
  se cor1 = "preto" ou cor2 = "preto" ou cor3 = "preto" entao
    cont <- cont + 1
  fimse

  se cor1 = "azul" ou cor2 = "azul" ou cor3 = "azul" entao
    cont <- cont + 1
  fimse

  se cor1 = "vermelho" ou cor2 = "vermelho" ou cor3 = "vermelho" entao
    cont <- cont + 1
  fimse
        
  se cont > 1 entao
    escrever "verdadeiro"
  senao
    escrever "falso"
  fimse
fim

Fluxogramas

Diagramas de bloco para auxiliar a descrição de algoritmos. Ajudam na compreensão do algoritmo, por poder visualizar o fluxo de execução.


Fluxograma-soma.png Fluxograma para o algoritmo da média de trẽs números.


Blocos de uso mais comum

Bloco Descrição Exemplo
Inicio.png Inicio do fluxograma
Processamento.png Processamento
Entrada.png Entrada de dados (ler do teclado)
Saida.png Saída de dados (mostrar na tela)
Decisao.png Decisão (testar uma condição e bifurcar) Ex-decisao.png
Conector.png Conector (juntar dos ou mais ramos do fluxograma) Ex-conector.png
Fim.png Fim

Obs: Arquivo de configuração das cores do fluxograma do Portugol.

20/04: estruturas de decisão

Há situações em que se precisa fazer um conjunto de comparações, como mostrado abaixo:

// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
// dia, nome do mês, ano.
Inicio
  inteiro dia, mes, ano
  texto nome_mes

  escrever "Dia: "
  ler dia
  escrever "Mes: "
  ler mes
  escrever "Ano: "
  ler ano

  se mes = 1 entao
    nome_mes <- "Janeiro"
  senao
    se mes = 2 entao
      nome_mes <- "Fevereiro"
    senao
      se mes = 3 entao
        nome_mes <- "Março"
      senao
        se mes = 4 entao
          nome_mes <- "Abril"
        senao
          se mes = 5 entao
            nome_mes <- "Maio"
          senao
            se mes = 6 entao
              nome_mes <- "Junho"
            senao
              se mes = 7 entao
                nome_mes <- "Julho"
              senao
                se mes = 8 entao
                  nome_mes <- "Agosto"
               senao
                  se mes = 9 entao
                    nome_mes <- "Setembro"
                  senao
                    se mes = 10 entao
                      nome_mes <- "Outubro"
                    senao
                      se mes = 11 entao
                        nome_mes <- "Novembro"
                      senao
                        nome_mes <- "Dezembro"
                      fimSe
                    fimSe
                  fimSe
                fimSe
              fimSe
            fimSe
          fimSe
        fimSe
      fimSe
    fimSe
  fimSe

  escrever dia, " de ", nome_mes, " de ", ano
fim

Além de ser trabalhoso esse encadeamento de Se ... entao ... senao, o algoritmo resultante fica pouco legível. Quer dizer, ele fica feio e difícil de entender.

Existe uma estrutura de decisão criada justamente para casos como esse, e que resulta em um algoritmo mais limpo e compreensível:

// Lê a data no formato numérico dia, mes, ano, e mostra a data no formato
// dia, nome do mês, ano.
Inicio
  inteiro dia, mes, ano
  texto nome_mes

  escrever "Dia: "
  ler dia
  escrever "Mes: "
  ler mes
  escrever "Ano: "
  ler ano

  Escolhe mes
    caso 1: 
      nome_mes <- "Janeiro"
    caso 2: 
      nome_mes <- "Fevereiro"
    caso 3: 
      nome_mes <- "Março"
    caso 4: 
      nome_mes <- "Abril"
    caso 5: 
      nome_mes <- "Maio"
    caso 6: 
      nome_mes <- "Junho"
    caso 7: 
      nome_mes <- "Julho"
    caso 8: 
      nome_mes <- "Agosto"
    caso 9: 
      nome_mes <- "Setembro"
    caso 10: 
      nome_mes <- "Outubro"
    caso 11: 
      nome_mes <- "Novembro"
    Defeito: 
      nome_mes <- "Dezembro"
  fimEscolhe

  escrever dia, " de ", nome_mes, " de ", ano
fim

A estrutura de decisão escolhe ... caso tem uma certa flexibilidade. No exemplo abaixo, mostra-se a possibilidade de testar mais de um valor no mesmo caso:

inicio     
  caracter sexo     

  escrever "Qual o seu sexo (f/m):"     
  ler sexo     

  escrever "Olá "     

  escolhe sexo         
    caso "m", "M" :             
      escrever "senhor"         
    caso "f","F" :             
      escrever "senhorita"         
    defeito :             
      escrever "Sexo indefinido"     
  fimescolhe     

  escrever ", benvindo ao portugol" 
fim

Atividades

  1. Faça um algoritmo que converta um número de 1 a 7 para o respectivo dia da semana (ex: 1 = domingo, 2 = 2a feira, e assim por diante).
  2. Faça uma calculadora com as quatro operações aritméticas. Sua calculadora deve ler (nesta ordem) o primeiro número, a operação aritmética (que deve ser informada usando o símbolo da respectiva operação: +, -, * ou /), e depois o segundo número. Ao final, seu algoritmo deve mostrar o resultado, ou uma mensagem de erro se a operação não for possível de realizar (ex: divisão por zero).
  3. A previsão do tempo na costa brasileira pode ser feita de forma aproximada usando-se um barômetro e um termômetro. Uma estimativa com boa chance de acerto se baseia na tabela abaixo:
    Previsao-barometro.png

    Faça um algoritmo que forneça uma estimativa da previsão do tempo usando essa tabela.
  4. Faça um algoritmo que mostre qual o último dia de um determinado mês informado pelo teclado. Caso seja informado o mês 2 (fevereiro), seu algoritmo deve identificar se é ano bissexto (assim o mês tem 29 dias), ou não (mês tem 28 dias). Obs: anos bissextos são dados pelas regras (segundo o calendário Gregoriano):
    1. De 400 em 400 anos é ano bissexto.
    2. De 100 em 100 anos não é ano bissexto.
    3. De 4 em 4 anos é ano bissexto.
      Solução vista em aula: (sem considerar anos bissextos):

27/04: Estruturas de decisão

Exercícios

  1. Um estudo sobre sensibilidade de pessoas a temperaturas da água identificou que a maioria das pessoas considera fria a água com temperaturas abaixo de 25 graus, morna entre 25 e 30 graus, e quente acima de 30 graus. Escreva um algoritmo que mostre as palavras "fria", "morna" ou "quente" dependendo da temperatura da água que for informada.
  2. Faça um algoritmo para fazer a divisão de dois números reais. Antes de dividi-los deve ser feito um teste de validade. Caso não seja possível dividi-los, deve ser mostrada uma mensagem de erro. Se for possível, deve-se mostrar o resultado da divisão.
  3. Escreva um programa que leia cinco números do teclado e mostre os dois maiores números.
  4. Faça um programa que leia o nome de um usuário e sua senha, e em seguida informe se o acesso foi concedido ou negado. Os usuários possíveis de entrar são "joao", "maria" e "zé".
  5. Jogo de Pedra, papel e tesoura: nesse jogo cada jogador faz sua escolha, e vence aquele que escolher um objeto que seja capaz de vencer o outro:
    • Tesoura corta papel
    • Pedra quebra tesoura
    • Papel embrulha a pedra
      Faça um algoritmo que pergunte ao usuário o objeto escolhido, e em seguida faça o computador escolher o seu próprio objeto (dica: use o gerador de números aleatórios visto no jogo do par ou ímpar). Finalmente o algoritmo deve informar o vencedor.

28/04: Estruturas de repetição

Alguns algoritmos vistos anteriormente possuem sequências repetitivas de instruções. Por exemplo, o algoritmo da média de quatro avaliações:

Inicio
  real m1, m2, m3, m4
  real media

  escrever "Avaliação 1:"
  Ler m1
  escrever "Avaliação 2:"
  Ler m2
  escrever "Avaliação 3:"
  Ler m3
  escrever "Avaliação 4:"
  Ler m4

  media <- (m1 + m2 + m3 + m4) / 4

  escrever "Média: ", media
Fim

O algoritmo acima repete quatro vezes a sequência que lê uma nota do teclado. Porém há uma forma de expressar a repetição de sequências de instruções, usando-se o que se chama de estrutura de repetição. A estrutura de repetição enquanto condição faz repete todas as instruções enquanto a condição for verdadeira. A condição é escrita como uma expressão lógica, da mesma forma que na estrutura de decisão se condição então ... senão. Veja como fica o algoritmo acima ao ser reescrito para usar essa estrutura de repetição:

Inicio
  constante inteiro NUMEROS <- 4
  real m
  real media <- 0
  inteiro contador <- 0

  enquanto contador < NUMEROS faz  
    escrever "Avaliação ", contador, ":"
    ler m
    media <- media + m
    contador <- contador + 1
  fimEnquanto

  escrever "Média: ", media/NUMEROS
Fim

Esse algoritmo funciona somente para médias de quatro números. Porém calcular a média de qualquer quantidade de números usa a mesma lógica: ler os números, somá-los e dividir o total pela quantidade de números lidos. Por exemplo, para fazer com que ele calcule a média de 7 números é necessário escrevê-lo assim:

Inicio
  constante inteiro NUMEROS <- 7
  real m
  real media
  inteiro contador <- 0

  enquanto contador < NUMEROS faz  
    escrever "Avaliação ", contador, ":"
    ler m
    media <- media + m
    contador <- contador + 1
  fimEnquanto

  escrever "Média: ", media/NUMEROS
Fim

Note que o algoritmo praticamente não foi modificado, pois somente se alterou o valor da constante NUMEROS.

Esses dois algoritmos podem também ser visualizados como fluxogramas, como mostrado na figura abaixo:

Ex-repeticao.png

A estrutura de repetição junto com a sequência de instruções a ser repetida é comumente chamada de 'laço. No fluxograma acima o laço está destacado em vermelho, e no algoritmo em Portugol o laço aparece em:

  enquanto contador < NUMEROS faz  
    escrever "Avaliação ", contador, ":"
    ler m
    media <- media + m
    contador <- contador + 1
  fimEnquanto

Um outro exemplo ainda mais simples é mostrar na tela uma contagem numérica:

inicio
    inteiro contador
    contador <- 0

    enquanto contador < 10 faz
        escrever contador , " "
        contador <- contador + 1
    fimenquanto
fim

Ao executar esse algoritmo, tem-se como resultado:

0 1 2 3 4 5 6 7 8 9

Seu fluxograma está mostrado a seguir:

Flux-Contagem.png


Repare no uso de uma variável auxiliar contador para controlar a quantidade de repetições. Essa é uma técnica comum para controlar a estrutura de repetição, e já foi usada no exemplo da média. Ela pode também ser combinada com outras condições, como mostrado abaixo:

inicio
    inteiro contador <- 1
    caracter opcao <- "s"
        
    enquanto ((opcao = "s") ou (opcao = "S")) e contador < 11 faz
        escrever contador, "\n"
        contador <- contador + 1
        
        escrever "\nContinuar (S/N) ? "
        ler opcao
    fimenquanto
    
    escrever "Terminou com contador = " , contador
fim

Nesse exemplo se usa um laço para mostrar uma contagem de 1 a 10. Porém há a possibilidade de encerrar o algoritmo. Observe a condição para continuidade do laço: ela é verdadeira somente se contador < 11 e se a variável opcao for igual a "s" ou "S".

Finalmente, apesar de ser comum o uso de contadores para controle do laço, pode-se usar qualquer condição para essa finalidade. Então podem existir laços que não usam contadores para controlar a quantidade de repetições, como no próximo exemplo:

inicio
    constante texto segredo <- "secreto"
    texto senha
    
    ler senha
    enquanto senha =/= segredo faz
        escrever "Voce continua preso ! Digite a senha correta: "
        ler senha
    fimenquanto
    escrever "Voce foi liberado ..."
fim

O fluxograma desse algoritmo segue abaixo:

Flux-Senha.png

Atividades

  1. Escreva um algoritmo que mostre a tabuada de um número fornecido pelo teclado. Esse número deve estar entre 1 e 10.
  2. Aproveite o algoritmo anterior para mostrar a tabuada de todos os números entre 1 e 10.
  3. Modifique o exemplo da média para que a quantidade de números a serem lidos seja previamente informada pelo teclado.
  4. Escreva algoritmos que mostrem as seguintes sequências de números:
    • Números pares menores que 100
    • Números ímpares menores que 100
    • Números pares de 100 até 4 (contagem decrescente)
    • Múltiplos de 3 menores que 100
    • Múltiplos de 4 e múltiplos de 5 menores que 100
    • Múltiplos comuns menores que 1000 de dois números fornecidos pelo teclado (ex: se forem fornecidos 6 e 8, deve aparecer 24, 48, 72, ...)
      inicio
          inteiro n <- 0 , n1 , n2, mmc
          
          escrever "Primeiro numero : "
          ler n1
          escrever "Segundo numero : "
          ler n2
          
          // descobre o mmc de n1 e n2
          se n1 > n2 entao
            mmc <- n1
          senao
            mmc <- n2
          fimse
          
          enquanto mmc % n1 =/= 0 OU mmc % n2 =/= 0 faz
             mmc <- mmc + 1
          fimenquanto
          
          // mostra os multiplos de mmc que sejam <= 1000
          enquanto n <= 1000 faz
              escrever "\n " , n
              n <- n + mmc
          fimenquanto
      fim
      
  5. Escreva um algoritmo para calcular o MMC (Mínimo Múltiplo Comum) de dois números. O MMC de dois números é um número que pode ser dividido por qualquer um deles sem deixar resto. Ex:
    • MMC de 8 e 6 = 24
    • MMC de 3 e 5 = 15
    • MMC de 3 e 27 = 27
  6. Escreva um algoritmo para calcular o MDC (Máximo Divisor Comum) de dois números. O MDC de dois números é um número capaz de dividi-los sem deixar resto. Ex:
    • MDC de 24 e 16 = 8
    • MDC de 5 e 3 = 1
    • MDC de 60 e 150 = 30
      inicio
          inteiro n1 , n2, mdc
       
          escrever "Primeiro numero : "
          ler n1
          escrever "Segundo numero : "
          ler n2
       
          // descobre o mmc de n1 e n2
          se n1 > n2 entao
            mdc <- n2
          senao
            mdc <- n1
          fimse
      
          enquanto mdc > 1 E NAO (n1 % mdc = 0 E n2 % mdc = 0) faz
             mdc <- mdc - 1
          fimenquanto
        
          escrever "MDC=", mdc
      fim
      
  7. Escreva um algoritmo que leia até 10 números do teclado, e informe ao final qual o maior e o menor deles.
  8. Escreva um algoritmo que teste se um número informado pelo teclado é primo.
  9. Aproveite o algoritmo do ítem anterior para mostrar todos os números primos menores que 1000.
  10. Modifique novamente o exemplo da média para que ela funcione para um quantidade de números desconhecida de antemão. Quer dizer, o algoritmo deve ler os números para calcular a média, mas não sabe quantos números existem (e isto não pode ser informado pelo teclado).
  11. Modifique o exemplo da senha para que o usuário tenha somente três tentativas permitidas para digitar a senha correta. Caso ao final as três senhas estejam erradas, o algoritmo deve informar que a conta foi bloqueada.
  12. Usando o algoritmo de teste de número primo, faça um novo algoritmo que fatore um número. Ex: a fatoração do número 15 resulta em 3 e 5 (3 X 5), e a fatoração de 54 resulta em 2, 3, 3 e 3 (2 X 3 X 3 X 3).
  13. Usando estrutura de repetição, escreva um algoritmo que converta um número de decimal para binário. Assuma um número binário com 16 bits.
  14. Faça um algoritmo para descobrir os dois maiores números dentre 6 números lidos do teclado.
  15. Faça um algoritmo para converter um número entre 0 e 16777215 para sua representação em hexadecimal.
  16. Escreva o algoritmo para uma calculadora aritmética (operações +, -, * e /) que lê continuamente do teclado números e operadores. A calculadora deve ler continuamente uma sequência do tipo número operador_aritmético, até que o operador informado seja =, quando então o resultado da conta deve ser mostrado na tela. Ex:
    • 1 + 2 * 5 / 3 = 5
    • 2 * 5 - 1 / 3 = 3

04/05: Estruturas de repetição

Resolução de exercícios (continuando aula anterior).

05/05: Estruturas de repetição

Continuando a resolução de exercícios ...

Uma pequena avaliação

Resolva o seguinte problema em 15 minutos:

Contador de moedas: faça um programa que leia valores de moedas e some as quantidades de tipos de moedas informadas. Por exemplo, se o usuário digitar 25, 50, 25, 5, 10, 5, o programa deve informar: 2 moedas de 5 centavos, 1 moeda de 10 centavos, 2 moedas de 25 centavos, 1 moeda de 50 centavos. São aceitos apenas valores de moedas de 1, 5, 10, 25 e 50 centavos. Seu programa deve ler 10 valores de moedas, e então apresentar o resultado.

inicio
  inteiro m1, m5, m10, m25, m50
  inteiro moeda 
  inteiro n
    
  enquanto n < 10 faz
    escrever "Moeda: "
    ler moeda
    
    escolhe moeda 
      caso 1:
        m1 <- m1 + 1
      caso 5:
        m5 <- m5 + 1
      caso 10:
        m10 <- m10 + 1
      caso 25:
        m25 <- m25 + 1
      caso 50:
        m50 <- m50 + 1
      defeito:
        escrever "Valor invalido ...\n"
        n <- n - 1
    fimescolhe
    n <- n + 1
  fimenquanto

  escrever "Moedas de 1 centavo: ", m1
  escrever "\nMoedas de 5 centavos: ", m5
  escrever "\nMoedas de 10 centavos: ", m10  
  escrever "\nMoedas de 25 centavos: ", m25
  escrever "\nMoedas de 50 centavos: ", m50
fim

11/05: Estruturas de repetição

Variáveis multidimensionais

Em matemática existem matrizes e vetores, que são variáveis multidimensionais. Por exemplo, uma matriz 2 x 2 (2 linhas e 2 colunas) pode ser:

Vetores são matrizes unidimensionais, portanto possuem somente uma linha ou uma coluna:

Cada elemento em uma matriz ou vetor é identificado pela sua posição. Por exemplo, na posição 1, 2 da matriz A acima está o valor 6, e na posição 4 do vetor v está o valor 18. Assim, a matriz A do exemplo pode ser entendida da seguinte forma:

... e o vetor v do exemplo:

Nas linguagens de programação em geral existe um conceito muito parecido, chamado de variáveis multidimensionais ou simplesmente matrizes (arrays em inglês). Para exemplificar, no Portugol se poderiam definir a matriz A e o vetor v:

inicio
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
  inteiro v[5] <- {1, 6, 2, 18, 5}

  escrever "Valor de A[0][0]: ", A[0][0], "\n"
  escrever "Valor de A[0][1]: ", A[0][1], "\n"
  escrever "Valor de A[1][0]: ", A[1][0], "\n"
  escrever "Valor de A[1][1]: ", A[1][1], "\n"

  escrever "Valor de v[0]: ", v[0], "\n"
  escrever "Valor de v[1]: ", v[1], "\n"
  escrever "Valor de v[2]: ", v[2], "\n"
  escrever "Valor de v[3]: ", v[3], "\n"
  escrever "Valor de v[4]: ", v[4], "\n"

fim

A declaração da matriz se faz como uma variável comum, porém indicando-se suas dimensões:

  inteiro A[2][2] <- {{1, 6}, {3, 5}}

Veja que a matriz A foi inicializada na declaração.

O acesso aos elementos da matriz se faz usando-se índices, que funcionam como coordenadas dos elementos que se quer acessar. Os índices iniciam em 0:

  # índice 0,0
  escrever "Valor de A[0][0]: ", A[0][0], "\n"

O exemplo acima pode ser reescrito usando-se estruturas de repetição:

inicio
  inteiro A[2][2] <- {{1, 6}, {3, 5}}
  inteiro v[5] <- {1, 6, 2, 18, 5}
  inteiro i, j

  enquanto i < 2 faz
    j <- 0
    enquanto j < 2 faz
       escrever "Valor do elemento de A na posição ", i, ", ", j, ": ", A[i][j], "\n"
       j <- j + 1
    fimenquanto
    i <- i + 1
  fimenquanto

  i <- 0
  enquanto i < 5 faz
    escrever "valor de v na posição ", i, ": ", v[i], "\n"
    i <- i + 1
  fimenquanto

fim

Como se vê, matrizes e vetores casam perfeitamente com estruturas de repetição, e são úteis para a resolução de inúmeros problemas. Por exemplo, o problema de mostrar o nome do mês a partir do número do mês, feito anteriormente com escolhe ... caso pode ser feito assim com um vetor:

inicio
  texto nome_mes[12] <- {"Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho",
                         "Julho", "Agosto", "Setembro", "Outubro", "Novembro",
                         "Dezembro"}
  inteiro dia, mes, ano

  escrever "Dia: "
  ler dia
  escrever "Mes: "
  ler mes
  escrever "Ano: "
  ler ano

  escrever dia, " de ", nome_mes[mes - 1], " de ", ano
fim

Outro problema comum é precisar guardar um conjunto de valores, e depois ordená-los:

inicio
  inteiro v[10]
  inteiro i
  inteiro quantidade

  escrever "Quantos valores serão digitados (máximo = 10) ? "
  ler quantidade

  i <- 0
  enquanto i < quantidade faz
    escrever "valor ", i, ": "
    ler v[i]
    i <- i + 1
  fimenquanto

  // ordena os valores ...

  escrever "valores ordenados: "
  i <- 0
  enquanto i < quantidade
    escrever v[i], " "
    i <- i + 1
  fimenquanto
fim

Atividades

  1. Escreva um algoritmo que leia 5 números do teclado, e depois mostre-os em ordem inversa à que foi digitada.
  2. Escreva um algoritmo que leia dois vetores de 5 números, e depois mostre se os vetores são iguais.
  3. Escreva um algoritmo que leia um palavra do teclado e depois procure-a numa lista de palavras preexistente. Essa lista deve ser implementada usando um vetor.
  4. Escreva um algoritmo que leia 5 números, e mostre-os em ordem crescente.
  5. Modifique o algoritmo anterior para mostrá-los em ordem decrescente.
  6. Em um jogo de bingo devem-se contar quantos números de uma aposta foram sorteados no bingo. Faça um algoritmo que, dados os números sorteados e uma aposta, conta quantos números apostados foram sorteados.

12/05: Estruturas de repetição

Para variavel de inicio ate fim passo incremento

Há casos em que se deseja repetir uma parte de um algoritmo certo número de vezes. Para esses casos há uma estrutura mais prática que enquanto condição faz:

inicio
  inteiro i
  inteiro v[5]

  escrever "Digite 5 números: \n"

  para i de 0 ate 4 passo 1
    escrever "Numero ", i, ": "
    ler v[i]
  proximo

  escrever "\nOs numeros em ordem inversa são:\n\n"

  para i de 4 ate 0 passo -1
    escrever "Numero ", i, ": ", v[i], "\n"
  proximo

fim

Atividades

  1. Escreva o algoritmo para comparar dois vetores, porém usando a estrutura para.
  2. Escreva um algoritmo para procurar se um vetor menor (com menos valores) está contido em um vetor maior. Caso afirmativo, mostre a partir de que posição ele aparece. Assuma vetores de números inteiros.
  3. Modifique o algoritmo acima para mostrar todas as ocorrências do vetor menor dentro do maior.
  4. Modifique o algoritmo do anterior para remover todas as ocorrências do vetor menor dentro do maior.
  5. Resolva o problema do bingo usando a estrutura para

18/05: Revisão

...exercícios

Resolver a avaliação de 2010-2:

1) Contador de moedas: faça um programa que leia valores de moedas e some as quantidades de tipos de moedas informadas. Por exemplo, se o usuário digitar 25, 50, 25, 5, 10, 5, o programa deve informar: 2 moedas de 5 centavos, 1 moeda de 10 centavos, 2 moedas de 25 centavos, 1 moeda de 50 centavos. São aceitos apenas valores de moedas de 1, 5, 10, 25 e 50 centavos. Seu programa deve ler 10 valores de moedas, e então apresentar o resultado.

2) Calculadora de troco: escreva um programa que calcule o troco a ser devolvido a um cliente. O troco deve ser composto de notas de 50, 20, 10, 5, 2 ou 1 real, e também moedas de 1, 5, 10, 25 ou 50 centavos. Dê preferência ao uso de notas e moedas de maior valor. Por exemplo, se o troco a ser devolvido for R$ 17,75, seu programa deve informar 1 nota de R$ 10, 1 nota de R$ 5, 1 nota de R$ 2, 1 moeda de R$ 0,50, 1 moeda de R$ 0,25.

3) Modificação de senha: Para modificar uma senha deve-se primeiro fornecer a senha atual, e então a nova senha. É necessário escrever duas vezes a nova senha para confirmá-la. Escreva um algoritmo que modifique uma senha predefinida (no seu programa ela pode estar guardada em uma variável), seguindo essa regra de mudança de senha.

4) Nota de uma prova olímpica: um atleta de saltos ornamentais tem seu salto avaliado por 6 juízes. Após descartar a maior e a menor nota, calcula-se a média aritmética das notas dadas. Faça um programa que calcule a nota de um atleta.

... mais algumas questões interessantes:

5) Filtro: Um programa possui 10 números inteiros em um vetor. No entanto, para a próxima etapa de processamento apenas os números entre 3 e 8 são válidos. Escreva um algoritmo que remova os números inválidos do vetor. Ao final o vetor deve conter somente os números válidos.

6) Tabuleiro: Escreva um programa que mostre o conteúdo de uma matriz 4 x 4 na tela. Essa matriz contém valores inteiros que podem ser positivos ou negativos. Ao mostrar cada valor da matriz, deve-se seguir a seguinte regra:

  • Valor negativo: deve ser mostrado um espaço em branco
  • Valor entre 1 e 8: mostrar o próprio valor
  • Valor acima de 8: mostrar o caractere H

Além disso, a matriz deve ser mostrada com uma grade. Por exemplo, a matriz com conteúdo:

... que em Portugol seria declarada assim:

inteiro A <- {{1, 6, 10, -1}, {3, 5, -1, -1}, 
              {10, 4, -1, 5}, {3, -1, 1, -1}}

... deveria ser mostrada da seguinte forma:

..| 1 | 2 | 3 | 4 |
-------------------
A | 1 | 6 | H |   |
-------------------
B | 3 | 5 |   |   |
-------------------
C | H | 4 |   | 5 |
-------------------
D | 3 |   | 1 |   |
-------------------

Quer dizer, as colunas devem ser identificadas por seus números, e as linhas devem ser identificadas pelas letras A, B,C, D.

Linguagem C: Projeto final

19/05: Introdução à linguagem C

Obs: durante as aulas usaremos o NetBeans, um Ambiente Integrado de Desenvolvimento (IDE - Integrated Development Environment). Um IDE é um programa que auxilia o programador. Ele integra um editor de programas, um gerenciador de projetos, compilador e um depurador (debugger).

Para instalar o Netbeans:

  1. Faça o download do instalador. Salve-o em algum subdiretório.
  2. Abrindo um terminal, entre no subdiretório onde está o instalador e execute esse comando:
    bash netbeans-6.8-ml-cpp-linux.sh
    
    • Caso o instalador falhe porque não encontrou a JVM (Máquina Virtual Java), instale-a com esse comando:
      sudo apt-get install -y sun-java6-jdk
      
  3. Aceite as opções de instalação sugeridas pelo instalador.
  4. Ao final da instalação, o Netbeans 6.8 estará acessível pelo menu Aplicativos -> Programação.
  5. Sempre que for escrever um novo programa com o Netbeans, crie um novo projeto (ver menu Arquivo->Novo Projeto). Selecione um projeto do tipo "Aplicativo C/C++".

Compilando o primeiro programa

Obs: para compilar um programa você deve salvá-lo em um arquivo com extensão .c. Por exemplo, ao editar o exemplo abaixo, você pode gravá-lo no arquivo hello.c. Em seguida você deve compilá-lo usando o seguinte comando:

gcc hello.c -o hello

Para executá-lo, faça assim:

./hello

O arquivo que contém o programa em linguagem C se chama arquivo de código-fonte. O arquivo que contém o programa compilado se chama arquivo de código-objeto.


  • O clássico Hello World!
#include <stdio.h>

int main(int argc, char *argv[])
{
	printf("Alô mundo!\n");
}
  • Mostrando mensagens de na tela: puts e printf:
    #include <stdio.h>
    
    int main() {
      int n;
    
      n = 5;
      
      puts("Demonstração de puts e printf");
    
      printf("Valor de n: %d\n", n);
    
      n = n + 1;
    
      printf("Novo valor de n: %d\n", n);
    
      return 0;
    }
    
  • Lendo dados do teclado: scanf
    #include <stdio.h>
    
    int main() {
      int n;
    
      printf("Digite um número inteiro: ");
    
      scanf("%d", &n);
    
      printf("Valor digitado foi %d\n", n);
    
      return 0;
    }
    

Tipos de dados

Portugol C Exemplo
inteiro int int x;
caracter char char letra;
real float ou double float pi = 3.1416;
texto char * ou vetor de char char * mensagem = "Hello world";
char palavra[16];
logico qualquer tipo (valor 0 = Falso, qualquer outro valor = Verdadeiro) int ok = 1;
char teste = 0;


Comandos/funções e Estruturas de controle

Portugol C Exemplo
inteiro n, m

escrever "Ola, mundo!\n"
escrever "valor de n: ", n, "\n"
escrever "valor de n: ", n, ", e valor de x: ", x, "\n"
int n, m;

printf("Ola, mundo!\n");
printf("valor de n: %d\n", n);
printf("valor de n: %d, e valor de x: %d\n", n, x);
inteiro n

ler n
int n;

scanf("%d", &n);
se condição então
//comandos
fimse
if (condição) {
//comandos
}
se condição então
  //comandos
senão
  //comandos
fimse
if (condição) {
 //comandos
} else {
 //comandos
}
escolhe expressão
  caso valor1:  
    //comandos
  caso valor2:
    //comandos
  defeito:
    //comandos
fimescolhe
switch (expressão) {
  case valor1:
    //comandos
  case valor2:
    //comandos
  default:
    //comandos
}
enquanto condição faz
//comandos
fimenquanto
while (condição) {
  //comandos
}
para variavel de inicio ate fim passo incremento
//comandos
proximo
for (variavel=inicio; variavel <= fim; variavel++) {
//comandos
}

Atividades

  • Traduza para C os seguintes algoritmos Portugol:
    1. Inicio
        inteiro x, y
       
        Escrever "Digite um numero: ",
        Ler x
        Escrever "Digite outro numero: ",
        Ler y
        Escrever "Soma = ", x+y
      Fim
      
    2. Inicio
        inteiro x, y, z
       
        Escrever "Digite um numero: ",
        Ler x
        Escrever "Digite outro numero: ",
        Ler y
        z <- x + y
        Escrever "Soma = ", z
      Fim
      
    3. Inicio
        Inteiro n1, n2, n3, r
       
        Escrever "Primeiro numero: "
        ler n1
        Escrever "Segundo numero: "
        ler n2
        Escrever "Terceiro numero: "
        ler n3
       
        r <- (n1 + n2 + n3) /3
       
        Escrever "Media=", r
      Fim
      
    4. Inicio
        inteiro anos, dias
       
        Escrever "Quantos anos se passaram ? "
        Ler anos
        dias <- anos * 365
        Escrever "... então se passaram ", dias, " dias"
      Fim
      
    5. Inicio
        constante inteiro diasPorAno <- 365
        inteiro anos, dias
       
        Escrever "Quantos anos se passaram ? "
        Ler anos
        dias <- anos * diasPorAno
        Escrever "... então se passaram ", dias, " dias"
      Fim
      
    6. inicio
        real massa
        real forca
        real aceleracao
        constante real g <- 9.8
        constante real ac <- 0.01
        
        escrever "força: "
        ler forca
        
        escrever "massa: "
        ler massa
        
        aceleracao <- forca/massa - ac*massa*g
        
        escrever "aceleracao=", aceleracao, " m/s2"
        
      fim
      
    7. inicio
        real a, b, c
        real delta
        real x1, x2
        
        escrever "Forneça os coeficientes da equação de 2o grau (formato: ax^2 + bx + c):\n"
        escrever "a="
        ler a
        escrever "b="
        ler b
        escrever "c="
        ler c
        
        delta <- b^2 - 4*a*c
        x1 <- (-b + delta^0.5)/(2*a)
        x2 <- (-b - delta^0.5)/(2*a)
        
        escrever "1a raiz=", x1
        escrever "\n2a raiz=", x2  
      fim
      
    8. inicio
            real x1, x2, x3, x4, x5
            real media
            real desvioPadrao
            constante inteiro N <- 5
            
            // Ler os cinco valores pelo teclado
            ler x1
            ler x2
            ler x3
            ler x4
            ler x5
            
            // Calcular a media
            media <- (x1 + x2 + x3 + x4 + x5) / N
            
            // Calcular o desvio padrao
            desvioPadrao <- (x1 - media)^2 + (x2 - media)^2 + (x3 - media)^2 + (x4 - media)^2 + (x5 - media)^2
            desvioPadrao <- (desvioPadrao / (N-1))^0.5
            
            escrever "Media=", media
            escrever "\nDesvio padrao=", desvioPadrao
      fim
      
    9. inicio
            real renda
            real irpf
            
            escrever "Informe sua renda: "
            ler renda
            
            se renda < 1372.81 entao
              irpf <- 0
            senao 
              se renda < 2743.25 entao
                irpf <- (renda - 205.92)*0.15 
              senao
                irpf <- (renda - 548.82)*0.275
              fimSe
            fimSe
            
            escrever "Imposto devido: ", irpf  
      fim
      
    10. inicio
          inteiro dividendo
          inteiro divisor
          inteiro resto , quociente
      
          escrever "Dividendo: "
          ler dividendo
          escrever "Divisor: "
          ler divisor
      
          se divisor = 0 entao
              escrever "Nao pode dividir: divisor = 0 !!!"
          senao
              resto <- dividendo % divisor
              se resto = 0 entao
                  quociente <- dividendo / divisor
                  escrever dividendo , " / " , divisor , " = " , quociente
              senao
                  escrever "Nao pode fazer divisao inteira"
                  escrever " (resto = " , resto , ")"
              fimse
          fimse
      fim
      
    11. inicio
          inteiro termometro , barometro , tempo
      
          escrever "Qual a condição do termômetro: \n"
          escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
          ler termometro
      
          escrever "Informe a condição do barômetro: \n"
          escrever "Digite 1 para baixando , 2 para estacionário e 3 para subindo. \n"
          ler barometro
      
          tempo <- termometro*10 + barometro
      
          escolhe tempo
              caso 11:
                  escrever "Chuvas abundantes e ventos de sul a sudoeste fortes"
              caso 12:
                  escrever "Chuva Provavel , ventos de sul a sudeste"
              caso 13:
                  escrever "Tempos Bons , ventos de sul a sudeste"
              caso 21:
                  escrever "Frente quente com chuva provavel"
              caso 22:
                  escrever "Tempo Incerto , ventos variaveis"
              caso 23:
                  escrever "Tempos Bons , ventos do leste frescos"
              caso 31:
                  escrever " Tempo instavel , aproximaçao de frente"
              caso 32:
                  escrever "Tempo Mudando para bom , ventos de leste"
              caso 33:
                  escrever "Tempos Bons , ventos quentes e secos"
              defeito:
                  escrever "Utilize somente os algorismos de 1 a 3 para indicar as condiçoes do equipamentos"
          fimescolhe
      fim
      
    12. inicio
          // coeficientes do polinomio  [ax^2 + bx + c = 0 ]
          real a , b , c
          
          escrever "Forneça os coeficientes da equação de 2o grau:\n"
          escrever "a="
          ler a
          escrever "b="
          ler b
          escrever "c="
          ler c
          
          // equação do tipo [ bx + c = 0 ]
          se a = 0 entao
              escrever " não é uma equação de 2o grau !!!"
          senao
              // calcular o delta => interior da raiz
              real delta
              
              delta <- b ^ 2 - 4 * a * c
              
              // não existem raizes  reais de números negativos
              se delta < 0 entao
                  escrever " não tem raizes reais"
              senao
                  // -----------  raiz dupla  ----------------
                  se delta = 0 entao
                      real x1
                      x1 <- -b / 2 * a
                      escrever "\nraiz dupla : " , x1
                  senao
                      // - ---------- duas raizes ---------------
                      real x1 , x2
                      x1 <- ( -b + raiz ( delta ) ) / 2 * a
                      x2 <- ( -b - raiz ( delta ) ) / 2 * a
                      escrever "\nraiz x1 : " , x1
                      escrever "\nraiz x2 : " , x2
                  fimse//raiz dupla
              fimse// delta >0
          fimse// a <> 0
      fim
      
    13. inicio
          inteiro num, bit
          inteiro base
          
          escrever "Este algoritmo converte um número inteiro para sua representação binária.\n"
          Escrever "O número deve estar entre 0 e 131071, e a representação binária\n"
          escrever "terá 17 bits\n\n"
          escrever "Digite o numero a ser convertido para binário: "
          ler num
      
          base <- 65536
          enquanto base > 0 faz
              bit <- ( num / base ) % 2
              escrever bit, " " 
              base <- base / 2
          fimenquanto
      fim
      
    14. inicio
          inteiro n , fat
          ler n
          fat <- 1
          enquanto n > 1 faz
             fat <- fat * n
             n <- n - 1         
          fimenquanto
          
          escrever "Fatorial=" , fat
      fim
      
    15. inicio
          inteiro valor , menor , contador
          ler menor
          contador <- 9
          enquanto contador > 0 faz
              ler valor
              se valor < menor entao
                  menor <- valor
              fimse
              contador <- contador - 1
          fimenquanto
          escrever "menor valor = ", menor
      fim
      
    16. inicio
          inteiro numero , n , resto <- 1
          escrever "Forneçaa um número: "
          ler numero
          n <- 2
          enquanto n <= numero / 2 e resto =/= 0 faz
              resto <- numero % n
              escrever "n=" , n , ": resto=" , resto , "\n"
              n <- n + 1
          fimenquanto
          
          se resto = 0 entao
              escrever "nao primo: divisivel por " , n - 1
          senao
              escrever "primo"
          fimse
      fim
      
    17. // Le 5 valores e identifica os dois maiores.
      inicio
          inteiro valor , maior1 , maior2 , contador
          
          ler valor
          maior1 <- valor
          
          ler valor
          se valor > maior1 entao
            maior2 <- maior1
            maior1 <- valor
          senao
            maior2 <- valor
          fimse
      
          contador <- 3
          enquanto contador > 0 faz
              ler valor
              se valor > maior1 entao
                  maior2 <- maior1
                  maior1 <- valor
              senao
                  se valor > maior2 entao
                      maior2 <- valor
                  fimse
              fimse
              contador <- contador - 1
          fimenquanto
      
          escrever "maior valor = " , maior1
          escrever "\nsegundo maior valor = " , maior2
      fim
      

25/05: Avaliação

No laboratório ...

Prova resolvida

1) Um programa em uma estação meteorológica converte os valores lidos de um higrômetro para um indicativo de umidade do ar. A tabela de conversão segue abaixo:

Valor do higrômetro Indicativo de umidade
0 a 0.3 muito baixa
0.3 a 0.5 baixa
0.5 a 0.8 normal
0.8 a 1 alta
1 precipitação

Escreva esse programa, que deve simular a leitura do higrômetro pelo teclado e mostrar o indicativo de umidade na tela.

inicio
    real h
    
    escrever " digite um valor de 0 a 1:  " , 
    ler h
    
    se h = 1 entao
        escrever "chuva" ,
    senao
        se h <= 0.3 entao
            escrever "umidade muito baixa" ,
        senao
            se h <= 0.5 entao
                escrever "umidade baixa" ,
            senao
                se h <= 0.8 entao
                    escrever "umidade normal"
                senao
                    se h < 1 entao
                        escrever "umidade alta"
                        
                    fimse
                fimse
            fimse
        fimse
    fimse
fim

2) Um programa possui o seguinte vetor de inteiros guardado em memória:

{10,12,7,4,5,9,2,3,1,3,4,2,2,9}

Esse programa deve ler uma sequência de 4 números pelo teclado, e procurá-la dentro do vetor que está em memória. Se a sequência for encontrada, a posição em que ela inicia dentro do vetor deve ser mostrada.

inicio
  inteiro numeros[14] <- {10,12,7,4,5,9,2,3,1,3,4,2,2,9}
  inteiro v[4]
  inteiro i, j
  
  Escrever "Digite 4 numeros inteiros:"
  para i de 0 ate 3 passo 1
    ler v[i]
  proximo
  
  para j de 0 ate 10 passo 1
    i <- 0
    enquanto i < 4 E v[i] = numeros[j+i] faz
      i <- i + 1
    fimenquanto
    se i = 4 entao
      escrever "Encontrou na posicao ", j
      fim
    fimse
  proximo
  escrever "Nao encontrou em nenhuma posicao"
fim

3) Um número primo é um número divisível somente por 1 ou por si mesmo. Quer dizer, se tentar dividi-lo por qualquer valor maior que 1 e menor que esse número, o resto será diferente de zero. Exemplos de números primos são 2, 3, 5, 7 e 11. Escreva um programa que leia um número pelo teclado e informe se ele é primo.

inicio
    inteiro numero , n

    escrever "Forneça um número: "
    ler numero

    n <- 2   
    enquanto n <= numero / 2 E resto =/= 0 faz       
        resto <- numero % n
        n <- n + 1
    fimenquanto
    
    se resto = 0 entao
        escrever "nao primo: divisivel por " , n - 1
    senao
        escrever "primo"
    fimse
fim

... uma versão com uma pequena melhoria:

inicio
    inteiro numero, n, resto

    escrever "Forneça um número: "
    ler numero

    
    resto <- numero % 2
    se resto =/= 0 entao
      n <- 1
      enquanto n <= numero / 2 E resto =/= 0 faz       
          n <- n + 2
          resto <- numero % n
      fimenquanto
      
      se resto = 0 entao
          escrever "nao primo: divisivel por " , n
      senao
          escrever "primo"
      fimse
    senao
      escrever "nao primo: divisivel por 2"
    fimse
fim

4) Jogo de Pedra, papel e tesoura: nesse jogo cada jogador faz sua escolha, e vence aquele que escolher um objeto que seja capaz de vencer o outro:

* Tesoura corta papel
* Pedra quebra tesoura
* Papel embrulha a pedra

Faça um algoritmo que pergunte ao usuário o objeto escolhido, e em seguida faça o computador escolher o seu próprio objeto (dica: use o gerador de números aleatórios visto no jogo do par ou ímpar). Ao final o algoritmo deve informar o vencedor.

inicio
    inteiro lancecomputador , lancejogador
    logico jogadorvenceu
    
    escrever "Escolha papel (1) , pedra (2) ou tesoura (3): "
    ler lancejogador
    se lancejogador > 3 ou lancejogador < 1 entao
        escrever "Lance inválido !!! (deve digitar um número entre 1 e 3)\n\n"
    senao
        lancecomputador <- 1 + 2 * aleatorio ( )
        escrever "Computador escolheu: " , lancecomputador , "\n"
        se lancejogador = lancecomputador entao
            escrever "Empate !\n\n"
        senao
          jogadorvenceu <- falso
          escolhe lancejogador
              caso 1:
                  se lancecomputador = 2 entao
                      jogadorvenceu <- verdadeiro
                  fimse
              caso 2:
                  se lancecomputador = 3 entao
                      jogadorvenceu <- verdadeiro
                  fimse
              caso 3:
                  se lancecomputador = 1 entao
                      jogadorvenceu <- verdadeiro
                  fimse
          fimescolhe
          escrever "Vencedor: "
          se jogadorvenceu entao
              escrever "jogador\n\n"
          senao
              escrever "computador\n\n"
          fimse
      fimse
    fimse
fim

... uma outra solução:

inicio

   inteiro lancecomputador , lancejogador
   logico jogadorvenceu
   texto vencedor , opcao
   
   escrever "Jogo de Papel , Pedra e Tesoura\n\n"
   escrever "Escolha papel (1) , pedra (2) ou tesoura (3): "
   ler lancejogador
   se lancejogador > 3 ou lancejogador < 1 entao
       escrever "Lance inválido !!! (deve digitar um número entre 1 e 3)\n\n"
   senao
       lancecomputador <- 1 + 2 * aleatorio ( )
       escrever "Computador escolheu: " , lancecomputador , "\n"
       se lancejogador = lancecomputador entao
           escrever "Empate !\n\n"
       senao
           escrever "Vencedor: "
           // repare em como se determina o vencedor nesta versao do jogo: a condicao abaixo
           // é verdadeira sempre que o lance do jogador estiver uma unidade atrás do computador:
           //
           // Jogador     Computador
           //    1             2
           //    2             3
           //    3             1 (nesta última preciso considerar que a contagem é circular)
               
           se (lanceJogador%3 + 1) = lanceComputador entao
               escrever "jogador\n\n"
           senao
               escrever "computador\n\n"
           fimse
       fimse
   fimse

fim </syntaxhighlight>

5) Um programa somente pode ser usado por usuários autorizados. Os usuários e seuas senhas estão armazenados em uma tabela dentro do próprio programa. Escreva o algoritmo desse programa que pede o usuário e senha pelo teclado, consulta a tabela de usuários e depois informa se o acesso foi concedido ou negado. Os usuários e respectivas senhas estão mostrados abaixo:

Usuário Senha
joao tainha
maria pirao
maneca onibus
bilica bilro

Obs: essa tabela pode ser modificada futuramente, então o algoritmo deve conseguir trabalhar com quaisquer usuários e senhas que ali sejam armazenados. Isso é, mesmo que essa tabela tenha seu conteúdo alterado, o algoritmo não pode ser modificado.

inicio
  texto usuarios <- {"joao", "maria", "maneca", "bilica"}
  texto senhas <- {"tainha", "pirao", "onibus", "bilro"}
  texto login, umaSenha
  inteiro i
  logico OK <- falso
  
  escrever "Usuario:"
  ler login
  
  escrever "Senha:"
  ler umaSenha
  
  enquanto i < 4 E nao OK faz
    ok <- usuarios[i] = login
    i <- i + 1
  fimenquanto

  se ok entao
    ok <- senhas[i] = umaSenha
  fimse
  
  se ok entao
    escrever "acesso concedido"
  senao
    escrever "acesso negado"
  fimse
  
fim

Conceitos

Legenda:

  • T: questão correta
  • P: questão parcialmente correta
  • N: questão incorreta ou em branco
Aluno Conceito Questão 1 Questão 2 Questão 3 Questão 4 Questão 5
Alfredo D T (mas não da forma mais eficiente) N N N N
Ângelo D T (mas não da forma mais eficiente) N P N N
Beatriz D N N P (primos somente menores que 12) N N
Bruno da Silva D N (erros de sintaxe) N T N (copiado de um exemplo do professor) N
Bruno D N N N N N
Caroline D T N N N N
Danilo B T (mas não da forma mais eficiente) T (mas não da forma mais eficiente) N (idêntico ao do Gabriel) T T (mas não da forma mais eficiente)
Dayze D N N N N N
Fabiana C T (mas não da forma mais eficiente) P N T N
Felipe C T N N P (computador deveria jogar) P (funciona só pra 4 contas)
Gabriel D T (mas não da forma mais eficiente) N N (idêntico ao do Danilo) N P (funciona só pra 4 contas)
Grazielle D N N N N N
Jayson D N N N N P (funciona só pra 4 contas)
Juliano C T (mas não da forma mais eficiente) N N T P (funciona só pra 4 contas)
Leandro D T (mas não da forma mais eficiente) N N N N
Leonardo D P (condições trocadas) N N P (não indica o vencedor) N

26/05: Começo do projeto

Para referência: apostila online sobre linguagem C.