PRG29002 - Programação I - Eng.Telecom 2016-1

De MediaWiki do Campus São José
Ir para navegação Ir para pesquisar

Professor da Disciplina: Cleber Jorge Amaral
e-mail: cleber.amaral@ifsc.edu.br

Critérios e instrumentos de avaliação

  • Conceito => Somatório(Nota)/QtNotas
    • Esta é uma previsão, eventuais mudanças serão comunicadas no decorrer das atividades
    • Nota[1]: Nota da Avaliação 1 (a definir data e formato)
    • Nota[2]: Nota da Avaliação 2 (a definir data, formato e necessidade desta segunda avaliação)
    • Nota[3]: Média das notas das Listas de exercícios
    • Nota[4]: Nota do Projeto final
  • Frequência
    • Mínimo 75%

Datas importantes

  • 13/04/2016
    • Lista de exercícios 1: Entregar por e-mail ou manuscrito
    • Desafio 1 (projeto de cafeteira): Entregar por e-mail ou manuscrito a Narrativa, Fluxograma e Pseudocódigo
  • 20/04/2016
    • Lista de exercícios 2: Entregar por e-mail (seguir instruções) ou manuscrito
    • Lista de exercícios 3: Entregar por e-mail (seguir instruções) ou manuscrito
      • O estudante deve entregar na forma de fluxograma cada desafio que resolvemos na aula de 13/04 na forma de pseudocódigo, e devem ser entregues na forma de pseudocódigo os fluxogramas que fizemos em sala -
      • Os títulos dos algoritmos são citados na mídia 1.3 (link abaixo) mas os detalhes foram trabalhados em sala e fotografados pelos próprios alunos
  • 27/04/2016
    • Lista de exercícios 4: Entregar por e-mail (seguir instruções) ou manuscrito
  • 11/05/2016
    • Lista de exercícios 5: Entregar por e-mail (seguir instruções)
  • 17/05/2016
    • Prova 1: Algoritmos e lógica utilizando pseudocódigo e fluxogramas
  • 25/05/2016
    • Lista de exercícios 6: Entregar via moodle
  • 28/05/2016
    • Desafio 2 (jogo da velha): Entregar via moodle
  • 08/06/2016
    • Lista de exercícios 7: Entregar via moodle (atraso nos 6 primeiros dias contarão -1)
  • 14/06/2016
    • Apresentação das propostas de projeto final
  • 22/06/2016
  • 04/07/2016 as 17:30
    • Prova Rec 1: Teórica (Lab. Redes 2)
  • 19/07/2016 (horário de aula)
    • Prova Rec 2: Prática
  • 26 e 27/07/2016
    • Avaliação 3: Apresentação do projeto

Material de aula

Inauguração

Introdução aos Algoritmos e Pseudocódigo

Programação em C

Introdução

Controle de fluxo

Estruturas de controle de fluxo
  • A estrutura mais elementar de decisão? Saloni em Code.org
  • O que é uma estrutura de repetição? Chris Bosh em Code.org
  • O que é um laço de repetição? - Mark Zuckerberg Code.org
  • O que é uma estrutura de decisão? Bill Gates em Code.org
  • Como funciona a estrutura de repetição tipo contagem (for no C)? - Code.org

Funções

Funções
  • O que é uma função? Khan Academy
  • O que é uma função - noção intuitiva - MeSalva!
Alguns cuidados com formatos de entrada do scanf
  • Captura de caractere: pode ser utilizar o especificador '%c', porém este especificador captura inclusive "\n" (o ENTER, ou nova linha). O efeito é que uma captura anterior de scanf pode ter deixado uma "sujeira". Para previnir isso utilize " %c" (observe o espaço antes do '%c'
  • Captura de strings incluindo espaços: o especificador '%s' não captura espaços em branco, isto pode ser muito útil se o desejado é exatamente separar uma string (frase) em palavras. Porém quando a string que se deseja capturar pode conter espaços e ainda assim deseja-se processar como uma string única é necessário utilizar outros especificadores. Um deles o "%[^\n]s" ou "%[^\n]" indica que deseja-se obter um conjunto de caracteres que pode ter qualquer valor exceto '\n'. Outra opção é um especificador como "%10[0-9a-zA-Z ]" que indica que deseja-se obter até 10 caracteres que enteja dentro das variações "0 a 9", "a a z", "A a Z" e " ".

Fonte: cppreference

Variáveis e operadores

Variáveis e memória computacional
  • O que é uma variável? - Khan Academy
  • Como funciona a memória de computadores? - Khan Academy
Exercicios complementares - Vetores
  1. Implementar um programa em C para ler 10 números reais (float) para um vetor. Usar o comando while.
  2. Modificar o exercício para computar a média dos 10 números que estão no vetor.
  3. Modificar o exercício anterior para computar a quantidade de números do vetor que estão acima da média.
  4. Refazer os exercícios anteriores usando o comando for;
  5. Considere um vetor global de floats chamado vetRnd de tamanho 100. Construa uma função que inicia este vetor com 100 números randômicos entre 1 e 6.
  6. Sobre o exercício anterior, acrescente uma função para imprimir o número de 0s,1s,...6s do vetor.
  7. Implementar uma função que recebe dois vetores de inteiros como parâmetro e o tamanho dos mesmos (suponha vetores de mesmo tamanho). A função deve retornar o número de elementos iguais comparados posição por posição. O esqueleto da função deve ser como:
    int compara_vetores(int ve1[],int vet2[], int tamanho)
    {
      int num_elementos;
    
      return num_elementos;
    }
    

Exemplo: Para os vetores x[]={1,1,3,4,5} e y[]={1,2,3,3,5} temos três elementos iguais (nas posições 0, 2 e 4).

Variáveis locais e Globais
  • Se variáveis são declaradas dentro de uma função, então a visibilidade (ESCOPO) destas variáveis é LOCAL. Nenhuma outra função tem acesso a estas variáveis. Uma variável pode ser GLOBAL, ou seja, declarada FORA das funções. Neste caso a variável é VISTA por todas as funções.
    • Neste exemplo, a variável media é declarada como GLOBAL. Ela é MODIFICADA diretamente pela função media_nums() e impressa pela função main()
      • NOTE que como a função media_nums() não retorna valor então declaramos seu tipo de retorno como void que significa aqui NADA ou VAZIO.
      • NOTE também que MESMO que a função retorne um valor, não é obrigatório colocá-la no lado direito do sinal de atribuição.
         #include <stdio.h>
         float media; /* Variável GLOBAL */
        
         void media_nums(float num1, float num2)
         {
           media = (num1 + num2)/2;
           return;
         }
        
         main()
         {
           float aux1, aux2; /* Variáveis LOCAIS */
        
           printf("\nEntre com numero 1: ");  
           scanf ("%f",&aux1);
        
           printf("\nEntre com numero 2: ");  
           scanf ("%f",&aux2);
        
           media_nums(aux1, aux2);
           printf ("\nmedia dos 2 numeros é %f\n", media);
         }
        
  • Uma função pode ser chamada dentro de qualquer expressão. Por exemplo, para o caso em que a função media_nums() retorna um valor, ela poderia ser usada como:
    • NOTE agora que a função media_nums() retorna um float, podemos utilizar este retorno no contexto em que a função foi chamada que é o que ocorre dentro do printf abaixo
    • NOTE também que MESMO que a função retorne um valor, não é obrigatório que este valor este sendo recebido por alguma variável, uma função que possui retorno e este retorno não está sendo recebido se comportará como um procedimento (sem retorno).
    • OBS.: Variáveis LOCAIS e GLOBAIS podem ter o mesmo nome. A variável LOCAL terá preferência no uso.
      #include <stdio.h>
       
      float media_nums(float num1, float num2)
      {
         float media_local;
       
         media_local = (num1 + num2)/2;
         return media_local;
      }
       
      int main()
      {
         float media, aux1, aux2;
         printf("\nEntre com numero 1: ");  
         scanf ("%f",&aux1);
       
         printf("\nEntre com numero 2: ");  
         scanf ("%f",&aux2);
       
         media = media_nums(aux1, aux2);
         printf ("\nmedia dos 2 numeros multiplicada por 10 é %f\n", 10*media_nums(aux1, aux2));
      }
      
Gerando números pseudo-aleatórios
  • Computadores executam instruções, portanto, não são realmente capazes de criar números aleatoriamente. Porém existem artifícios de se criar números que variam conforme determinadas condições como sequencias variáveis associadas a data e hora atual por exemplo, a isso chamamos de números pseudo aleatórios. Para conseguir este recurso em C podemos utilizar a função rand() da stblib associada a função srand que configura uma semente de aleatoriedade a função rand, o resultado pode ser conferido no código a seguir:
    #include <stdlib.h>
    #include <time.h>
    #include <stdio.h>
    
    int main(void) {
    
        srand( (unsigned)time(NULL) );
        printf("Numero gerado: %d\n", rand() % 10); //Gera números que variam de 0 a 9
    
        return 0;
    }
    
Tabela ASCII
  • Os computadores armazenam todos os tipos de dados na forma numérica, incluindo letras. Para apresentar textos em tela neste caso o programa em C precisa conhecer o tipo de dado que está escrito em memória, sendo um dado do tipo caractere (ou string) ele será então tratado como uma letra. O conjunto de letras, números e símbolos imprimíveis está sintetizado na tabela ASCII.
    • Para ver a tabela, acesse o link Tabela ASCII
    • Observe que as letras maiúsculas variam de 65 ('A') até 90 ('Z')
    • Observe também que as letras minúsculas estão em outra faixa, variam de 97 ('a') até 122 ('z')
    • Os números imprimíveis também tem seus representantes, variam de 48 ('0') a 57 ('9')
    • Caracteres especiais como '$', '%', '*', '+' também estão relacionados na tabela
    • A tabela também apresenta códigos de caracteres não imprimíveis como o 9 (TAB), 13 ('\r' presente no ENTER)
    • Para representações de outros caracteres pode ser necessário acessar a extensão da tabela ASCII, podendo-se obter 'Ç' e caracteres acentuados
    • A tabela que está sendo apresentada esta em acordo com o padrão ISO 8859-1 e Microsoft® Windows Latin-1, outros caracteres podem ainda ser obtidos se alterado o padrão de codificação
Dicas para resolução dos exercícios da lista 7
  • Dicas para resolução dos exercícios da lista
    1. Uma forma de resolver é criando dois vetores de inteiros (ex.: l1[26] e l2[26]). Então, recebidas as palavras 1 e 2, inicia-se a decomposição da palavra 1. Em um laço, para cada 'A' ou 'a' encontrado incrementamos uma vez o l1[0], para cada 'B' ou 'b' incrementamos l1[1] e assim por diante. O mesmo será feito com a palavra 2 em outro laço. Na prática, se a letra que está sendo analisada for maiúscula (pela tabela ASCII de 65 a 90, o índice de "lx" que deve ser incrementado é letra-65, e se form minúscula é letra-97). O resultado será dois vetores contendo a quantidade de vezes que a letra se repetiu. Um terceiro laço compara l1[0] com l2[0] e assim por diante. Havendo qualquer divergência, NÃO é um anagrama. Para simplificar solicite ao usuário para digitar palavras sem acentos (algoritmos com tratamento de palavras acentuadas terão bônus).
    2. Para resolver basta criar um laço para checar cada caractere, se o usuário solicitou converter para maiúsculas, por exemplo, e a letra que está sendo indicada no laço está entre 97 e 122, então basta subtrair de 32 o valor da letra que esta será convertida para maiúscula. Se a letra já está entre 65 e 90, não precisa alterar. Para criar as funções faça antes a prototipagem, facilitará a codificação.
    3. A solução deste é muito parecida com o primeiro exercício, porém este pede que o programa seja capaz de tratar maiúsculas e minúsculas juntas. Então, basta fazer uma "normalização" primeiro, que é o processo do exercício 2, por exemplo, converta tudo para maiúscula primeiro e depois faça a contagem, isso garante que maiúsculas e minúsculas sejam tratadas da mesma forma.
    4. Para este exercício é sugerido criar um vetor char nomes[n][10], sendo 10 o tamanho máximo de um nome e 'n' a quantidade de alunos que é variável. Crie então um outro vetor de float notas [n][5] onde 5 são as notas. O programa então começa solicitando a quantidade de alunos gravando em 'n'. Depois recebe nomes e notas. Pode-se utilizar uma variável de apoio para computar o índice do aluno com a maior nota que será no final do processo impresso. Durante o calculo das médias já pode-se imprimir os alunos que ficaram em recuperação.
    5. Para resolver este, utilize as instruções presentes nesta página da wiki. Atente-se que para se obter um número de 1 a 60, um jeito fácil é pegar o resto da divisão por 60 do resultado de rand(), porém o resultado desta matemática será um número de 0 a 59, então no final basta somar com 1 para se obter entre 1 e 60. Para garantir que os 6 números sejam distintos pode-se criar um laço while que dentro está a geração do randômico e teste se este número já está presente no vetor de int numeros[6], se sim, basta não incrementar a variável de apoio e retornar ao início do laço para que um novo número seja gerado.
Obtendo o código de um caractere UTF-8
Para obter o código UTF-8 de um caracter especial (retorno 0xC3) é necessário executar um segundo scanf, conforme exemplo:
#include <stdio.h>

int main()
{   
  unsigned char c;
  printf("Digite um caracter especial ou não:\n");
  //Primeiro scanf
  scanf("%c", &c);
  //Se for um caracter especial
  if(0xC3 == c)
  {
    //Segundo scanf para obter segunda codificação deste char UTF8
    scanf("%c", &c);
    printf("Digitado um caracter especial: 0x%x\n",c);
  }
  else
  {
    printf("Digitado um caracter convencional: 0x%x\n",c);
  }  
  return 0;
}

Tipos de dados compostos

Estruturas
Estruturas

Assim como o vetor a estrutura é um conjunto de dados, mas traz uma vantagem: a possibilidade de possuir "campos" de diferentes tipos de variáveis. Por exemplo, a struct TPessoa poderia ter os campos nome (char[40]) e idade (int).

A declaração genérica da estrutura é:

struct TNome_do_tipo {
  //variável 1
  //variável 2
  //variável N
} nome_instancia;
  • Convencionalmente damos ao tipo da estrutura um nome "TNome_do_tipo", onde "T" representa Tipo e a letra seguinte também vem em maiúscula
  • nome_instancia representa a instancia de variável (do tipo struct) que será alocada em memória, esta declaração também pode ser um vetor "nome_instancia[10]", por exemplo
  • Em uma declaração é necessário ao menos definir um dos parâmetros "TNome_do_tipo" ou "nome_instancia são opcionais"


#include <stdio.h>

struct TUsuario /* struct TUsuario é o nome do tipo que está sendo criado */
{ 
  char userID[20];
  char senha[20];
} Usuario; /* aqui é definida uma variável do  tipo struct TUsuario */

struct TUsuario TabelaUsuario[20];

main()
{
  scanf("%s", Usuario.userID);
  scanf("%s", Usuario.senha);
  scanf("%s", TabelaUsuario[10].userID);
  scanf("%s", TabelaUsuario[10].senha);
}

Neste exemplo, foi definido um tipo (modelo) para o registro (struct TUsuario) e foi criada uma variável chamada Usuario a partir deste tipo. Na sequência foi criada mais uma variável (um vetor de estruturas) chamada TabelaUsuario. Note que basta usar as palavras chave struct Usuario para criar novas variáveis. O tipo completo é definido uma única vez no início.

Exercícios
  1. Criar um programa que define uma struct para armazenamento do nome e das notas bimestrais de um aluno. Atualizar a estrutura usando o scanf.
  2. Alterar o programa para que ele calcule e imprima a média de cada aluno.
#include <stdio.h>

#define NUM_MAX 3

struct TAluno {
  char nome[30];
  char matricula[11];
  float b1,b2,b3,b4;
} Turma[NUM_MAX];

void print_aluno(struct TAluno aux)
{
  printf("Nome -> %s\n", aux.nome);
  printf("Matrícula -> %s\n", aux.matricula);
  printf("Bimestre 1 -> %f\n", aux.b1);
  printf("Bimestre 2 -> %f\n", aux.b2);
  printf("Bimestre 3 -> %f\n", aux.b3);
  printf("Bimestre 4 -> %f\n", aux.b4);          
}

main()
{
  int i;
  
  for(i=0;i<NUM_MAX;i++) {
  	printf("Entre com o nome do aluno\n");
  	scanf("%s", Turma[i].nome);
  	printf("Entre com a matrícula do aluno\n");
  	scanf("%s", Turma[i].matricula);
  	printf("Entre com a nota do bimestre 1\n");
  	scanf("%f", &Turma[i].b1);
  	printf("Entre com a nota do bimestre 2\n");
  	scanf("%f", &Turma[i].b2);
  	printf("Entre com a nota do bimestre 3\n");
  	scanf("%f", &Turma[i].b3);
  	printf("Entre com a nota do bimestre 4\n");
  	scanf("%f", &Turma[i].b4);
  }
  for(i=0;i<NUM_MAX;i++) {
    printf("=========== Aluno %d ============\n", i);  
  	print_aluno(Turma[i]); 
  }      
}
Copiando Estruturas

O exemplo a seguir demonstra como se pode copiar uma variável struct para outra do mesmo tipo.

#include <stdio.h>
  
struct THoras{
   int hora;
   int minuto;
   int segundo;
};

struct THoras Ontem = {2,10,57};

void main()
{
     struct THoras Hoje;
     Hoje = Ontem;

     printf("Hora hoje = %d, Minuto hoje = %d e Segundo hoje %d\n", Hoje.hora, Hoje.minuto, Hoje.segundo);
}
Estruturas dentro de estruturas

Vamos ver um exemplo com estruturas definidas dentro de estruturas:

#include <stdio.h>
  
struct TEndereco{
     char rua[50];
     char numero[10];
};

struct TCidadao{
  char nome[50];
  char cpf[20];
  struct TEndereco endereco;
  int num_filhos;
};

void main()
{
  struct TCidadao Cidadao;

  printf("Entre com o nome\n");
  scanf ("%s",Cidadao.nome);

  printf("Entre com o cpf\n");
  scanf ("%s",Cidadao.cpf);

  printf("Entre a rua\n");
  scanf ("%s",Cidadao.endereco.rua);

  printf("Entre a numero\n");   
  scanf ("%s",Cidadao.endereco.numero);

  printf("Entre com o número de filhos\n");
  scanf ("%d",&Cidadao.num_filhos);
  
}
Iniciando structs na definição

Como toda variável, é possível dar valores para uma variável do tipo struct definida no programa:

#include <stdio.h>
 
struct TEndereco {
     char rua[50];
     int numero;
};
 
struct TCidadao{
  char nome[50];
  char cpf[20];
  struct TEndereco endereco;
};
 
int main(void)
{
  //Inicializando com parâmetros em sequencia (ordem tem que ser respeitada)
  struct TCidadao CidadaoMaria = {"Maria","42342342234",{"Rua AlfaBeta",145}};
  //Inicializando com parâmetros via campo (não é necessário respeitar qualquer ordem)
  struct TCidadao CidadaoJose = {.cpf = "1234567890", .endereco.numero = 541,.nome = "Jose",.endereco.rua = "Rua GamaDelta"};

  printf("Rua do cidadao %s = %s\n", CidadaoMaria.nome, CidadaoMaria.endereco.rua);
  printf("Rua do cidadao %s = %s\n", CidadaoJose.nome, CidadaoJose.endereco.rua);
 
}
Passando estruturas como parâmetro e retornando estruturas

Se não for usado o operador "&" , um parâmetro que é estrutura será passado por cópia. Não apresentaremos agora a passagem por endereço pois necessita do conceita de ponteiro. Observe o exercício abaixo.

#include <stdio.h>
  
struct TEndereco{
     char rua[50];
     char numero[10];
};

struct TCidadao{
  char nome[50];
  char cpf[20];
  struct TEndereco endereco;
  int num_filhos;
};

void print_struct (struct TCidadao aux)
{
  printf("nome=%s cpf=%s\n", aux.nome, aux.cpf);
  printf("endereço inicial do aux %p\n", &aux);
}

void main()
{
  struct TCidadao Cidadao;

  printf("Entre com o nome\n");
  scanf ("%s",Cidadao.nome);

  printf("Entre com o cpf\n");
  scanf ("%s",Cidadao.cpf);

  printf("Entre a rua\n");
  scanf ("%s",Cidadao.endereco.rua);

  printf("Entre a numero\n");   
  scanf ("%s",Cidadao.endereco.numero);

  printf("Entre com o número de filhos\n");
  scanf ("%d",&Cidadao.num_filhos);
  
  print_struct(Cidadao);

  printf("endereço inicial do Cidadao %p\n", &Cidadao);
}
Exercícios
  1. Implementar um contador de acesso que permita bloquear o usuário após 3 tentativas seguidas. Note que caso o usuário acerte a senha, este contador deverá ser zerado.
  2. Implementar uma funcionalidade do administrador para desbloquear o usuário bloqueado.
  3. No programa de controle de senha inserir um campo na estrutura do usuário de forma a acomodar uma mensagem de boas vindas particularizada para cada usuário. A mensagem "DEFAULT" é Bom dia!
  4. Implementar na função administrar a inserção da mensagem no exercício anterior.
  5. Na solução acima criar uma função que procura usuário na tabela (já que este código é utilizado em mais do que um luga). A função deve receber o UserID a ser procurado e deve retornar um inteiro correspondente ao índice do usuário encontrado ou -1 se não for encontrado.
Unions
Union é um recurso do C que permite declarar um conjunto de dados que irá ocupar um mesmo espaço. É bastante empregado quando se deseja economizar espaço ou não se tem certeza sobre qual tipo de dado deve ser armazenado para determinada instancia. No exemplo a seguir é criada uma struct chamada TProduto e dentro destra estrutura há uma área de detalhamento do produto que é de uso genérico, para alguns produtos há campos específicos para preenchimento e outros não se tem ao certo os detalhes, portanto fica um campo de uso geral.
 
#include <stdio.h>

struct TRoupeiro{
  char cor[20];
  int volume;
  float peso;
};

struct TProduto{
  int id;
  char nome[20];
  union {
    struct TRoupeiro roupeiro;
    char descricao_generica[sizeof(int)+sizeof(float)+20];
  };
};
 
int main(void)
{
  struct TProduto vaso_decorativo = {
      .id = 2,.nome = "Vaso decorativo 1",
      .descricao_generica = "em vidro - peça única"
  };
  
  struct TProduto guarda_roupas_solteiro = {
      .id = 1,.nome = "Roupeiro 3 portas",
      .roupeiro.cor = "CZ", .roupeiro.volume = 304,.roupeiro.peso = 50.0
      
  };
  printf("nome=%s, descrição=%s, cor=%s, volume=%d, peso=%f\n", 
        guarda_roupas_solteiro.nome, 
        guarda_roupas_solteiro.descricao_generica,
        guarda_roupas_solteiro.roupeiro.cor,
        guarda_roupas_solteiro.roupeiro.volume,
        guarda_roupas_solteiro.roupeiro.peso
  );
  printf("nome=%s, descrição=%s, cor=%s, volume=%d, peso=%f\n", 
        vaso_decorativo.nome, 
        vaso_decorativo.descricao_generica,
        vaso_decorativo.roupeiro.cor,
        vaso_decorativo.roupeiro.volume,
        vaso_decorativo.roupeiro.peso
  );
}
  • Observe que a union é feita de duas entre uma struct e um vetor de caracteres. O autor neste código criou este vetor baseado no tamanho que ocupa a struct aproveitando todo o espaço já alocado, porém o C não exige que os dados ocupem o mesmo espaço, neste caso a alocação ocorrerá em relação a maior estrutura.
  • Observe que ao imprimir os valores da instancia "guarda_roupas" o "descricao_generica" apesar de não ter sido formalmente preenchido, foi indiretamente preenchido quando no caso "roupeiro.cor" recebeu um valor. Como este valor foi "Cinza" o C escreveu um "\0" no final da string "roupeiro.cor" que acabou também servindo como final da string "descricao_generica", por isso neste print ambos os campos apresentam o mesmo valor. Porém observe também que os valores de "volume" e "peso" estão perfeitamente preservados.
  • Agora foi interessante o cado da instancia "vaso_decorativo", ela foi descrita pelo campo "descricao_generica" e apenas para fins didáticos o autor imprimiu o que teria dentro de "roupeiro.cor", "roupeiro.volume" e "roupeiro.peso". Neste caso cairam valores oriundos da constante "em vidro - peça única" que não passam de sujeira neste contexto.

Ponteiros

Ponteiros

A memória de um computador pode ser vista como um vetor de bytes. Neste espaço vimos a utilização de variáveis diversas que podem armazenar valores que podem ser obtidos do usuários, serem resultados de ariméticas e muitas outras operações. O ponteiro nada mais é que um tipo de dado igualmente armazenado em memória, porém este dado se refere a um endereço da memória, ou seja, a um outro objeto.

Este recurso é muito útil para diversos propósitos, basta pensar na própria aplicação do conceito "endereço", imagine como seria localizar uma casa em uma cidade sem haver uma forma de endereçar e armazenar os endereços das casas. Explorando esta analogia, cada lote possui um endereço e pode ter um conteúdo de diferentes tipos como uma casa, um prédio ou um conjunto de lojas, enfim, trazendo para o C seria como os tipos int, char, vetores diversos, etc.

Assim é a memória, cada byte possui um endereço. O tamanho da memória é definido pelo tamanho do barramento de endereços usado para acessá-la. Uma variável ocupa uma área da memória. Tipicamente uma variável to tipo char se utiliza de um byte. Já uma variável do tipo int pode (dependendo do sistema) usar 4 bytes contíguos.

Uma variável possui um endereço e um conteúdo (dados).
Uma variável ponteiro tem como conteúdo um endereço. Portanto a variável ponteiro possui um endereço e contém um endereço como conteúdo. Este recurso é largamente utilizado para passar parâmetros como referência (ao invés de copiar uma variável quando se quiser processá-la em alguma função), bem como para algoritmos diversos que buscam formas mais otimizadas de executar operações. Rode o código a seguir e compare com as respostas que foram obtidas:
#include <stdio.h>
int main(void) 
{
  int i = 10;
  int *p;
  long int li;
  p = &i;
  printf("Conteúdo de i:      i = %d\n",i);
  printf("Endereço de i:     &i = %p\n",&i);
  printf("Conteúdo de p:      p = %p\n",p);
  printf("Endereço de p:     &p = %p\n",&p);
  printf("Conteúdo apontado: *p = %d (conteúdo do endereço apontado por p)\n",*p);
  printf("Tamanho do ponteiro   = %li bytes\n",sizeof(p));
  printf("Tamanho do lont int   = %li bytes\n",sizeof(li));
  printf("Tamanho do int        = %li bytes\n",sizeof(i));
  return 0;
}

Resposta obtida através do gcc em uma máquina Linux Ubuntu:

Conteúdo de i:      i = 10
Endereço de i:     &i = 0x7ffeb25859e4
Conteúdo de p:      p = 0x7ffeb25859e4
Endereço de p:     &p = 0x7ffeb25859e8
Conteúdo apontado: *p = 10 (conteúdo do endereço apontado por p)
Tamanho do ponteiro   = 8 bytes
Tamanho do lont int   = 8 bytes
Tamanho do int        = 4 bytes
  • Observações:
    • "&i" e "p" são iguais, isso porque "p" tem como conteúdo um endereço (neste caso o endereço de "i")
    • O conteúdo de "i" (10) é igual "*p", que exatamente está extraindo o conteúdo da variável apontada (é o próprio "i")
    • O endereço de "p" (&p) é um valor próprio o que prova que p é também uma variável alocada na memória armazenando um valor próprio
    • Nesta máquina o ponteiro está representado por uma variável inteira de 8 bytes (uintptr_t).


Resposta obtida através do codechef.com (gcc-4.9.2):

Conteúdo de i:      i = 10
Endereço de i:     &i = 0xbfadc2b8
Conteúdo de p:      p = 0xbfadc2b8
Endereço de p:     &p = 0xbfadc2bc
Conteúdo apontado: *p = 10 (conteúdo do endereço apontado por p)
Tamanho do ponteiro   = 4 bytes
Tamanho do lont int   = 4 bytes
Tamanho do int        = 4 bytes
  • Observações:
    • O endereço de "p" e "i" são completamente diferentes da resposta anterior. Isso ocorre em diferentes máquinas e cada vez que o programa for rodado deverá também gerar novos endereços. O endereço é atribuído pelo sistema operacional que por várias condições naquele instante disponibilizou estes endereços ai listados.
    • Nesta máquina o ponteiro está representado por uma variável inteira de 4 bytes (uintptr_t), o tamanho de uma variável ponteiro varia conforme a plataforma.
Ponteiro para inteiro

Observe o programa abaixo. A variável p é um ponteiro para inteiro. Isto significa que ela pode armazenar um endereço de um inteiro.

#include <stdio.h>

main()
{
  int x;
  int *p;

  x=5;
  printf("Valor de x antes = %d\n", x);
  
  p = &x;
  *p=10;

  printf("Valor de x depois = %d\n", x);
  printf("Valor de p = %p\n", p);
}

Observe que para se referenciar o conteúdo da posição de memória apontada por p deve-se usar o asterisco: *p


EXERCÍCIO 1
Considere o programa abaixo:
main()
{
  int x=10;
  int y, *p;

}

Complete o código para copiar o conteúdo de x para y, sem que qualquer variável apareçam no lado esquerdo de um sinal de atribuição. Ou seja, sem envolver diretamente x e y.

EXERCÍCIO 2
Tente inferir qual seria o valor da variável y no final do programa abaixo:
main()
{
  int x,y,w,*p1,*p2;
  x = 20;
  w = 30;
  p1 = &x;
  p2 = &w;
  y = *p1 + *p2;
}
EXERCÍCIO 3
Tente inferir qual seria o valor da variável y no final do programa abaixo:
main()
{
  int x,y,w,*p1,*p2, *p3;
  x = 20;
  w = 30;
  p1 = &x;
  p2 = &w;
  y = *p1 + w;
  p3 = &y;
  *p3 = *p3 + 10;
  y = *p1 + *p2 + *p3;
}
EXERCÍCIO 4
Qual seria o valor das variáveis y e x no final do programa abaixo:
#include <stdio.h>
void main()
{
	int x,y;
	int *p;
	y=0;
	p=&y;
	x=*p; 	
	x=4; 	
	(*p)++;	
	x--;	
	(*p) += x; 
	printf("\ny=%d x=%d\n",y,x);
}
Ponteiro para char

Os ponteiro para char são muito utilizados pois permitem apontar para strings. A ideia é que ele aponte para o primeiro caracter (char) da string. Veja o exemplo abaixo.

#include <stdio.h>
main()
{
   char x[10]="ifsc";
   char *p;
   p = &x[2];
   printf("x[2] = %c\n", *p);
   p = x;
   printf("string %s\n", p);
   while (*p!=0) {
       printf("Endereco %p conteúdo %c\n", p,*p);
       p++;
   }	
}

Neste foi usado o incremento de um ponteiro, o que implica em adicionar ao endereço armazenado em p uma quantidade relativa ao tamanho do tipo apontado. No caso é 1 (tamanho de um char é um byte).

EXERCÍCIO
Sem executar o programa abaixo, determine o valor de y no final do programa:
main()
{
   char x[10]="ifsc";
   char *p, y;
   p = x + 2;
   y= *p;
}
Apontando para um vetor de inteiros


Da mesma forma que usamos um ponteiro para char para apontar uma string, podemos fazer um ponteiro para int apontar para para um elemento de um vetor de inteiros.

#include <stdio.h>
main()
{
   int x[10]= {0,1,2,3,4,5,6,7,8,9};
   int *p;
   int i;
   p = x;
   i=0;
   while (i<10) {
       printf(" endereco %p e conteudo %d\n", p, *p);
       p++;
       i++;       
   }	
}

OBSERVE que p++ incrementa em 4 unidades.

Apontando para estruturas

Ponteiros podem apontar para qualquer "objeto" de qualquer tipo. Vamos verificar como é possível apontar para uma estrutura:

#include <stdio.h>
struct TRegistro {
   char nome[20];
   int idade;
} Tabela[4] = {
          {"joao",18,},
          {"maria",18,},
          {"jose",19,},
          {"lara",17,},
};
struct TRegistro *p;
main()
{
  p = &Tabela[3]; /*p aponta para o registro 3 da tabela */
  printf("O nome na posição 3 é %s e idade = %d\n", p->nome,p->idade);
}

NOTE que o uso de p->nome é uma alternativa ao uso de (*p).nome

No primeiro caso pode-se ler: o campo nome do objeto que é apontado por p.

Retornando uma estrutura em uma função

No exemplo a abaixo a função RetornarStruct() retorna um ponteiro para uma estrutura. O cuidadado que se deve ter é que a função não deveria apontar para uma estrutura que foi criada localmente na função!

#include <stdio.h>
struct TRegistro {
   char nome[20];
   int idade;
} Tabela[4] = {
          {"joao",18,},
          {"maria",18,},
          {"jose",19,},
          {"lara",17,},
};
struct TRegistro *p;
struct TRegistro * RetornarStruct(int indice)
{
  return &Tabela[indice];
}
main()
{
  p = RetornarStruct(2); /*p aponta para o registro 3 da tabela */
  printf("O nome na posição 2 é %s e idade = %d\n", p->nome,p->idade);
}
Passando uma estrutura como parâmetro
#include <stdio.h>
struct TRegistro {
   char nome[20];
   int idade;
} Tabela[4] = {
          {"joao",18,},
          {"maria",18,},
          {"jose",19,},
          {"lara",17,},
};
struct TRegistro *p;
void MudarStruct(struct TRegistro *p1, int indice)
{
  Tabela[indice] = *p1;
}
main()
{
  struct TRegistro aux = {"luisa",16};
  MudarStruct(&aux,2);
  p = &Tabela[2];
  printf("O nome na posição 2 é %s e idade = %d\n", p->nome,p->idade);
}

Mais sobre a função main()

Início e fim do programa
O programa inicia pela primeira instrução contida na função main() e também se encerra na última instrução. O retorno padrão da função main é um int que representa um código de erros reconhecidos por muitos sistemas operacionais. Se o programa terminou sua execução corretamente o retorno deverá ser 0 (zero).
 int main(void)
 {
  //Programa
  return 0;
 }
  • A omissão do retorno da função main (por padrão int) ou utilização de outro tipo de retorno pode ser feita no C99 porém desta forma o código não fica portável a alguns padrões de C e não estaria adequado ao retorno de um programa que convencionalmente é esperado por sistemas operacionais.
  • Quando o programador não especifica este retorno o compilador provavelmente estará atribuindo o retorno como int.
  • Programas que devem ficar rodando indefinidamente normalmente definem um laço infinito na função main para que esta não alcance o final do código salvo se for dado um comando de terminação, por exemplo.
Os argumentos argc e argv

Os argumentos argc e argv

A função main() pode ter parâmetros formais, mas o programador não pode escolhores quais serão eles. A declaração que se pode ter para a função main() é: int main (int argc, char *argv[]); Exemplo: Escreva um programa que faça uso dos parâmentros argv e argc. O programa deverá receber da linha de comando o dia, mês e ano correntes (dd/mm/aaaa), e imprimir a data em formato apropriado. Veja o exemplo, supondo que o executável se chame data: $ data 07 06 2016 O programa deverá imprimir:

$ 07 de junho de 2016
#include <stdio.h>
#include <stdlib.h>
void main(int argc, char *argv[])
{
  int mes;
  char *nomemes [] = {"janeiro","fevereiro","março","abril","maio","junho","julho","agosto","setembro","outubro","novembro","dezembro"};
  if(argc == 4) /* Testa se o numero de parametros fornecidos esta' (nome do programa, o dia, o mes e os dois ultimos algarismos do ano */
  {
    /* argv contem strings. A string referente ao mes deve ser 
     * transformada em um numero inteiro. A funcao atoi esta sendo 
     * usada para isto: recebe a string e transforma no inteiro equivalente 
     */
    mes = atoi(argv[2]);
    if (mes<1 || mes>12) /* Testa se o mes e' valido */
      printf("Erro!\nUso mes: mm, deve ser de 1 a 12.\n");
    else
      printf("\n%s de %s de %s\n\n", argv[1], nomemes[mes-1],argv[3]);
  }
  else
    printf("Erro!\nUso: dd/mm/aaaa, devem ser inteiros, ou estão faltando.\n");
}

A função exit

A função exit

Uma alternativa a terminação do programa chegando ao fim da função main é a função exit da biblioteca <stdlib.h>. Para esta função deve-se passar um argumento inteiro que tem o mesmo significado do código de retorno da função main, portanto exit(0) representa uma terminação normal, alternativamente exit(EXIT_SUCCESS). Para representar uma terminação anormal pode-se utilizar exit(EXIT_FAILURE).

Diretivas de pré-compilação

Uma diretiva de pré-compilação é um código processado pelo pré-compilador que "prepara", então, o código que será efetivamente compilado. Nesta execução pode haver definições, mudanças de comportamento do compilador e mesmo blocos lógicos que decidirão o que será ou não compilado. As diretivas de pré-compilação são úteis para configurações diversas que o compilador precisa conhecer para poder gerra o código objeto como também são úteis para criar "macros" diversas que podem tornar processos mais simples ou tornar o código mais inteligível.

#include

O pré-processador busca o conteúdo de um outro arquivo e traz este código para que o compilador possa realizar a compilação e posterior link.

#include <stdio.h>
int main(void) {
	int i = 20;
	printf("i = %d\n",i);
	return 0;
}

No exemplo acima a função "printf" está definida na biblioteca "stdio", portanto, é necessário incluir este código para que o compilador passe a "conhecer" o que é a função printf e possa então chamar este código quando esta função é evocada. Quando a biblioteca é incluída com sinais de maior e menor (< e >) envolvando o nome do arquivo significa que o pré-compilador deve buscar esta biblioteca nos diretórios listados como endereços de biblioteca do compilador. Portanto precisa estar definido nos arquivos de configuração do compilador. O diretório onde se encontra a libc (biblioteca padrão do C) já vem por padrão listada nestes endereços. Quando a biblioteca é incuída entre aspas duplas (" ") envolvendo o nome do arquivo significa que o pré-compilador deve procurar em um endereço específico. Se o caminho não está todo definido será o endereço relativo, havendo apenas o nome do arquivo da biblioteca será buscado no diretório em que se encontra o código c que está sendo compilado.

#define

A diretiva "define" cria uma macro que é uma substituição de um valor por um código ou lista de códigos de programa. Há algumas vantagens de se utilizar macros:

  • O programa fica mais fácil de ler e entender pois pode-se substituir códigos complexos por textos auto-explicativos (exemplo: #define PI 3.14159)
  • Fica mais fácil de modificar já que o identificador pode se repetir várias vezes no código, bastando modificar o #define para modificar todo o código.
  • Evita inconsistências e erros de digitação possíveis como por exemplo definindo PI como 3.14159 e utilizando esta definição garante-se que este valor será utilizado em todo o programa igualmente, evita o engano de usar em algum lugar 3.1416, por exemplo.
#include <stdio.h>
#define qtnotas 5
int main(void) 
{
  float notas[qtnotas], soma=0;
  int i;
  for(i = 0;i < qtnotas;i++)
  {
    printf("Digite a nota = %d\n",i+1);
    scanf("%f",&notas[i]);
    soma += notas[i];
  }
  printf("A média é: %.1f\n",soma/qtnotas);
}

Observe acima como o uso da macro qtnotas traz benefícios como os listados anteriormente.

Trabalhando com funções

Usando ponteiros como parâmetros de entrada saída de funções
Enviar um ponteiro a uma função tem diversas aplicações, uma delas é a de evitar redundância de dados e realizar a leitura de informações "diretas da fonte". Estas informações quando são de grande volume também poderiam requisitar um grande volume de memória para copiar, então mais um motivo para se passar a referência (ponteiro). Observe como podemos usar ponteiros na passagem de parâmetros:
#include <stdio.h>
void str_cpy(char *pdest, char *pfonte)
{
   while (*pfonte!=0) {
        *pdest++ = *pfonte++;
   }
   *pdest = 0;
}
int str_len (char *p)
{
   int i=0;
   while (*p++!=0)
	i++;
   return i;
}
main()
{
   char fonte[10]="ifsc";
   char destino[10];
   str_cpy(destino, fonte);
   printf("string destino = %s\n", destino);
   printf("tamanho de dest = %d\n", str_len(destino));
}
Este recurso também tem especial utilidade quando se deseja que a função retorne mais de um valor. Como sabemos a função só pode ter um retorno. Utilizando ponteiro, portanto, pode-se passar um endereço de referência para que a função utilize este espaço de memória para copiar um resultado que será posteriormente aproveitado pela função invocadora:
#include <stdio.h>
void alfa(int *p)
{
  *p=10;
}
main()
{
  int x;
  x =5;
  printf("Valor de x antes da chamada de alfa = %d\n", x);
  alfa(&x);
  printf("Valor de x depois da chamada de alfa = %d\n", x);
}
Exercícios
  1. Construir uma função que imprime a Tabela usando ponteiros. A função deve receber como parâmetro um ponteiro para o início da tabela e o tamanho da tabela.
    Implementar a função str_cat que concatena duas strings usando ponteiros.
    Ordenar valores de um vetor de inteiros passando por referencia o ponteiro para esse vetor.
    Implememtar um programa que recebe 3 parâmetros na linha de comando: dois números reais e um operador (char). Operador pode ser + ou menos. O programa deve mostrar o resultado da operação.
Exemplo: calcula 3.5 + 2.6
para usar a função atof para converter string em float.
  1. Implementar um programa chamado cmpcadeia que testa se duas strings passadas na linha de comando são iguais. O programa deve imprimir uma mensagem indicando se são iguais ou diferentes. Usar a função strcmp da biblioteca. Caso sejam passados mais ou menos que dois parâmetros o programa deve se encerrar mostrando uma indicão do tipo:
cmpcadeia: dois parametros devem ser passados.
Referências Complementares
Exercício de preparação para a prova 2
  1. Elabore um programa em C que declara duas estruturas (TRetangulo e TTriangulo) tendo uma instancia para cada na forma global (retangulo e triangulo). Cada estrutura deve conter dois campos (base e altura, ambos unsigned int). Na função main atribua valores quaisquer para as bases e alturas das formas geométricas. Por fim imprima em tela os valores de base, altura e área obtendo dos campos da struct sabendo que a área do retângulo é base x altura e do triangulo é base x altura dividido por 2.
  2. Crie uma funcao calculaAreaRetangulo e outra calculaAreaTriangulo para executar o calculo das áreas sem receber parâmetros e retornando a área calculada na forma unsigned int. A função deve ser chamada de dentro das chamadas de impressão dos valores já feitas no main. Declare os protótipos de ambas as funções no início do arquivo ".c".
  3. Mova as declarações de instancia (retangulo e triangulo) originalmente criadas como globais para dentro do main e modifique as funções de cálculo para que passem a receber o ponteiro da struct como parâmetro.
  4. Declare as variáveis retangulo e triangulo como vetores de structs (com quantidade definida por MAX_R e MAX_T respectivamente). Altere o código para que os valores de base e altura sejam obtidos do usuário (crie loops para receber os valores e calcular as áreas utilizando as funções já existentes)

Alocação dinâmica de memória

Alocação dinâmica de memória

As estruturas de dados em C são normalmente de tamanho fixo. Mesmo num vetor de tamanho variável, seu tamanho apesar de ser determinado em tempo de execução, seu tamanho se mantém fixo até que seja destruído. Nos exemplos que trabalhamos a maioria tinha número de registros fixo, esta situação é bastante restringente. Em um programa de registros de notas de alunos, por exemplo, se determinado um número de registros pequeno pode-se deparar com a necessidade de aumentar este vetor, havendo necessidade de recompilação. Se por outro lado for alocado um grande espaço de memória, pode-se estar desperdiçando bastante espaço caso fiquem obsoletos. Para resolver este problema há funções de alocação dinâmica de memória, que permitem que seja alocada e desalocada memória conforme crescimento ou redução da necessidade de espaço. As funções mais conhecidas são (http://en.wikipedia.org/wiki/C_dynamic_memory_allocation):

  • malloc: aloca n bytes de memória;
  • free: libera memória;
  • realloc: realoca memória
  • calloc: aloca n bytes de memória e zera.
EXEMPLO
Alocação dinâmica de números inteiros (exercício puramente didático):
#include <stdlib.h>

main()
{
  int *px, *py;
  int resultado;
  px = (int *) malloc(sizeof(int));
  *px = 5;
  py = (int *) malloc(sizeof(int));
  *py = 2;
  resultado = *px + *py;

  free (px);
  px = NULL;
  free (py);
  py = NULL;
}

Observe que há duas chamadas de alocação utilizando "malloc", onde são alocados a quantidade de bytes que uma variável "int" ocupa (normalmente 4 bytes). Conforme consta na documentação lincada acima, o malloc retorna um void*, que é um ponteiro genérico. Para que este ponteiro seja tratado como um ponteiro de int (que é nossa necessidade neste código) é realizado um typedef de (int*). Após as alocações são escritos e lidos valores destes espaços de memória através do apontamento de "px" e "py". Por fim, é chamada a função "free" que recebe como argumento um ponteiro e libera o bloco de memória apontado por e este ponteiro.

A função free deve ser chamada sempre que a memória alocada (por malloc, calloc ou realloc) não for mais necessária. A não desalocação correta de memória pode causar "vazamentos de memória" (ou memory leak). Uma chamada de free em um espaço de memória já desalocado poderá também causar falha, trata-se de um comportamento não previsível, possivelmente uma falha de segmentação.

Após desalocar o bloco de memória apontado por um ponteiro, setar este ponteiro para NULL é uma boa prática. Esta ação evita que acidentalmente será tentado utilizar ou desalocar um espaço de memória já livre o que poderia causar falhas graves. Esta prática de "anular" o ponteiro transforma este em um "ponteiro nulo" (ou NULL pointer), sendo um tipo especial que não aponta para lugar nenhum (nulo não é um endereço de memória válido), isso significa que uma desalocação em um ponteiro nulo não deverá ter nenhum efeito.

As funções malloc e calloc são muito parecidas, o efeito prático é que o calloc além de alocar a memória também "zera" este espaço. O ato de "zerar" a memória faz do calloc uma função menos performática, isso ocorre por duas razões: o simples processo de escrever "zeros" é dispendioso e também o ato de escrever algo na memória obriga o sistema operacional a "tocar" na memória e realizar possíveis ações de swap de outros processos. No caso do malloc estas ações só ocorrerão quando houver real necessidade de escrever naquele determinado espaço de memória.

EXEMPLO
Alocando memória com calloc

Diferente da função malloc, calloc além de inicializar os espaços de memória ainda atribui o valor 0 (zero) para cada um deles.

Observe o calloc sendo realizado no início do loop, garantindo que as variáveis iniciem em zero
#include <stdio.h>
#include <stdlib.h>
 
int main(void){
 
  int *valores, *aux;
  int qtd, i, j, vlr;
 
  printf("\nEntre com a quantidade de números: ");
  scanf("%d", &qtd);
 
  if(qtd == 0) exit(0);
 
  for (j=0;j<2;j++)
  {
    //Aloca um vetor com a quantidade recebida em qtd
    valores = (int *) calloc(qtd, sizeof (int));

    printf("Rodada %d:\n", j);

    //Imprime o endereco e valor (zero) para a quantidade de números informada
    aux = valores;
    for(i = 1; i <= qtd; i++){
        printf("Situação inicial da memória: %p\t%d\n", aux, *aux);
        aux++;
    }

    aux = valores; //Ponteiro para o primeiro espaço de memória
    for(i = 1; i <= qtd; i++){
        printf("Digite o número %d ->: ", i);
        scanf("%d", &vlr);
        *aux = vlr;
        aux++;
    }
    aux = valores; //Retorna à posição inicial no mapa de memória
    for(i = 1; i <= qtd; i++){
        printf("Situação final da memória  : %p\t%d\n", aux, *aux);
        aux++;
    }
    printf("\n");

    free(valores);
  } 

  return 0;
}
EXEMPLO
malloc versus calloc
Observe o malloc sendo realizado no início do loop, não há garantia de qual valor se encontra naquele espaço de memória.
#include <stdio.h>
#include <stdlib.h>
 
int main(void){
 
  int *valores, *aux;
  int qtd, i, j, vlr;
 
  printf("\nEntre com a quantidade de números: ");
  scanf("%d", &qtd);
 
  if(qtd == 0) exit(0);
 
  for (j=0;j<2;j++)
  {
    //Aloca um vetor com a quantidade recebida em qtd
    valores = (int *) malloc(qtd * sizeof (int));

    printf("Rodada %d:\n", j);

    //Imprime o endereco e valor (zero) para a quantidade de números informada
    aux = valores;
    for(i = 1; i <= qtd; i++){
        printf("Situação inicial da memória: %p\t%d\n", aux, *aux);
        aux++;
    }

    aux = valores; //Ponteiro para o primeiro espaço de memória
    for(i = 1; i <= qtd; i++){
        printf("Digite o número %d ->: ", i);
        scanf("%d", &vlr);
        *aux = vlr;
        aux++;
    }
    aux = valores; //Retorna à posição inicial no mapa de memória
    for(i = 1; i <= qtd; i++){
        printf("Situação final da memória  : %p\t%d\n", aux, *aux);
        aux++;
    }
    printf("\n");
    free(valores);
  } 

  return 0;
}
EXEMPLO
Alocando uma estrutura e testando o memory overflow
#include <stdio.h>
#include <stdlib.h>
#define QT_INSTANCIAS 100000
 
void main()
{
  struct TTeste{
    long int x;
    long int y;
    char st[50];
  };

  struct TTeste *teste;
 
  teste = (struct TTeste *) malloc (QT_INSTANCIAS*sizeof(struct TTeste));
  //Boa pratica: se o retorno do malloc é NULL houve erro de alocação
  if (teste!=NULL) {
    printf("%li MBytes de memória alocados com sucesso!\n",QT_INSTANCIAS*sizeof(struct TTeste)/(1024*1024));
    printf("Digite qualquer tecla para prosseguir: ");
    getchar();
  } else {
      printf("Erro ao tentar alocar %li MBytes de memória!\n",QT_INSTANCIAS*sizeof(struct TTeste)/(1024*1024));
      exit(1);
  }
 
  long int i;
  struct TTeste *aux;
  aux = teste;
  for (i=0;i<QT_INSTANCIAS;i++,aux++)
  {
    aux->x=i+1;
    aux->y=aux->x*2;
    sprintf(aux->st,"%li",i*1000000);
    printf("%li: teste.x=%li, teste.y=%li e teste.st=%s\n",i,aux->x,aux->y,aux->st);
  }
  aux=NULL;
  free(teste);
  teste=NULL;
}
EXEMPLO
Alocando dinamicamente uma tabela de estruturas

Observe agora o uso da função "realloc". Como argumentos além do tamanho em bytes (idem ao malloc), também é necessário setar o ponteiro que será redimensionado. Este tamanho poderá ser maior ou menor ao tamanho anteriormente alocado, fazendo com que seja liberada memória ou alocado um bloco ainda maior. O retorno da função é o próprio ponteiro do bloco realocado. Se o retorno for nulo, a operação de alocação não foi bem sucedida.

#include <stdio.h>
#include <stdlib.h>

void main()
{
  struct TTeste{
     int x;
     int y;
  } *teste;

  if ((teste = (struct TTeste *) malloc (100*sizeof(struct TTeste)))==NULL) {
      printf("erro de alocação");
      exit(1);
  }

  teste[10].x= 5;

  if ((teste = realloc(teste, 10000*sizeof(struct TTeste)))==NULL) {
      printf("erro de alocação");
      exit(1);
  }

  teste[9000].x=20;
  free(teste);

}
EXERCÍCIOS
  1. Refazer o exemplo "Alocação dinâmica de números inteiros" tentando desalocar duas vezes o mesmo ponteiro e também tentando desalocar o ponteiro após se tornar um ponteiro nulo.
  2. Estude atentamente o exemplo "Alocando memória com calloc", por que foi criado um ponteiro aux? Crie linhas de código com printf para exibir o endereço de memória que aux tem em cada iteração do loop, o que se conclui desta observação? aux não está sendo liberado por free, o que isso significa? O que aconteceria se fosse dados free de aux também? aux e valores não estão sendo setados para NULL, que risco isso traz?
  3. Altere o exemplo "Alocando uma estrutura e testando o memory overflow" na quantidade de instancias e/ou tamanho do vetor teste.st para obervar até quanto de memória é possível alocar e quando não há memória disponível suficiente para esta operação.
  4. Rode o exemplo "Alocando uma estrutura e testando o memory overflow" com uma quantidade grande de memória sendo alocada. Enquanto são impressas as linhas mostrando que os espaços de memória estão sendo preenchidos abra um outro terminal e rode o programa chamado "free" que mostra a ocupação de memória ram. Rode o free algumas vezes seguidas para observar o aumento de uso de memória e finalmente após finalizar o programa veja a memória ser desalocada.
  5. Rode o exemplo "Alocando uma estrutura e testando o memory overflow" com uma quantidade grande de memória sendo alocada. Enquanto são impressas as linhas mostrando que os espaços de memória estão sendo preenchidos abra um outro
  6. O que aconteceria se fosse tentado acessar um espaço de memória apontado por um ponteiro nulo ou de memória já desalocada? Utilize um dos exemplos anteriores para testar ambos os casos.

Acessando arquivos em C

Introdução

Arquivos em C são tratados como "streams" que literalmente significa córrego (riacho), em computação significa dados em fluxo. Utilizado para tratar quantidade desconhecida de informações (potencialmente infinita). Por esta característica arquivos são acessados através de ponteiros do tipo FILE * (um tipo de dados declarado na biblioteca <stdio.h>).

Ainda na stdio, há três streams padrão (stdin, stdout e stderror) que não precisam ser declarados e estão prontos para uso. Funcões como scanf e getchar, por exemplo, na prática estão obtendo dados do teclado (stdin) e funções como printf e putchar estão escrevendo da tela (stdout). Eventuais falhas serão também enviadas para a tela (stderr).

A quantidade de streams que um programa mantém aberto pode ser limitado pelo sistema operacional.

A biblioteca stdio.h suporta dois tipos de arquivos: binários ou texto. Os arquivos tipo texto são compostos por caracteres humanamente compreensíveis. Já os arquivos binários são codificados. Arquivos binários podem armazenar mais informações em menor espaço e são normalmente mais fáceis de serem lidos por programas tornando-os também mais performáticos.

Operações de abertura e fechamento de arquivos
Operação abrir arquivo

A primeira ação será a de abertura de uma arquivo. É realizado com a função fopen que deve receber como parâmetros o nome do arquivo a ser aberto e o modo de abertura que especifica se será para leitura ou escrita. A diretiva restrict que aparece em ambos os argumentos não é muito relevante para o momento, basicamente está dizendo que estes espaços de memória não podem ser compartilhados (C99).

FILE *fopen(const char * restrict filename, const char * restrict mode)
Um exemplo de chamada seria
FILE * arquivo;
arquivo = fopen("IFSC.txt", "r");

Onde "r" significa somente leitura

Outro exemplo de chamada seria
arquivo = fopen("c:\\temp\\temp.txt", "w");

Onde "w" significa que o arquivo está sendo aberto para escrita Observe também o uso de "\\" e não apenas "\". Isto de deve pois algo como "C:\temp\temp.txt" teria então um "\t" que significa TAB. O uso de "\\" previne esta má interpretação. Uma outra alternativa é escrever "C:/temp/temp.txt", ou seja, com as barras invertidas. O operação fopen deve retornar um endereço válido se for bem sucedida, se falhar retornará NULL. Razões para falhas estão listadas aqui.

Modos de acesso a arquivos (arquivos tipo texto)
  • "r": Abre para leitura
  • "w": Abre para escrita (o arquivo não necessariamente precisa existir, o conteúdo possivelmente será sobrescrito caso não tratado, ver modo "a")
  • "a": Abre para escrita adicionando dados no final do arquivo (o arquivo não necessariamente precisa existir)
  • "r+": Abre para leitura e escrita
  • "w+": Abre para leitura e escrita (trunca se o arquivo existir, apagando o conteúdo anterior)
  • "a+": Abre para leitura e escrita adicionando dados no final do arquivo (o arquivo não necessariamente precisa existir)

O caractere '+' representa o modo atualização (update mode). Para arquivos binários devem ter a letra "b" associada ("rb", "wb", "ab", "r+b"/"rb+", "w+b"/"wb+", "a+b"/"ab+"). No sistema operacional Linux a especificação de abertura de arquivo como binário não trará mudança pois é utilizado apenas um caractere para representar nova linha (\n). Porém, para manter portabilidade com o Windows que utiliza dois caracteres de nova linha '\r\n' é importante utilizar o modo 'b' para arquivos binários.

Operação fechar arquivo

Permite o programa fechar um arquivo que não está mais sendo utilizado. Deve receber como parâmetro o ponteiro (FILE *) para o arquivo. Se fechar com sucesso retornará zero, caso ocorre erro retornará EOF (uma macro definida na stdio.h.

int fclose(FILE *stream);
Um exemplo de chamada seria
fclose(arquivo);
Operações de escrita de arquivos
Operação escrever texto formatado em arquivo

Escreve uma saída em um stream apontado. A string apontada no segundo argumento se parece as utilizadas na função "printf".

int fprintf(FILE * restrict stream, const char * restrict format, ...);
Outras operações de escrita
int fputc(int c, FILE *stream); //Escreve um caracter
int fputs(const char * restrict s, FILE restrict *stream); //Escreve uma string
Exemplo: Abrindo e escrevendo um arquivo
#include <stdio.h>
 
int main(void)
{
  FILE *p_arq;
  int i;
 
  if ((p_arq=fopen("IFSC.txt", "w")) == NULL) {
     printf("Problemas na abertura do arquivo\n");
     return 0;
  } else {
    printf("Arquivo aberto com sucesso. Iniciando escrita no arquivo...\n");
  }

  for (i = 0; i<10;i++) {
    /* A funcao fprintf devolve o número de bytes gravados ou EOF se houve erro na gravação */
    if((fprintf(p_arq,"Linha %d\n",i))==EOF) {
      printf("Erro ao escrever no arquivo!\n");
      return -1;
    }
  }
  printf("Fim da escrita, observe o arquivo 'IFSC.txt' criado na mesma pasta deste executável!\n");
  fclose(p_arq);
  return 0;
}

Note que se o arquivo IFSC.txt não existir, ele será criado.

Exercícios
  1. Após criar um arquivo IFSC.txt com o código acima, modifique o texto deste arquivo manualmente, escreva seu nome, por exemplo, salve e feche. Execute novamente o executável acima. O que ocorreu? O que você observa que foi realizado com o modo 'w' utilizado acima?
  2. Modifique o programa exemplo para que abra o arquivo com modo "r+", Aumente e diminua a quantidade de iterações do loop para escrever mais e menos linhas. Observe o comportamento nos modos "w"/"w+" e "r+" . O que você conclui de diferença no comportamento dos modos 'r+' e 'w'/'w+'?
  3. Crie um programa que recebe como parâmetro de entrada o nome de um arquivo que deve ser criado e num segundo argumento um texto qualquer que deve ser inicializado neste arquivo. Os parâmetros de entrada são argc e argv - veja documentação aqui na wiki.
  4. Utilize as funções fputc e fputs para escrever em arquivos
Exemplo: Abrindo e lendo um arquivo
#include <stdio.h>
 
int main()
{
  FILE *p_arq;
  int i,j;
  char buff[100];
 
  if ((p_arq=fopen("IFSC.txt", "r")) == NULL) {
    printf("Problemas na abertura do arquivo, o arquivo existe?\n");
    return 0;
  } else {
    printf("Arquivo aberto com sucesso. Inicio da leitura do arquivo...\n");
  }
 
  while(1) {
    if((fscanf(p_arq,"%s %d",buff,&j))==EOF) {  					  	    
      printf("Fim de leitura\n");
      break;
    }
    printf("%s %d\n",buff,j);
  }
  fclose(p_arq);
  return 0;
}

Note que o fscanf se comporta de forma similar ao scanf. A função retorna o caractere EOF (end-of-file) quando não existe mais dados a serem lidos.

Adicionando uma linha no final de um arquivo
#include <stdio.h>
#include <time.h>
 
int main(void)
{
   time_t ltime;
   FILE *fp;
 
   if ((fp=fopen("leituras.log", "a")) == NULL) {
       printf("Problemas na abertura do arquivo\n");
       return;
   }

   time(&ltime); 
   if ((fputs(ctime(&ltime), fp)) != EOF ) {
       fclose(fp);
   } else {
       printf("Erro na escrita do arquivo!\n");
   }
}

Tipicamente, quando se abre uma arquivo para leitura/escrita, o "cursor" de acesso fica na posição 0 (início do arquivo). Se o arquivo for aberto em modo append ('a'), o "cursor" é posicionado no final. A cada acesso (leitura ou escrita), este cursor é incrementado conforme o número de dados lidos ou escritos. Execute este código algumas vezes e vá observando o que ocorre com o arquivo 'leituras.log'.

Exemplo 2
#include <stdio.h>
#include <time.h>
#include <string.h> 
int main(void)
{
  time_t ltime;
  FILE *fp;
  struct tm *info;
  char buffer[100],b2[100];
 
  if ((fp=fopen("tst.log", "a+")) == NULL) {
    printf("Problemas na abertura do arquivo\n");
    return;
  }
 
  time(&ltime); 
  info = localtime( &ltime );
  strftime(buffer,sizeof(buffer),"%x-%I:%M%p", info);
  printf("Formatted date & time : |%s|\n", buffer );
  strcat(buffer," Joao\n");

  if ((fputs(buffer, fp)) == EOF ) {
    printf("Erro na escrita do arquivo!\n");
  }

  fseek(fp,0,SEEK_SET);
  while(1) {
    if((fscanf(fp,"%s %s",buffer,b2))==EOF) {
      printf("Fim de leitura\n");
      break;
    }
    printf("Marcação lida: %s %s\n",buffer,b2);
  }
  fclose(fp);
  return 0;
}
Alterando o cursor com fseek

Observe este exemplo que escreve em um arquivo e depois lê o conteúdo escrito. Neste caso é utilizado fseek para voltar o cursor para o início do arquivo.

#include <stdio.h>
int main(int argc, char *argv[])
{
  FILE *p_arq;
  int i,j;
  char buff[100];

  if ((p_arq=fopen("IFSC.txt", "w+")) == NULL) {
     printf("Problemas na abertura do arquivo\n");
     return 0;
  } else {
    printf("Aberto com sucesso. Iniciando escrita no arquivo...\n");
  }

  for (i = 0; i<10;i++) {
    if((fprintf(p_arq,"LINHA %d\n",i))==EOF) {
      printf("Erro ao escrever no arquivo!\n");
      return -1;
    }
  }
  printf("Escrito! 'IFSC.txt' criado na pasta deste executável!\n");
 
  /*Muda a posição do cursor para o início do arquivo*/
  fseek(p_arq,0,SEEK_SET);
  while(1) {
    if((fscanf(p_arq,"%s %d",buff,&j))==EOF) {
      printf("Fim de leitura\n");
      break;
    }
    printf("%s %d\n",buff,j);
  }
  fclose(p_arq);
  return 0;
}
Exercícios
  1. Utilize no exemplo acima o modo "w" ao invés de "w+". Qual sua conclusão?
  2. Teste agora com o modo append, da mesma maneira qual a diferença de "a" e "a+"?
Tratando erros de abertura de arquivo

Quando não é possível abrir um arquivo a função retorna NULL, porém para se saber o real motivo da falha é necessário consultar o "errno" que fica setado na biblioteca <errno.h>. Há diversos códigos de erro para váriadas funções, no caso da fopen, consulte aqui os erros possíveis.

#include <stdio.h>
#include <errno.h>

int main()
{
  FILE *fp;
 
  printf("Para simular o erro certifique-se de não haver um arquivo 'testeErro.txt' na pasta\n");
  if ((fp = fopen("testeErro.txt","r")) == NULL) {
    if (errno == ENOENT) 
      printf("Código de erro %d, arquivo não existe!\n",errno);
    else
      printf("Erro não previsto (código: %d)!\n",errno);
    return 0;
  }
  fclose(fp);
  return 0;
}
Outras funções com arquivos

int getc(FILE *stream), int fgetc(FILE *stream) int putc(char ch, FILE *s), int fputc(char ch, FILE *s) These are like getchar, putchar. getc is defined as preprocessor MACRO in stdio.h. fgetc is a C library function. Both achieve the same result!!

fflush(FILE *stream) -- flushes a stream. fclose(FILE *stream) -- closes a stream. We can access predefined streams with fprintf etc.

fprintf(stderr,"Cannot Compute!!\n"); fscanf(stdin,"%s",string);

Lista de exercícios 8

1) Implemente um programa em C que lê de um arquivo chamado "anagrama.txt" duas palavras (uma na primeira linha e a outra na segunda linha). O programa analisa estas palavras e diz se são ou não anagramas (palavra formada pela alteração da ordem ou transposição de letras) ex.: “America” e “Iracema”, “Roma” e “Amor”, “Semolina” e “Is no meal” são anagramas). Utilize o modo de acesso a arquivo "r".
2) Implemente um programa em C que dada uma frase obtida através de um scanf, torne todas as letras em maiúsculas (código 1), torne todas em minúsculas (código 2) ou a primeira letra de cada palavra em maiúscula e demais minúsculas (código 3) e escreve em tela o resultado. O limite para a frase digitada é 100 caracteres. Faça estas operações em funções distintas (tudoMaiusculo(), tudoMinusculo(), primeiraMaiuscula()), utilizando o vetor de caracteres como uma variável global. A escolha das opções dependerá de um numero inteiro lido de um arquivo chamado "config.cfg" este número deve estar escrito no início do arquivo e pode ser ser 1, 2 ou 3. O arquivo config.cfg deve ser criado manualmente e preenchido com diferentes opções para testar o algoritmo. Utilize o modo "r".
3) Implemente um algoritmo em C que conta a ocorrência de cada letra (incluindo números de 0 a 9) dada uma frase. Deve ignorar se a letra foi digitada em maiúscula ou minúscula (portando somar junto). Limite a 100 caracteres. Para cada palavra digitada deve gerar um log em uma arquivo chamado "ocorrencias.log" que contém a data/hora em que foi gerado o registro e quantas letras de cada tipo foram lidas. Utilize o modo "a" ou "a+".
4) Utilize a função rand() e gerador de semente srand() para gerar 6 números aleatórios distintos que variam de 1 a 60 e grava esta sequencia em um arquivo chamado "aposta.txt". Utilize o modo "w" ou "w+".
5) Crie uma estrutura chamada TReferencia que contém um campo "livro" de 100 caracteres e um campo "autores" de 100 caracteres. Crie um loop infinito para receber uma quantidade qualquer de livros que compõem as referencias bibliográficas de um trabalho acadêmico. O programa deve então receber duas string (que podem conter espaços em branco), uma sendo o nome do livro e a outra os autores. Recebidos estes dados devem ser instanciados na struct. Se recebido uma string contendo a palavra "sair" significa que deve-se então proceder com a finalização do programa gravando as instancias da struct em um arquivo chamado "referencias.txt". Como resultado este arquivo deve ter em cada linha o nome do livro entre aspas ("nome do livro") depois uma vírgula e o nome dos autores entre aspas também ("S. AUTOR1, S. AUTOR2").Utilize o modo "r+".

Utilizando typedef

Typedef introdução

O compilador possui uma lista de tipos definidos como 'char', 'int', 'float' e outros. O C permite ainda que outros tipos sejam adicionados por código através da diretiva 'typedef'. A vantagem é principalmente tornar o programa mias inteligível, utilizando-se nomes intuitivos.

Um exemplo, em um programa que processa cálculos financeiros poderia-se criar um tipo chamado para representar valores monetários, o tipo "Real", por exemplo. O comando seria conforme segue:

typedef float Real;

Então, declarar variáveis como salário, comissão e outras sendo do tipo "Real" tornaria o programa mais fácil de compreender.

Real salario, comissao; //Seria equivalente a declarar "float salario, comissao", porem com 'Real' fica mais inteligível

Outra vantagem de uso do typedef é a portabilidade entre plataformas. Imaginando um programa que roda em uma máquina 64 bits e é portado para uma 32 bits, ou 16 bits. Neste exemplo, a tratativa de números 64 bits é bastante diferente entre plataformas. Imaginando um campo 'quantidade' que pode armazenar números inteiros que podem variar até 2 bilhões (positivo ou negativo). Em uma máquina em que o inteiro ocupa 32 bits este variação é possível em um 'int'. Porém em uma plataforma cujo o inteiro ocupa 16 bits (short) já não será possível tal variação. Uma possível solução:

typedef int quantidade; //Plataformas 32 bits ou superior

Ou:

typedef long quantidade; //Plataformas 16 bits
Utilizando typedef em estruturas
Usando o typedef para ajudar na definição e declaração de estruturas
#include <stdio.h>
#include <stdlib.h>
 
void main()
{
  typedef struct {
     int x;
     int y;
  } TTeste;
  
  TTeste *teste;
  
  teste = (TTeste *) malloc (sizeof(TTeste));
  if (teste==NULL) {
      printf("erro de alocação");
      exit(1);
  }
  
  teste->x=10;
  
  free(teste);
  teste=NULL;
}

Recursividade

Introdução a Recursividade

A recursividade ocorre quando uma função chama a si própria. O retorno da função é sempre dado a função que a chamou, então se a função chama a si própria ela mesma recebe o retorno de uma segunda chamada, a segunda chamada receberá o retorno se esta se chamar uma terceira vez e assim por diante. Esta característica é possível pois o C quando inicia a execução de uma função, salva o contexto da anterior em uma pilha, esta função anterior oportunamente receberá o retorno de uma função que chamou.

É importante que a função quando projetada para trabalhar recursivamente possui um teste seguro para que não se chame infinitamente.

Há ainda dois modos de operação, simples ou múltiplo. Na recursividade simples a função se chama apenas uma vezes dentro de uma execução. Na recursividade múltipla em um ciclo de execução a função se chama mais de uma vez (na mesma recursão). O modo simples tende a ser mais eficiente, o tempo e espaço tendem a aumentar linearmente conforme a quantidade de recursões. No modo múltiplo o tempo de execução e o espaço de memória que a função precisa aumentam exponencialmente.

Fontes:

  • KING, K.N. C Programming: A Modern Approach; 2ª ed. [S.l]:W. W. Norton & Company, 2008. 832p. ISBN 9780393979503.
  • wikipedia
Exemplo de recursividade simples
Exemplo 1
Cálculo do fatorial

O fatorial de um número natural n, representado por n!, é o produto de todos os inteiros positivos menores ou iguais a n fonte.

<stdio.h>

int main()
{
  int in, out = 0;
  printf("Entre com o valor de entrada da fatorial: ");
  scanf("%d",&in);
  out = fact(in);
  printf("Resultado: %d\n",out);
  return 0;
}

int fact(int n)
{
  if (n <= 1)
  {
    printf("Finalmente n = 1, a função para se se chamar e retorna o resultado final.\n");
    return 1;
  }
  else
  {
    printf("Nesta chamada n = %d, esta função se chamará recursivamente se for maior que 1.\n",n);
    return n * fact(n-1);
  }
}

No exemplo acima, se entrarmos com o valor 3 (para realizar o fatorial de 3), a seguinte sequencia será processada:

fact(3) Como 3 não é menor ou igual a 1, realiza chamada de si mesma porém com n-1 que será 2
  fact(2) Como 2 não é menor ou igual a 1, realiza chamada de si mesma com n-1 que será 1
    fact(1) Finalmente 1 é menor ou igual a 1, então o retorno será 1, este retorno será entregue a chamada anterior, quando era fact(2)
  fact(2) retornará n * 1 (retorno recebido), resultado 2
fact(3) retornará n * 2 (2 foi o retorno que ela recebeu), resultando 6

Este modo é chamado de recursividade simples pois a função fact() é chamada apenas uma vez em uma recursão.

Exercícios
  1. Qual a limitação de amostragem do resultado do fatorial? Por que existe limitação?
  2. Como se comporta o tempo de processamento a medida que se solicita uma quantidade maior de recursões?
Exemplo de recursividade múltipla
Exemplo 2
Cálculo do número de Fibonacci e da sequência

Na matemática os números de Fibonacci são definidos pela equação F(n) = F(n-1) + F(n-2) onde F(0) = 0 e F(1) = 1 fonte.

#include <stdio.h>

int main()
{
  int in, out = 0, i;
  printf("Entre com o sequencial que deseja saber o fibonacci: ");
  scanf("%d",&in);
  out = fib(in);
  printf("Resultado: %d\n",out);

  printf("A sequencia fica assim:\n");
  for (i=1;i<=in;i++) printf("%d ",fib(i));
  printf("\n");
  return 0;
}

int fib(int n)
{
  if (n-1 <= 0)
    return 0;
  else if (n-1 == 1)
    return 1;
  else {
    return fib(n-1) + fib(n-2);
  }
}
Exercícios
  1. Observe que a função fib() se chama duas vezes em cada recursão, observe como o tempo de processamento é afetado a medida que se aumenta o número de fibonacci requisitado
  2. Qual o número máximo de fibonacci que pode ser obtido neste exemplo, por que existe esta limitação?
  3. Execute este código em um terminal e em outro rode um programa como o 'top' para observar o uso de recursos que cada programa está fazendo. Vá aumentando a sequencia para verificar como o programa se comporta. (para sair do 'top' pressione 'q').

Referências

Referências bibliográficas

  • Araújo, Everton Coimbra de. Algoritmos: fundamento e prática; 3ª ed. [S.l]:Visual Books, 2007. 414p. ISBN 9788575022092.
  • KERNIGHAN, Brian W.; RITCHIE, Dennis M C: a linguagem de programação padrão ANSI; 1ª ed.[S.l]:Campus, 1989. 304p. ISBN 9788570015860.
  • SCHILDT, Herbert C Completo e Total; 3ª ed. [S.l]:Makron Books, 2009. 827p. ISBN 9788534605953.
  • FORBELLONE, Andre L. Lógica de Programação; 3ª ed. [S.l]:Makron Books, 2005. 197p. ISBN 9788576050247.
  • KING, K.N. C Programming: A Modern Approach; 2ª ed. [S.l]:W. W. Norton & Company, 2008. 832p. ISBN 9780393979503.
  • MANZANO, Jose Augusto Navarro Garcia Estudo Dirigido em Linguagem C. ; 16ª ed. [S.l]:Erica, 2012. 216p. ISBN 9788571948877.
  • NEVES, Júlio Cézar Programação Shell Linux; 5a ed. Rio de Janeiro:Brasport, 2005. 408p. ISBN 8574522031.
  • VEIGA, Roberto G. A. Comandos do Linux: guia de consulta rápida; ed. São Paulo:Novatec, 2004. 144p. ISBN 85-7522-060-8.

Referências adicionais

Ferramentas úteis

  • VisualG3: Uma IDE para desenvolvimento de programas em pseudocódigo (freeware), permite editar e compilar programas utilizando uma sintaxe própria de pseudocódigo muito parecida com a que trabalhamos em sala. Muito útil para verificar o funcionamento real dos algoritmos.
  • LibreOffice: O LibreOffice é um programa gratuito (freeware) e de código aberto (opensource). Além de editor de textos, planilhas e apresentações tem a ferramenta Draw que permite a criação de fluxogramas.
  • VirtualBox: O Oracle VirtualBox é um programa gratuito (freeware) que permite criar e instanciar máquinas virtuais. O uso de máquinas virtuais é bastante interessante quando desejamos ter diferentes sistemas operacionais em um computador bem como quando se está realizando ensaios e deseja-se isolar estes experimentos do sistema principal.
  • Ubuntu: O Ubuntu é uma distribuição linux (freeware e opensource) bastante estável e com uma comunidade bastante ativa que está sempre atualizando o sistema e presente nos foruns e redes sociais para dirimir dúvidas.
  • LinuxMint: O LinuxMint é uma distribuição linux (freeware e opensource) bastante estável e confortável aos usuários windows, pois traz um gerenciador de janelas configurado de uma forma mais natural para estes usuários e vem com um conjunto de programas pré-instalados que consegue atender a maior parte das demandas inicias.

Sites úteis

  • cplusplus.com: Traz tutoriais, artigos e descrições de funções C e C++
  • codechef.com: Permite a edição, compilação e testes online de códigos em várias linguagens inclusive ANSI C
  • codecademy.com: Tem cursos gratuitos de programação, bastante didáticos em inclusive em português. Porém não tem curso de C, uma alternativa interessante para quem quiser aprender uma outra linguagem que tem boa aceitação inclusive para desenvolvimento de sistemas embarcados é o Python
  • kaggle.com: Site tem publicado centenas de algoritmos em diversas linguagens para resolver os mais variados problemas. Tem também competições de algoritmos
  • Interface PythonTutor para C: Permite compilar programas em C e acompanhar passo a passo o que ocorre com as variáveis

Eventos da área de desenvolvimento

Orientação

Horário de Atendimento Paralelo

Horário de Monitoria

Orientações para entrega dos trabalhos

  • Para listas a serem entregues via moodle:
    • Nome do arquivo: "seu nome completo/ demais colegas do grupo" - "lista de exercícios 1", "desafio 1", etc.
    • Formato do arquivo em anexo: PDF (cada lista num único arquivo feito em editor de texto, exportado para PDF).
  • Para listas a serem entregues por email (para cleber.amaral@ifsc.edu.br), período válido até a meia noite do prazo:
    • Assunto do e-mail: "PRG29002 - xxx" onde xxx é "lista de exercicios 1", "desafio 1", etc.
    • Nome do arquivo anexo: "seu nome completo/ demais colegas do grupo" - "lista de exercícios 1", "desafio 1", etc.
    • Formato do arquivo em anexo: PDF (cada lista num único arquivo feito em editor de texto, exportado para PDF).
      • Regra válida a partir de 14/04 (isento apenas trabalhos "lista 1")

Trabalhos entregues com atraso

  • Para os trabalhos não entregues no prazo (não justificados) temos a penalidade de 1 ponto por dia de atraso
    • Excepcionalmente para as lista1 e lista7 com até 6 dias de atraso terá desconto de 1 ponto apenas, seguindo a regra acima para entregas deste trabalho após este prazo
    • Desafios não tem extensão de prazo, não adianta enviar se o prazo se esgotou
      • Para o desafio1 o sistema de desconto por dia de atraso foi utilizado, este ficou como exceção

Projeto final

O aluno deve propor ao professor um projeto de sua preferência que respeite os requisitos mínimos. Sendo aceito deverá desenvolver o projeto e apresentá-lo.

Requisitos mínimos

  • Realizar acesso a arquivo, lendo e escrevendo informações
  • Utilizar funções (ao menos duas sendo ao menos uma com argumentos)
  • Apresentar menu utilizando switch case e conter laço infinito
  • Aceitar argumento de entrada no programa
  • Utilizar comentários
  • Utilizar alguma biblioteca (além da stdio.h)
  • Utilizar ao menos 3 dos seguintes recursos
    • Utilizar diretivas de pré-compilação
    • Utilizar Ponteiros
    • Utilizar Structs ou Unions
    • Utilizar alocação dinâmica de memória

Modelo

  • Trabalho individual

Metodologia

  1. Apresentar a proposta de projeto ao professor
  2. Documentar o escopo do projeto utilizando descrição narrativa (descrição simples)
    1. Cenário
    2. Problema
    3. Dados de entrada e saída
    • O planejamento do cronograma não será cobrado porém cabe ao aluno se organizar quanto ao tempo para entrega no prazo
  3. Desenvolver o projeto
  4. Apresentar individualmente ao professor
    • Serão realizados testes diversos, arguido sobre o funcionamento, possibilidades de alterações, etc

Algumas ideias de projetos

  • Sugestão geral: veja em outras disciplinas que processos podem ser automatizados e proponha um projeto que realiza esta tarefa como de cálculos diversos de eletrônica, de rádio transmissão, etc.
  • Implementar o jogo Pedra, papel ou tesoura. Neste jogo dois ou mais jogadores em diferentes computadores devem rodar um aplicativo que fará a leitura de um arquivo compartilhado. O algoritmo deve tratar as etapas do jogo (Setup do aplicativo, entrada na sala, escolha da figura e apresentação do resultado)
  • Implementar o jogo da velha escrevendo em arquivo. Neste jogo dois jogadores em diferentes computadores devem rodar um aplicativo que fará a leitura de um arquivo compartilhado. O algoritmo deve tratar as etapas do jogo (Setup do aplicativo, entrada na sala, seleção das casas e apresentação do resultado)
  • Implementar controle de empréstimo de objetos. Neste software o usuário poderá digitar nomes de objetos que emprestou, a pessoa a quem emprestou e automaticamente o software guarda a data. Deve haver uma opção para gerar relatório dos itens emprestados e opção para marcar a devolução (podendo manter o registro em histórico ou apagando o registro).
  • Implementar software gerador de lista de compras. Neste software o usuário poderá digitar itens de supermercado com nome e quantidade. O software escreve num arquivo que poderá depois ser impresso. O software também pode ter função de numa segunda execução já trazer a antiga listagem digitada e permitir que o usuário apenas selecione novas quantidades ou inclua novos itens.
  • Implementar software para realização de cálculos de eletrônica. Neste software um menu apresenta várias opções de cálculo como de potencia através de tensão e corrente, como obtenção do valor de um resistor, como solução de equivalência de paralelo de vários resistores e outras. Num arquivo texto pode ser armazenado um histórico de operações realizadas.
  • Implementação de software para apostas na mega sena. Neste software são dadas sugestões de números para apostas de acordo com o número do sorteio da mega sena. Com este histórico armazenado é possível então entrar com um número de sorteio e digitar quais foram os números verdadeiramente sorteados na loteria federal checagem os acertos.

Tópicos avançados

Calculo de tempo recursividade simples

Na recursividade simples o tempo aumenta linearmente como mostra o código abaixo

#include <stdio.h>
#include <stdint.h>
#include <time.h>

int64_t timespecDiff(struct timespec *timeA_p, struct timespec *timeB_p)
{
  return ((timeA_p->tv_sec * 1000000000) + timeA_p->tv_nsec) -
           ((timeB_p->tv_sec * 1000000000) + timeB_p->tv_nsec);
}

int fact(int n)
{
  if (n <= 1)
  {
    printf("Finalmente n = 1, a função para se se chamar e retorna o resultado final.\n");
    return 1;
  }
  else
  {
    printf("Nesta chamada n = %d, esta função se chamará recursivamente se for maior que 1.\n",n);
    return n * fact(n-1);
  }
}

int main()
{
  uint64_t tempoFact10 = 0;
  uint64_t tempoFact20 = 0;
  uint64_t tempoFact30 = 0;
  int i;
  int in, out = 0;
  struct timespec start, end;

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    // Some code I am interested in measuring 
    in = 10;
    out = fact(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFact10 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    // Some code I am interested in measuring 
    in = 20;
    out = fact(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFact20 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    // Some code I am interested in measuring 
    in = 30;
    out = fact(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFact30 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  printf("\nMédia de tempo Fact(10): %li\n",tempoFact10);
  printf("\nMédia de tempo Fact(20): %li\n",tempoFact20);
  printf("\nMédia de tempo Fact(30): %li\n",tempoFact30);
}
Calculo de tempo recursividade multipla

Na recursividade simples o tempo aumenta linearmente como mostra o código abaixo

#include <stdio.h>
#include <stdint.h>
#include <time.h>

int64_t timespecDiff(struct timespec *timeA_p, struct timespec *timeB_p)
{
  return ((timeA_p->tv_sec * 1000000000) + timeA_p->tv_nsec) -
           ((timeB_p->tv_sec * 1000000000) + timeB_p->tv_nsec);
}

int main()
{
  uint64_t tempoFib10 = 0;
  uint64_t tempoFib20 = 0;
  uint64_t tempoFib30 = 0;
  uint64_t tempoFib40 = 0;
  int i;
  int in, out = 0;
  struct timespec start, end;

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    in = 10;
    out = fib(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFib10 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    in = 20;
    out = fib(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFib20 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    in = 30;
    out = fib(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFib30 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  for (i=0;i<10;i++)
  {
    clock_gettime(CLOCK_MONOTONIC, &start);

    in = 40;
    out = fib(in);
 
    clock_gettime(CLOCK_MONOTONIC, &end);

    uint64_t timeElapsed = timespecDiff(&end, &start);
    tempoFib40 += timeElapsed / 10;
  }
  printf("Resultado: %d\n",out);

  printf("\nMédia de tempo Fib(10): %li\n",tempoFib10);
  printf("\nMédia de tempo Fib(20): %li\n",tempoFib20);
  printf("\nMédia de tempo Fib(30): %li\n",tempoFib30);
  printf("\nMédia de tempo Fib(40): %li\n",tempoFib40);
}

int fib(int n)
{
  if (n-1 <= 0)
    return 0;
  else if (n-1 == 1)
    return 1;
  else {
    return fib(n-1) + fib(n-2);
  }
}

Exemplo de resultado:

Média de tempo Fib(10): 1113
Média de tempo Fib(20): 113658
Média de tempo Fib(30): 9817635
Média de tempo Fib(40): 427706133

Ou seja, há um aumento exponencial no tempo