Apostila - Linguagem C Aplicada A Engenharia - 127 Paginas

Fazer download em pdf ou txt
Fazer download em pdf ou txt
Você está na página 1de 127

Exemplos em Linguagem C Aplicada Engenharia

1) INTRODUO: Introduo bsica: #include <stdio.h> /* Programa-exemplo #1. */ main() { int idade; idade = 33; printf ("A minha idade %d\n", idade) ; } #include <stdio.h> /* Programa-exemplo #2 - converso de ps para metros. */ main() { int ps; float metros; printf("Informe o nmero de ps: "); scanf("%d", &ps); metros = ps * 0.3048; printf("%d ps %f metros\n", ps, metros); }

/* & -> operador "endereo de" */ /* converso de ps para metros */

#include <stdio.h> /* Programa-exemplo #2, 2 verso - converso de ps para metros. */ main() { float ps, metros; /* torna a varivel ps um float */ printf("Informe o nmero de ps: "); scanf("%f", &ps); /* l um float */ metros = ps * 0.3048; /* converso de ps para metros */ printf("%f ps %f metros\n", ps, metros); } Introduo s Funes em C: #include <stdio.h> /* Um programa-exemplo com duas funes. */ main() { hello(); } hello() { printf("Al\n"); }

/* chama a funo hello */

#include <stdio.h> /* Um programa que usa uma funo com um argumento.*/ main() {

int num; num=100; sqr(num); } sqr(int x) { } #include <stdio.h> /* Outro exemplo de funo com argumentos.*/ main() { mul(10, 11); } mul(a,b) int a,b; { } printf("%d", a*b); printf("%d elevado ao quadrado %d\n", x, x*x);

/*chama sqr() com num */ /*a declarao do parmetro est dentro dos parnteses*/

/* observe a forma diferente de declarar a funo mul() */

#include <stdio.h> /* Um programa que usa retorno. */ main() { int resposta; resposta = mul (10, 11); printf("A resposta %d\n", resposta); } { } /* Esta funo retorna um valor */ mul(int a, int b) return a*b;

/* atribui o valor de retorno */

Forma geral de uma funo: tipo-de-retorno nome-da-funo (lista de parmetros) identificao dos parmetros (caso no identificados na lista) { corpo do cdigo } Para funes sem parmetros, no haver lista de parmetros.

Introduo ao controle de programas em C: A declarao if: if(condio)declarao;

onde condio uma expresso que resulta V ou F. Em C, V no zero e F zero. Exemplos: if(10 < 11) printf("10 menor que 11"); if(10 = =11) printf("Al"); /* imprime */ /* no imprime */

O loop for: for(inicializao; condio; incremento)declarao; inicializaao usado p/ atribuir varivel de controle do loop um valor inicial condio uma expresso que testada toda vez que o loop continua a ser executado incremento incrementa a varivel de controle do loop #include <stdio.h> /* Um programa que ilustra o loop for */ main() { int contador; for(contador=1; contador<=100; contador++) printf("%d ", contador); } Obs) contador++ tem o mesmo efeito que contador=contador+1 contador- - tem o mesmo efeito que contador=contador-1

Blocos de cdigo: Como C uma linguagem ESTRUTURADA, ela suporta a criao de blocos de cdigo: if(x<10) { int novo; printf(Muito baixo, tente outra vez!\n); novo=x+2; printf(Que tal x=%d?,novo); scanf("%d", &x); } Nota: a varivel novo s existe dentro deste bloco de cdigo, sendo eliminada da memria aps o fechamento da ltima chave }. Bsico sobre Caracteres e Strings: #include <stdio.h> /* Um exemplo usando caracteres.*/ main() { char ch; ch = 'A'; printf("%c", ch) ; ch = 'B'; printf("%c",ch) ; ch = 'C' ; printf("%c", ch) ; } #include<stdio.h> #include<conio.h> main() {

char ch; ch = getche() ; /* l um caractere do teclado */ if(ch= ='H') printf("voc pressionou a minha tecla mgica\n"); } Em C, uma string uma seqncia de caracteres (char) organizada em uma matriz unidimensional (vetor) terminada pelo caracter nulo (null). O caracter nulo o 0. A linguagem C no tem uma varivel tipo string propriamente como o Basic. Em vez disso voc declara uma matriz de caracteres e usa as vrias funes de strings encontradas na biblioteca adequada para manipul-las. Para ler uma string do teclado, primeiro cria-se uma matriz de caracteres suficientemente grande para armazenar a string mais o terminador nulo. Por exemplo, para armazenar a palavra "hello", necessrio uma matriz de caracteres de 6 elementos: 5 p/ a string e 1 p/ o terminador nulo: 'h' 'e' 'l' 'l' 'o' '\0'

Declara-se a matriz de caracteres como qualquer varivel: char txt[6]; Assim txt[0] = 'h' txt[1] = 'e' txt[2] = 'l' txt[3] = 'l' txt[4] = 'o' txt[5]= '/0' Para a leitura de uma string do teclado, pode-se usar funo scanf() como veremos adiante, ou usar a funo gets(). A funo gets() toma o nome da string como argumento e faz a leitura dos caracteres do teclado at que a tecla ENTER seja pressionada. A tecla ENTER nao armazenada como caracter,mas sim substituda pelo terminador nulo. #include<stdio.h> /* Um exemplo com string. */ main() { char str[80]; printf("Digite o seu nome: "); gets(str); printf("Al %s", str); } Fazendo um breve resumo do que se viu de printf() at agora: A forma geral da funo printf() : printf("string de controle", lista de argumentos) Cdigo %d %f %c %s Significado Exibe um inteiro no formato decimal Exibe um tipo float no formato decimal Exibe um caractere Exibe uma string

Exemplos: printf("%s %d", "Esta uma string ", 100); exibe: Esta uma string 100 printf("esta uma string %d", 100); exibe: esta uma string 100 printf("o nmero %d decimal, %f ponto flutuante.",10, 110.789); exibe: o nmero 10 decimal, 100.789 ponto flutuante.

printf("%s", "Al"\n"); exibe: Al e avana uma linha 2) VARIVEIS, CONSTANTES , OPERADORES E EXPRESSES: Tipos de dados: Tamanho e intervalo dos tipos bsicos de dados do Turbo C: _________________________________________ Tipo Tamanho(em bits) Intervalo __________________________________________ char 8 -128 a 127 int 16 -32768 a 32767 float 32 3.4E-38 a 3.4E+38 double 64 1.7E-308 a 1.7E+308 void 0 sem valor __________________________________________

Modificadores de Tipos: signed, unsigned, long, short: __________________________________________________________________________ Todas as combinaes possveis dos tipos bsicos e modificadores do Turbo C. __________________________________________________________________________ Tipo Tamanho(em bits) Intervalo decimal [8 bits = 1 byte] __________________________________________________________________________ char 8 -128 a 127 unsigned char 8 0 a 255 signed char 8 -128 a 127 int 16 -32768 a 32767 unsigned int 16 0 a 65535 signed int 16 -32768 a 32767 short int 16 -32768 a 32767 unsigned short int 16 0 a 65535 signed short int 16 -32768 a 32767 long int 32 -2147483648 a 2147483647 signed long int 32 -2147483648 a 2147483647 unsigned long int 32 0 a 4294967295 float 32 3.4E-38 a 3.4E+38 double 64 1.7E-308 a 1.7E+308 long double 80 3.4E-4932 a 1.1E+4932 ____________________________________________________________________________

#include <stdio.h> /* Ilustra a diferena entre inteiros com sinal e sem sinal. */ main() { int i; /* um inteiro com sinal */ unsigned int j; /* um inteiro com sinal */

j = 60000; i = j; printf("%d %u", i, j); } /* Este programa imprime o alfabeto. */ #include <stdio.h> main() { char letra; for(letra = 'A' ; letra <= 'Z' ; letra ++) printf("%c ", letra); } Declarando variveis:

varivel local: varivel global:

declararar aps declarao de main() ou funo(). declarar antes da declarao de main().

int i, j, l; short int si; unsigned int ui; double balano, lucro, perda; Exemplo: /* Somatrio de nmeros de 0 a 9.*/ #include <stdio.h> int soma; main() { int count; soma = 0; for(count=0; count<10; count++) { total(count); display(); } }

/* Observe a separao por vrgulas */

/* Varivel Global */ /* Varivel Local */ /* inicializacao*/

total(int x) { soma = x + soma; } display() { it count; na */ for(count=0; count<10; count++) printf("."); printf("o somatrio corrente %d\n", soma); }

/* adiciona ao total corrente */ /* Parmetro Formal */

/* Varivel Local, este count diferente daquele count /*funo main()*/

Constantes: ___________________________________________________________________________ Tipo de Dado Exemplos de Constantes ___________________________________________________________________________ Char a n 9 int 1 123 21000 -234

long int 35000 -34 short int 10 -12 90 unsigned int 10000 987 40000 float 123.23 4.34e-3 double 123.23 12312333 -0.9876324 ____________________________________________________________________________ Constantes Hexadecimais e Octais : hex = 0xFF; oct 011; Cdigos de Barra Invertida: ch = \t ; printf(este um teste\n); _____________________________________________________________________________ Cdigos de Barra Invertida. _____________________________________________________________________________ Cdigo Significado \b Retrocesso \f Alimentao de formulrio \n Nova linha \r Retorno de carro \t Tabulao horizontal \" Aspas \' Apstrofo \0 Nulo \\ Barra invertida \v Tabulaco vertical \a Sinal sonoro \N Constante Octal (onde N uma constante octal) \xN Constante hexadecimal (onde N uma constante hexadecimal) ___________________________________________________________________________________ Inicializao de Variveis: tipo nome_da_varivel=constante; char ch = 'a'; int primeiro = 0; float balano = 123.23; /* Um exemplo usando inicializao de variveis. */ #include <stdio.h> main() { int t; printf("informe um nmero: "); scanf("%d", &t); total(t); } total(int x) { int sum=0, i, count; for(i=0; i<x; i++) { sum = sum + i; for(count=0; count<10; count++) printf("."); printf("a somatria corrente %d\n", sum); } } Operadores Aritmticos: ____________________________________________________________________________ Operador Ao ____________________________________________________________________________ /* 255 em decimal */ /* 9 em decimal */

subtrao, tambm o menos unrio (-1*x) + adio * multiplicao / diviso % resto da diviso -decremento ++ incremento ____________________________________________________________________________ Obs) Operador unrio aquele que opera sobre um nico argumento. /* clculo do quociente e do resto de uma diviso */ #include <stdio.h> main() { int x,y; printf("informe o dividendo e o divisor: "); scanf("%d%d", &x, &y); printf("quociente da diviso = %d\n", x/y); printf("resto da diviso = %d", x%y); } Incremento e Decremento: x= x+1; pode ser escrito como: ++x; ou contedo */ x++; incrementa */ Exemplos: x=10; y= ++x; x=10; y= x++; O mesmo vale para decremento x-- e --x. Precedncia de operadores: 1) ++ -2) - (unrio) 3) */% 4) +Obs1) Operadores do mesmo nvel de precedncia so avaliados pelo compilador da esquerda para a direita. Obs2) Parnteses podem ser colocados para alterar a ordem de avaliao, forando uma operao ou conjunto de operaes para um nvel de precedncia mais alto. Aconselha-se sempre o uso de parnteses, j que alm de deixarem o cdigo mais claro, eles no acrescentam nenhum tempo de execuo adicional ou consumo extra de memria. Operadores Relacionais e Lgicos: ___________________________ Operadores Relacionais ___________________________ > >= < <= == Maior que Maior ou igual que Menor que Menor ou igual que Igual _____________________________ Operadores Lgicos _____________________________ && AND(E) || OR(OU) ! NOT(NO) _____________________________

/* primeiro incrementa a varivel depois usa o /* primeiro usa o contedo da varivel depois

/* x=10 */ /* primeiro faz x=x+1 e depois faz y=x */ /* resultado: y=11 e x=11 */ /* x=10 */ /* primeiro faz y=10 e depois faz x=x+1 */ /* resultado: y=10 e x=11 */

!= Diferente ___________________________

/* Este programa ilustra os operadores relacionais. */ #include <stdio.h> main() { int i, j; printf("informe dois numeros: "); scanf("%d%d", &i, &j); printf("%d == %d %d\n", i, j, i==j); printf("%d != %d %d\n", i, j, i!=j); printf("%d <= %d %d\n", i, j, i<=j); printf("%d >= %d %d\n", i, j, i>=j); printf("%d < %d %d\n", i, j, i<j); printf("%d > %d %d\n", i, j, i>j); } Obs) Os operadores relacionais podem ser aplicados a qualquer tipo de dado bsico. O exemplo a seguir mostra a comparao entre dois char: ch1='A'; ch2='B'; if(ch2 > ch1) printf("maior que"); /* Este programa ilustra os operadores lgicos. */ #include <stdio.h> main() { int i, j; printf("informe dois nmeros(cada um sendo 0 ou 1): "); scanf("%d%d", &i, &j); printf("%d AND %d %d\n", i, j, i && j); printf("%d OR %d %d\n", i, j, i || j); printf("NOT %d %d\n", i, !i); } Observe que todas as expresses lgicas e relacionais produzem um resultado 0 ou 1. Portanto, o seguinte programa no s est correto como produzir o nmero 1 no ecr: #include <stdio.h> main() { int x; x=100; printf("%d", x>10); } /* Imprime os nmeros pares entre 1 e 100. */ #include <stdio.h> main() { int i; for(i=1; i<=100; i++) if(!(i%2)) printf("%d ",I); } resultado*/ Operador de Atribuio: Em C o operador de atribuio o sinal de igual ' = '. Ao contrrio de outras linguagens, o C permite que o operador de atribuio seja usado em expresses relacionais ou lgicas:

/* o operador de resto dar falso (zero) */ /* quando usada c/ nmero par. Esse /* invertido pelo NOT */

#include <stdio.h> main() { int x, y, produto; printf("informe dois nmeros: "); scanf("%d%d", &x, &y); if( (produto=x*y) < 0) atribuio */ printf ("Um dos nmeros negativo\n"); else printf("O produto dos nmeros positivos : %d", produto); } Modeladores(Casts): (tipo) expresso onde tipo qualquer tipo de dado padro em C.

/* expresso relacional c/ operador de /* caso contrrio da condio do if */

Uso: Forar o resultado de uma expresso a ser determinado tipo de dado. Exemplo: Seja x um int e y um float. x=3; y= (float) x/2; y= (float) (x/2) #include <stdio.h> main() { int i; for(i=1; i<=100; ++i ) printf("%d/3 : %f\n", i, (float) i/3); } 3) DECLARAES DE CONTROLE DO PROGRAMA: Declarao if: A forma geral da declarao if : if(condio)declarao; else declarao; A clusula else opcional. Se condiao V (!=0), declarao (ou bloco de declaraes) que forma o destino de if executada, caso contrrio a declarao ou bloco que forma o destino do else executada. O programa a seguir ilustra o uso do if. Este programa introduz dois novos formatos para scanf() e printf() que so o %x e %o, associados leitura (scanf) e impresso (printf) respectivamente de dados hexadecimais e octais. /* Programa de converso de base numrica #1. decimal --> hexadecimal hexadecimal --> decimal decimal --> octal octal --> decimal*/ #include<stdio.h> main() { int opo; int valor; printf("Converter:\n"); printf(" 1: decimal para hexadecimal\n"); printf(" 2: hexadecimal para decimal\n"); printf(" 3: decimal para octal\n"); printf(" 4: octal para decimal\n"); printf("informe a sua opo:"); scanf("%d", &opo); /* y=1.5 */ /* y=1.0 */

/*imprime i e i/3 com a parte fracionria */

10

if(opcao==1) { printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em hexadecimal : %x", valor, valor); } if(opo==2) { printf("informe um valor em hexadecimal:"); scanf("%x", &valor); printf("%x em decimal : %d", valor, valor); } if(opo==3){ printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em octal : %o", valor, valor); } if(opo==4){ printf("informe um valor em octal:"); scanf("%o", &valor); printf("%o em decimal : %d", valor, valor); } } Declaraao else: /* Um exemplo de if_else.*/ #include<stdio.h> main() { int i; printf("informe um nmero:"); scanf("%d", &i); if(i<0) printf("o nmero negativo:"); else printf ("o nmero positivo ou zero"); }

Encadeamento if-else-if: if(condio) declarao; else if(condio) declarao; else if(condio) declarao; . . . else declarao; /* Programa de converso de base numrica #2 - encadeamento if-else-if. decimal --> hexadecimal hexadecimal --> decimal decimal --> octal octal --> decimal */ #include<stdio.h> main() { int opo;

11

int valor; printf("Converter:\n"); printf(" 1: decimal para hexadecimal\n"); printf(" 2: hexadecimal para decimal\n"); printf(" 3: decimal para octal\n"); printf(" 4: octal para decimal\n"); printf("informe a sua opo:"); scanf("%d", &opo); if(opo==1) { printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em hexadecimal : %x", valor, valor); } else if(opo==2) { printf("informe um valor em hexadecimal:"); scanf("%x", &valor); printf("%x em decimal : %d", valor, valor); } else if(opo==3){ printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em octal : %o", valor, valor); } else if(opo==4){ printf("informe um valor em octal:"); scanf("%o", &valor); printf("%o em decimal : %d", valor, valor); } } Validade da Expresso Condicional: Zero=Falso Exemplo: /* Divide o primeiro nmero pelo segundo. */ #include<stdio.h> main() { int a, b; printf("informe dois nmeros: "); scanf("%d%d", &a, &b); if(b) printf("%f\n",(float) a/b); else printf("no posso dividir por zero\n"); }

No Zero=Verdadeiro:

/* s executa se b!=0 */

No necessrio explicitamente usar if(b == 0) printf("%f\n",(float) a/b); Declarao switch: (similar a ON-GOTO do BASIC): Difere do if-else-if por s poder testar igualdades. No entanto, gera um cdigo .EXE menor e mais rpido que if-else-if. switch(varivel) { case constante1: sequencia de declaracoes break; case constante2: sequencia de declaracoes break; case constante3: sequencia de declaracoes break; . . . default

12

sequencia de declaracoes

/* Programa de conversao de base numerica #3 usando-se a declaracao switch. decimal --> hexadecimal hexadecimal --> decimal decimal --> octal octal --> decimal */ #include<stdio.h> main() { int opcao; int valor; printf("Converter:\n"); printf(" 1: decimal para hexadecimal\n"); printf(" 2: hexadecimal para decimal\n"); printf(" 3: decimal para octal\n"); printf(" 4: octal para decimal\n"); printf("informe a sua opcao:"); scanf("%d", &opcao); switch(opcao) { case 1: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em hexadecimal : %x", valor, valor); break; case 2: printf("informe um valor em hexadecimal:"); scanf("%x", &valor); printf("%x em decimal : %d", valor, valor); break; case 3: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em octal : %o", valor, valor); break; case 4: printf("informe um valor em octal:"); scanf("%o", &valor); printf("%o em decimal : %d", valor, valor); break; } } Declaracao default: Usada para prever a possibilidade de nunhuma correspondencia ser encontrada: switch(opcao) { case 1: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em hexadecimal e : %x", valor, valor);

13

break; case 2: printf("informe um valor em hexadecimal:"); scanf("%x", &valor); printf("%x em decimal e: %d", valor, valor) break; case 3: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em octal e: %o", valor, valor); break; case 4: printf("informe um valor em octal:"); scanf("%o", &valor); printf("%o em decimal e: %d, valor, valor); break; default: /* quando nenhuma correspondencia encontrada */ printf(opcao invalida, tente outra vez\n"); break; } Loop for: A forma geral do loop for : for(inicializaao; condiao; incremento) declaraao; -inicializaao uma declaraao de atribuiao que inicializa a variavel de controle do loop. -condiao uma expressao relacional que determina qual situaao terminar a execuao do loop pelo teste da variavel de controle contra algum valor. Uma vez falsa a condiao o programa seguir na declaraao seguinte ao trmino do loop for. -incremento define como a variavel de controle mudar ao longo da execuao do loop. -declaraao (ou bloco de declaraoes) o que o loop for executa enquanto ativo. /* Imprime numeros de 1 a 100 no ecr */ #include <stdio.h> main() { int x; for(x=1; x<=100; x++) printf ("%d ",x); }

/* Imprime numeros de 100 a 1 no ecr */ #include <stdio.h> main() { int x; for(x=100; x>0; x--) printf("%d ",x); } /* Imprime numeros de 0 a 100 de 5 em 5 */ #include <stdio.h>

14

main() { int x; for(x=0; x<=100; x=x+5) printf("%d ",x); } /* Imprime quadrado dos numeros numeros de 0 a 99 */ #include <stdio.h> main() { int i; for(i=0; i<100; i++){ printf("Este o valor de i: %d ",i); printf(" E i ao quadrado : %d\n", i*i); } } Obs) O teste condicional realizado no incio do loop. Isto significa que o cdigo pode nunca ser executado se a condiao falsa logo de incio: x=10 for(y=10; y!=x, ++y) printf("%d", y); printf("fim do loop"); O loop acima nunca executado j que de incio x=y. Portanto a condiao avaliada como falsa e o loop acaba prematuramente. Variaoes do loop for: /* Este programa imprime os numeros de 0 a 98 de 2 em 2 */ #include<stdio.h> main() { int x, y; for(x=0,y=0; x+y<100; ++x,y++) /* Duas variveis de controle: x e y */ printf("%d ", x+y); /* Em C a vrgula um operador que signifi- */ /* ca em essncia: "faa isto E aquilo". Por */ /* isso vlida a construao deste loop for. */ } /* O incremento poderia ser todas as combinaoes de incremento anterior e posterior, j que neste caso este fato nao influencia a performance do programa */ /*Ensino da adiao.*/ #include<stdio.h> #include<conio.h> main() { int i, j, resposta; char fim = ''; for(i=1; i<100 && fim!='N';i++){ /* loop ativo quando i<100 e fim != 'N' */ for(j=1; j<10; j++) { printf("quanto %d + %d?", i, j); scanf("%d", &resposta); if(resposta != i+j) printf("errado\n"); else printf("certo\n"); } printf("mais?"); fim=getche(); printf("\n"); } } Usos nao corriqueiros do loop for:

15

/* Um uso nao usual do lao for.*/ #include <stdio.h> main() { int t; for(prompt(); t=readnum(); prompt()) /* se t==0 loop for termina */ sqrnum(t); } /* O loop for acima funciona porque (inicializaao;condiao;incremento) podem consistir de qualquer expressao vlida em C */ prompt() { printf("informe um inteiro:"); } readnum() { int t; scanf("%d",&t); return t; } sqrnum(int num) { printf("%d\n", num*num); } Inclusive possvel deixar faltando partes da definiao do loop. O programa abaixo pega a senha do teclado e testa se igual 135. Se for igual, condiao torna-se falsa e o loop encerra com uma mensagem: #include <stdio.h> main() { int x; for(x=0; x!=135;) scanf("%d", &x); printf("senha correta"); } Loop infinito: for(;;) { printf("este lao sera executado para sempre.\n"); printf("use CTRL-BREAK para parar!"); } Interrupao de um loop for: for(;;){ ch=getche(); /* pega um caractere */ if(x=='A') break; /* break: sai do lao e vai p/ a declaraao seguinte */ } printf("voce digitou um A"); Loop while: A forma geral : while(condiao)declaraao; O conjunto de declaraoes em declaraao repetido enquanto condiao verdadeira.

16

Exemplo: A rotina a seguir pra o processamento e entra em loop at que o caractere A maisculo seja pressionado: espera_por_caractere() { char ch; ch='\0'; /*inicializa ch*/ while(ch!='A') ch= getche(); }

/* Um programa que centraliza o texto no ecr.*/ #include <stdio.h> #include<string.h> main() { char str[255]; printf("insira uma string:"); gets(str); centraliza(strlen(str)); printf(str); } /* Calcula e exibe um numero adequado de espacos para centralizar uma string de comprimento tamanho. */ centraliza(int tamanho) { tamanho = (80-tamanho)/2; /* 80-tamanho o que sobra na linha de 80 col.*/ while(tamanho>0){ printf(" "); tamanho--; } } Onde muitas condioes distintas podem terminar o loop while, comum usar uma nica varivel como expressao condicional com o valor dessa varivel sendo indicado em vrios pontos ao longo do loop: funcl() { int trabalhando; trabalhando =1; /*isto , verdadeira*/ while(trabalhando){ trabalhando=process1(); if(trabalhando) trabalhando= process2(); if(trabalhando) trabalhando= process3(); } } No cdigo acima, qualquer das trs rotinas process() podem retornar falso (zero) e fazer com que o loop seja encerrado. Nao h em absoluto a necessidade de qualquer declaraao no corpo do loop while. Por exemplo: while( (ch=getche()) != 'A' ); /* a atribuiao ch dentro da expressao */ /* condicional do while funciona por que o */ /* sinal de igual na realidade somente um */ /* operador que especifica o valor do seu */ /* operando direita */ simplesmente ser executado at que o caractere A seja digitado.

17

Loop infinito c/ while: while(1){conjunto de declaraoes}; idntico ao for(;;){conjunto de declaraoes}; j que a condiao do while sempre 1 = verdadeiro. Loop do-while (faa enquanto): do{ declaraao: }while(condiao); Exemplos: /* l numeros do teclado at que um deles seja menor que 100 */ #include<stdio.h> main() { int num; do { scanf("%d",&num); printf("voce digitou %d\n", num); }while(num>100); } O uso mais comum do loop do-while em rotinas de seleao de menu. Depois que a scanf() pegar a opao, o loop se repetir at que o usurio digite uma opao vlida no exemplo abaixo: /* certifica-se de que o usuario especificou uma opao valida*/ #include <stdio.h> main() { int opcao; do{ printf("Converter:\n"); printf(" 1: decimal para hexadecimal\n"); printf(" 2: hexadecimal para decimal\n"); printf(" 3: decimal para octal\n"); printf(" 4: octal para decimal\n"); printf("informe a sua opao:"); scanf("%d", &opcao); } while(opcao<1||opcao>4); printf("Voce digitou a opao %d", opcao); }

Loops aninhados: /*Exibe uma tabela das quatro primeiras potencias dos numeros de 1 a 9.*/ #include<stdio.h> main() { int i, j, k, temp; printf(" i i^2

i^3

i^4\n");

18

for(i=1;i<10; i++){ /* lao externo: define o numero base*/ for(j=1; j<5; j++){ /* primeiro nivel do aninhamento */ temp = 1; for (k=0; k<j; k++) {temp = temp*i;} /* lao mais interno: eleva a j */ printf("%9d",temp); /* se um numero colocado entre o % e o d, ele */ } /* diz a printf() p/ adicionar os espaos neces- */ printf("\n"); /* srios largura especificada */ } } /* Programa de conversao de base numerica: versao final usando laos while aninhados. decimal --> hexadecimal hexadecimal --> decimal decimal --> octal octal --> decimal */ #include <stdio.h> main() { int opcao; int valor; /* repete ate que o usuario diga para terminar */ do{ /* certifica-se de que o usuario especifica uma opao valida */ do{ printf("\nConverter:\n"); printf(" 1: decimal para hexadecimal\n"); printf(" 2: hexadecimal para decimal\n"); printf(" 3: decimal para octal\n"); printf(" 4: octal para decimal\n"); printf("informe a sua opcao:"); scanf("%d", &opcao); } while(opcao<1||opcao>5); switch(opcao) { case 1: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em hexadecimal : %x", valor, valor); break; case 2: printf("informe um valor em hexadecimal:"); scanf("%x", &valor); printf("%x em decimal : %d", valor, valor); break; case 3: printf("informe um valor em decimal:"); scanf("%d", &valor); printf("%d em octal : %o", valor, valor); break; case 4: printf("informe um valor em octal:"); scanf("%o", &valor); printf("%o em decimal : %d", valor, valor); break; } printf("\n"); } while(opcao!=5); } Quebrando um loop:

19

/* imprime no ecr os numeros de 1 a 10 */ #include <stdio.h> main() { int t; for(t=0; t<100; t++){ printf("%d ", t); if(t==10) break; } } /* Como #include #include #include esta o seu senso de tempo? */ <stdio.h> <time.h> <conio.h>

main() { long tm; printf("Este programa testa o seu senso de tempo!\n"); printf("Quando estiver pronto, pressione a tecla return, espere cinco segundos\n"); printf("e pressione qualquer tecla."); getche(); printf("\n"); tm = time(0); for (;;) if(kbhit()) break; if(time(0)-tm==5) printf("Voce venceu!!!"); else printf("O seu sincronismo est mal. Erro de %d segundos!",(int) time(0)-tm-5); } Observe que um break causar uma sada somente do lao mais interno. Por exemplo: for(t=0; t<100; ++t){ count=1; for(;;){ printf("%d", count); count++; if(count==10) break; } } O cdigo acima imprimir os nmeros de 1 a 10 cem vezes no ecr. Toda vezque o break encontrado, o controle devolvido para o lao for externo. Outra observaao o fato que um break usado dentro de uma declaraao switch afetar somente os dados relacionados com o switch e nao qualquer outro lao em que o switch estiver. Declaraao continue: Continue opera da mesma maneira que o break, mas em vez de terminar o loop e seguir na declaraao seguinte, continue fora a ativaao da prxima iteraao do loop, pulando qualquer cdigo que esteja entre eles. Exemplos: /* Imprime no ecr os nmeros pares at 99 */ #include<stdio.h> main() { int x; for(x=0; x<100; x++){ if(x%2) continue; /* se o resto zero o loop continua */ printf("%d ", x); }

20

} /*Um codificador simples.*/ #include<stdio.h> #include<conio.h> main() { printf("Insira as letras que voce quer codificar.\n"); printf("Digite um $ quando estiver no final das letras.\n"); code(); }

code() /* Codifica as letras */ { char pronto, ch; pronto=0; while(!pronto){ ch=getch(); if(ch=='$'){ pronto=1; continue; } printf("%c", ch+1); /* desloca o alfabeto uma posiao */ } } Labels e goto: Semelhante ao "GOTO line" do BASIC e FORTRAN, com a diferena que o destino um LABEL (rtulo) e nao um nmero de linha do programa. Somente pode ser usado em desvios dentro da mesma funao, nao podendo transferir a execuao do programa de uma funao para outra. Exemplos: O cdigo abaixo implementa um loop de 1 a 100: x=1; loop1: /* 2 pontos aps o nome do label! (loop1) */ x++; if(x<100)goto loop1; O goto torna-se bastante til quando necessrio sair de uma rotina profundamente aninhada: for(...){ for(...){ while(...){ if(...)goto Lstop; . . . } } }

/* desvio condicional p/ o label Lstop */

21

Lstop: printf("erro no programa\n") Observe que se usarmos um break em vez do goto seremos forados a usar uma srie de testes adicionais, j que o break s sai do loop mais interno: stop=0; /* stop a varivel de controle da sada da situaao detetada pelo if */ for(...){ for(...){ while(...){ if(...) /* se ocorre determinada situaao stop=1 */ {stop=1; break;} } . . . } if(stop)break; } if(stop) break; } printf("erro no programa\n") 4) MATRIZES E STRINGS: Matrizes Unidimensionais: A forma geral de uma matriz de simples dimensao tipo var_nome[tamanho] "tipo" -------- declara o tipo base da matriz "tipo base" --- determina o tipo de dado de cada elemento que compreende a matriz "tamanho" ----- define quantos elementos a matriz armazenara int exemplo [10] Na linguagem C, todas as matrizes tem o zero como indice do seu primeiro elemento. Portanto, a linha anterior declara uma matriz que possui 10 elementos tipo inteiro, de "exemplo[0]" a "exemplo[9]". O programa seguinte carrega uma matriz de inteiros com os numeros de 0 a 9: main() { int x [10];/* esta declaraao reserva 10 elementos inteiros*/ int t; for(t=0; t<10; ++t) x[t]=t; }

Para uma matriz unidimensional, o tamanho total da matriz em bytes computado como mostrado aqui: "Bytes total = tamanho do tipo base * numero de elementos" /*Encontra a mdia de dez numeros. */ #include<stdio.h> main()

22

int exemplo [10], i, media; printf("\nInforme os nmeros X[i] a medida que forem solicitados:\n\n"); for(i=0; i<10; i++){ printf("X[%d]? ",i); scanf("%d", &exemplo[i]) ; } media=0; /* agora soma os numeros */ for (i=0; i<10; i++) media = media+exemplo[i]; printf("A media %f\n",(float)media/10);

} Sem verificaao de limites: /* programa incorreto. Nao execute-o!*/ main() { int quebra [10], i; for (i=0; i<100; i++) quebra [i] =i; /* excede o ndice de quebra[10] */ } Matrizes Unidimensionais Sao Listas: char str [7]: main() { int i; for(i=0; i<7; i++) str i = 'A'+i; } a varivel str ficar semelhante ao que segue: str[0] A Strings: Uma "string" definida como sendo constituida de uma matriz de caracteres (tipo de dado char - 1 byte) que terminada por um "nulo". Um nulo especificado usando-se '\0' que zero. Por exemplo, se quisermos declarar uma matriz "str" que possa armazenar uma string de 10 caracteres, escrevemos: char str [11]; Uma "constante string" uma lista de caracteres entremeados por aspas. Por exemplo: "Alo" "este um teste" Nao necessario adicionar manualmente o nulo no final das constantes string - o compilador faz isso automaticamente. Lendo uma string do teclado: A maneira mais fcil de inserir uma string pelo teclado com a funao de biblioteca gets(). A forma geral da funao gets() : gets(nome_da_matriz); /* Um programa simples com string.*/ #include <stdio.h> main() { str[1] B str[2] C str[3] D str[4] E str[5] F str[6] G

23

char str [80]; printf("informe uma string:"); gets(str); /*le a string do teclado*/ printf("%s", str); } Algumas funoes de biblioteca que manipulam string: A linguagem C suporta um grande numero de funoes que manipulam strings. As mais comuns sao: strcpy() strcat() strlen() strcmp() A funao strcpy(): Uma chamada a strcpy() tem esta forma geral: strcpy(para,de); A funao strcpy() usada para copiar o conteudo da string "de" para a string "para". /* exemplo de uso de strcpy() */ #include <stdio.h> #include<string.h> main() { char str [80]; strcpy(str, "Alo"); printf("%s", str); } A funao strcat(): Uma chamada a strcat() tem esta forma: strcat(s1,s2); A funao strcat() anexa (concatena) "s2" em " s1"; "s2" nao modificada. As duas strings devem ser terminadas com nulo e o resultado tambem ser terminado com um nulo. Por exemplo, este programa imprimir "Alo para todos" no ecr: #include <stdio.h> #include <string.h> main() { char s1[20], s2[15]; strcpy(s1, "Alo"); strcpy(s2," para todos"); strcat(s1,s2); printf("%s",s1); } Como strcpy(), strcat() retorna um valor, cujo significado ser visto adiante. A funao strcmp(): Uma chamada a strcmp() tem esta forma: strcmp(s1,s2); A funao strcmp() compara duas strings e retorna 0 se elas forem iguais. Se s1 lexicograficamente maior (Exemplos: "BBBB">"AAAA" e "AAA">"X") que s2, entao um numero positivo retornado; se for menor que s2, um numero negativo retornado.

24

A seguinte funao pode ser usada como uma rotina de verificaao de senha: /* Retorna verdadeiro se a senha aceita; falso, caso contrrio.*/ #include<stdio.h> main() { char s[80]; printf("informe a senha:"); gets(s); if(strcmp(s, "senha")) { /*strings diferentes*/ printf("senha invalida\n"); return 0; } /* a string comparada a mesma*/ return 1; } O segredo da utilizaao da funao "strcmp()" que ela retorna falso quando as strings sao iguais. Portanto, voce precisar usar o operador NOT se desejar que alguma coisa ocorra quando a string igual. /* le strings at que se digite 'sair' */ #include <stdio.h> #include <string.h> main() { char s[80]; for (;;){ printf("Informe uma string:"); gets(s); if(!strcmp("sair",s))break; } } A Funao strlen(): A forma geral da chamada "strlen()" : strlen(s); onde s uma string. A funao "strlen()" retorna o tamanho de s. O programa seguinte imprimir o tamanho da string que for inserida pelo /* imprime o tamanho da string digitada */ #include<stdio.h> #include<string.h> main() { char str [80]; printf("digite uma string:"); gets(str); printf("%d", strlen(str)); } /* Imprime uma string na ordem inversa.*/ #include <stdio.h> #include <string.h> main() { char str [80]; int i; printf("digite uma string:"); teclado:

25

gets(str); for(i=strlen(str)-1; i>=0; i--) printf("%c", str[i]); } /* informa tamanho e igualdade entre duas strings e concatena */ #include<stdio.h> #include<string.h> main() { char s1 [80], s2 [80]; printf("digite duas strings:"); gets(s1); gets(s2); printf("tamanhos: %d %d\n", strlen(s1), strlen(s2)); if(!strcmp(s1,s2))printf("As strings sao iguais\n"); strcat(s1,s2); printf("%s", s1); } Usando o terminador nulo: /* converte uma string para letras maisculas */ #include<stdio.h> #include<string.h> #include<ctype.h> main() { char str[80]; int i; printf("Informe uma string em letras minsculas:"); gets(str); for(i=0;str[i];i++) str[i] = toupper(str[i]); /* for termina quando */ printf("%s",str); /* str[i] = '\0' */ } Obs) O complementar de toupper() tolower(). Uma Variaao da Funao printf(): #include <stdio.h> #include <string.h> main() { char str[80]; strcpy(str, "Alo Tom"); printf(str);/*funciona porque o primeiro argumento de printf est entre aspas e pode ser considerado uma string*/ }

Matrizes Bidimensionais: Para declarar por exemplo uma matriz bidimensional de inteiros chamada de nome bid, com tamanho 10,20 , escrevemos: int bid[10] [20]; Exemplos: /* carrega uma matriz bidimensional com os nmeros de 1 a 12 */ #include<stdio.h>

26

main() { int t, i, num[3][4]; for(t=0; t<3; t++){ for(i=0;i<4;i++){ num[t][i]=(t*4)+i+1; /* lei de formaao dos elementos da matriz num */ printf("num[%d][%d]= %d\n",t,i,num[t][i]); } } } No caso de uma matriz bidimensional, a seguinte expressao encontrar o bytes=linha*coluna*tamanho do tipo de dado Portanto, uma matriz de inteiros com dimensoes [10][5] dever ter 10x5x2 ou 100 bytes alocados, j que um int tem o tamanho de 16bits = 2bytes (1 byte= 8 bits). Matrizes de strings: A linha abaixo declara uma matriz de 30 strings, cada uma com tamanho mximo de 80 caracteres: char matriz_str[30][80]; muito fcil acessar uma string individual: voce especifica somente o ndice da esquerda. Por exemplo, esta declaraao chama a funao gets() com a terceira string em "matriz_str": gets(matriz_str)[2]); O que equivalente a (embora nao tenhamos visto apontadores ainda): gets(&matriz_str[2][0]); Para entender melhor como as matrizes de caracteres funcionam, consideremos o programa seguinte, que aceita linhas de texto inseridas pelo teclado e exibe-as assim que uma linha em branco inserida: /*Recebe e exibe strings.*/ #include <stdio.h> main() { register int t, i; char texto [100][80]; for(t=0;t<100;t++){ printf("%d:",t); gets(texto[t]); /* t indexa linhas */ if(!texto[t][0]) break; /*sai com uma linha em branco*/ } /*reapresenta a string:*/ for(i=0;i<t;i++) printf("%s\n",texto[i]); /* i indexa linhas aqui */ } Matrizes Multidimensionais: A linguagem C permite matrizes com mais de duas dimensoes. A forma geral de uma matriz multidimensional : tipo_nome[tamanho1][tamanho2]...[tamanhoN]; Por exemplo, a limha abaixo cria uma matriz 4x10x3 de inteiros: int trid [4][10][3]; numero de bytes da memria:

27

Matrizes de trs ou mais dimensoes usualmente requerem uma quantidade grande de memoria para armazenamento. A rea requerida para armazenar elementos de matrizes alocada permanentemente durante a execuao do programa. Por exemplo, uma matriz quadridimensional de caracteres, com dimensoes 10, 6, 9, 4, deve requerer 10x6x9x4 ou 2.160 bytes, j que 1 char = 8 bits = 1 byte. Se a matriz for de inteiros de dois bytes, serao necesarios 4.320 bytes. Se a matriz for do tipo "double"(8 bytes de comprimento), entao 34.560 bytes serao requeridos. O espao necessrio aumenta exponencialmente com o nmero de dimensoes. Um programa com matrizes com mais de trs ou quatro dimensoes pode ficar rapidamente sem memria suficiente para ser executado, dependen do do tipo de modelo de memria usado pelo programa (usualmente usa-se o mo delo small, c/ 64Kb p/ cdigo e 64Kb p/ dados). Inicializaao de Matrizes: A linguagem C permite a inicializaao de matrizes. A forma geral da inicializaao de matrizes similar de outras variveis, como mostrado aqui: especificador_de_tipo nome da matriz[tamanho1]...[tamanhoN]={lista_de_valores}; A lista_de_valores uma lista de constantes separadas por vrgulas que do tipo compatvel com o tipo base da matriz. A primeira constante ser colocada na primeira posiao da matriz, a segunda constante, na segunda posiao e assim por diante. Note que um ponto e vrgula segue a chave }. No exemplo seguinte, uma matriz de 10 elementos inteiros inicializada como os nmeros de 1 a 10: int i[10]= {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; Isso significa que i[0] ter o valor 1 e i[9], o valor 10. Matrizes de caracteres que armazenam strings permitem uma inicializaao abreviada com a seguinte forma: char nome_da_matriz[tamanho]="string"; Por exemplo, este fragmento de cdigo inicializa a varivel "str" com a palavra "Al": char str[4] = "Al"; Que o mesmo que escrever: char str[4] = {'A', 'l', '','\0'}; J que as strings na linguagem C devem terminar com um nulo, certifique-se de que a matriz que voc declara grande o suficiente para inclu-lo. por isso que a varivel "str" tem o tamanho de quatro carateres apesar de "Al" ter somente trs. Quando a constante string usada,o compilador prov automaticamante o terminador nulo. Matrizes multidimensionais sao inicializadas da mesma maneira que as unidimensionais. Por exemplo, as linhas abaixo inicializam a matriz "sqrs" com os nmeros de 1 a 10 e seus respectivos quadrados: int sqrs[10][2] = { 1, 1, 2, 4, 3, 9, 4, 16, 5, 25, 6, 36, 7, 49, 8, 64, 9, 81, 10, 100, }; Inicializaao de Matrizes sem Especificaao de Tamanho: Imagine que voc esteja usando inicializaao de matriz para construir como mostrado aqui: char e1 [20] ="informaao invlida\n": uma tabela de mensagens de erros,

28

char e2 [23] = "seleao fora_do_limite|n"; char e3 [19] = "autorizaao negada\n"; Como voc pode supor, bastante montono contar manualmente os caracteres em cada mensagem para determinar a dimensao correta da matriz. possvel deixar a linguagem C dimensionar automaticamante as matrizes, neste exemplo, pelo uso de "matrizes sem determinaao de tamanho. Se, em uma declaraao de inicializaao de matriz, o tamanho nao especificado, o compilador criar uma matriz grande o suficiente para armazenar todos os inicializadores presentes. Usando este mtodo, a tabela de mensagem fica assim: char e1 [ ] ="informaao invalida\n": char e2 [ ] = "seleao fora_do_limite\n"; char e3 [ ] = "autorizaao negada\n"; /*O compilador conta ele prprio o nmero de elementos e cria as matrizes de tamanho adequado*/ A inicializaao de matrizes sem a especificaao de tamanho nao est restrita somente a matrizes unidimensionais. Para matrizes multidimensionais, deve-se especificar tudo, menos a dimensao mais esquerda (compiladores mais modernos aceitam especificao em aberto da dimenso mais direita) , para permiitr que a linguagem C indexe adequadamente a matriz. Dessa maneira, possvel construir tabelas de diferentes tamanhos com o compilador alocando espao suficiente para elas. Por exemplo, a declaraao da matriz "sqrs" como uma matriz sem especificaao de tamanho mostrada aqui: int sqrs[ ][2] = { 1, 1, 2, 4, 3, 9, 4, 16, 5, 25, 6, 36, 7, 49, 8, 64, 9, 81, 10, 100, }; Vantagem: A tabela pode ser aumentada indefinidamente porque o compilador se encarrega de contar as linhas da matriz. Um Jogo de Caa ao Submarino: No jogo de caa ao submarino o computador controla um submarino e voc um couraado. O objetivo do jogo uma embarcaao destruir a outra. Contudo, os navios tm capacidades de radar limitadas. A nica maneira do submarino encontrar o couraado estar diretamente sob ele. O vencedor aquele que encontrar o outro primeiro, com o submarino e o couraado alternando movimentos aleatrios sobre o "oceano". O "oceano" um pequeno campo de jogo que mede 76,2 x 76,2 mm. Tanto o submarino quanto o couraado tratam esse plano como coordenadas X, Y, com 0,0 sendo o canto superior esquerdo. Uma matriz bidimensional, chamada de "matriz", usada para guardar o tabuleiro de jogo e cada elemento nele inicializado com um espao, o que denota um elemento nao usado. A funao main() c/ suas variveis globais sao mostradas aqui: char matriz [3] [3] = { /* vars. globais */ ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', }; int compX, compY, jogX, jogY; main() { randomize(); /*inicializa (gera a semente inicial) aleatoriamente o gerador de nmeros randmicos*/ compX=compY=jogX=jogY=0; /* inicializa vars de coordenadas */

29

for(;;){ if(tent_sub()){ printf("O submarino venceu!\n"); break; } if(tent_jog()){ printf("O couraado (voc) venceu!\n"); break; } exibe_tabuleiro(); } } A funao randomize() serve para randomizar o gerador de nmero randmico da linguagem C, que usado para gerar o movimento do computador, como veremos em breve. A primeira funao necessria aquela que cria o movimento do computador. O computador (submarino) gera o seu movimento usando o gerador de nmeros randmicos da linguagem C rand(), que retorna um nmero randmico entre 0 e 32.767. Tanto a funao rand() como randomize() usam o arquivo cabealho (header) stdlib.h . A funao randomize() requer ainda o arquivo time.h . Um valor randmico gerado tanto para a coordenada X quanto para a Y, toda a vez que o computador chamado a gerar um movimento. Esses valores sao usados entao em uma operaao de mdulo para prover um nmero entre 0 e 2. Finalmente, o computador verifica se encontrou (e por definiao destruiu) o couraado. Se o movimento gerado tem as mesmas coordenadas da localizaao atual do couraado, o submarino vence. Se o computador venceu, a funao retorna verdadeira, caso contrrio retorna falso. A funao tent_sub() apresentada aqui: tent_sub() /* gera o prximo movimento do computador usando um gerador de nmeros randmicos.*/ { matriz [compX] [compY]= ' '; /* apaga ltima posiao do submarino */ compX=rand() % 3; /* o resto de uma divisao por 3 sempre 0 1 ou 2 */ compY=rand() % 3; if(matriz[compX][compY] == 'B') return 1; /* O submarino ganhou a batalha*/ else{ matriz[compX][compY]= 'S'; return 0; /* Ele nao acertou*/ } } O couraado recebe o seu prximo movimento do jogador.A funao requisita ao usurio as novas coordenadas e entao verifica se o submarino foi encontrado ou nao. Em caso positivo, o jogador vence e a funao retorna verdadeiro. Caso contrrio a funao retorna falso. A funao tent_jog() mostrada aqui: tent_jog() /*pega o prximo movimento do jogador*/ { matriz[jogX] [jogY]= ' '; /* apaga ltima posiao do couraado */ do{ printf("Informe as novas coordenadas (X,Y): "); scanf("%d%d",&jogX,&jogY); } while (jogX<0||jogX>2||jogY<0||jogY>2); /* testa validade das coords */ if(matriz[jogX][jogY]== 'S') return 1; /*O couraado ganhou a batalha*/ else{ matriz[jogX][jogY]='B'; return 0; /* Ele nao acertou*/ } } Aps cada srie de movimentos, o tabuleiro do jogo exibido pela funao "exibe_tabuleiro()". Uma clula nao usada fica vazia. A clula que guarda a ltima posiao do submarino conter um S e a clula que guarda a posiao do couraado, um B: exibe_tabuleiro() /*Exibe tabuleiro do jogo*/

30

printf("\n"); printf("%c | %c | %c\n", matriz[0][0],matriz[0][1],matriz[0][2]); printf("--|---|--\n"); printf("%c | %c | %c\n", matriz[1][0],matriz[1][1],matriz[1][2]); printf("--|---|--\n"); printf("%c | %c | %c\n", matriz[2][0],matriz[2][1],matriz[2][2]); O programa inteiro mostrado aqui: /* Jogo Couraado x Submarino */ #include <stdio.h> #include <stdlib.h> #include <time.h>/*requerido por randomize()*/ char matriz [3] [3] = { /* vars. globais */ ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', }; int compX, compY, jogX, jogY;

main() { randomize(); /*inicializa (gera a semente inicial) aleatoriamente o gerador de nmeros randmicos*/ compX=compY=jogX=jogY=0; for(;;){ if(tent_sub()){ printf("O submarino venceu!\n"); break; } if(tent_jog()){ printf("O couraado (voc) venceu!\n"); break; } exibe_tabuleiro(); } } tent_sub() /* gera o prximo movimento do computador usando um gerador de nmeros randmicos.*/ { matriz [compX] [compY]= ' '; compX=rand() % 3; compY=rand() % 3; if(matriz[compX][compY] == 'B') return 1; /* O submarino ganhou a batalha*/ else{ matriz[compX][compY]= 'S'; return 0; /* Ele nao acertou*/ } } tent_jog() /*pega o prximo movimento do jogador*/ { matriz[jogX] [jogY]= ' '; do{ printf("Informe as novas coordenadas (X,Y): "); scanf("%d%d",&jogX,&jogY); } while (jogX<0||jogX>2||jogY<0||jogY>2); if(matriz[jogX][jogY]== 'S') return 1; /*O couraado ganhou a batalha*/ else{ matriz[jogX][jogY]='B'; return 0; /* Ele nao acertou*/ } }

31

exibe_tabuleiro() /*Exibe tabuleiro do jogo*/ { printf("\n"); printf("%c | %c | %c\n", matriz[0][0],matriz[0][1],matriz[0][2]); printf("--|---|--\n"); printf("%c | %c | %c\n", matriz[1][0],matriz[1][1],matriz[1][2]); printf("--|---|--\n"); printf("%c | %c | %c\n", matriz[2][0],matriz[2][1],matriz[2][2]); } 5) APONTADORES: A compreenso e o uso correto de apontadores sao crticos para criaao de muitos programas de xito na linguagem C. H trs motivos para isso. Primeiro, os apontadores provem o meio para que as funoes possam modificar os seus argumentos de chamada. Segundo, eles sao usados para suportar rotinas de alocaao dinmica da linguagem C. Terceiro, podem ser substitudos pelas matrizes em muitas situaoes - proporcionando aumento de eficincia. Alm disso, muitas das caractersticas do C apiam-se firmemente nos apontadores, portanto, um completo entendimento de apontadores muito importante. Alm de ser uma das caractersticas mais fortes da linguagem C, os apontadores tambm sao muito perigosos. Por exemplo, quando nao-inicializados ou descuidados, eles podem causar o travamento do sistema. E pior: fcil usar apontadores incorretamente, o que causa "bugs" (erros) difceis de serem encontrados. Apontadores Sao Endereos: Um ponteiro uma varivel que guarda um endereo de memria de outro objeto. Mais comumente, esse endereo a localizaao de outra varivel na memria, embora ele possa ser o endereo de uma porta ou de propsito-especial na RAM, como uma rea auxiliar (buffer). Se uma varivel contm o endereo de uma outra varivel, a primeira conhecida como um ponteiro para a segunda. Essa situaao ilustrada abaixo:

Seja o seguinte cdigo: float mult= 315.2; int sum = 32000; char letter = A, /* equivalente a char letter = 65 */ int *set; /* declara o ponteiro set */ float *ptr1; set=&sum; /* inicializa pointer set */ ptr1=&mult; /* inicializa pointer ptr1 */ O cdigo anterior gera o seguinte mapa de memria: Endereo de memria: (hexadecimal) FA10 FA11 FA12 FA13 FA14 FA15 FA16 FA17 FA18 FA19 Valor da varivel na memria (decimal exceto indicao contrria): 3.152 E 02 mult ( 4 bytes)

32000 65 FA14 (hex) FA10 (hex)

sum ( 2 bytes) letter (1 byte) Valor ASCII do caracter A set ( 2 bytes) set aponta para sum ptr1 ( 2 bytes) ptr1 aponta para mult

32

FA1A

Variveis Ponteiro: Se uma varivel vai armazenar um ponteiro, ela deve ser declarada como tal. Uma declaraao de ponteiro consiste em um tipo base, um * e o nome da varivel. A forma geral para declaraao de uma varivel ponteiro : tipo * nome-da-varivel; onde o tipo pode ser qualquer tipo vlido da linguagem C e o nome-da-varivel o nome da varivel ponteiro. O tipo base do ponteiro define qual tipo de variveis o ponteiro pode apontar. Por exemplo, estas declaraoes criam um ponteiro caractere e dois apontadores inteiros. char *p; int *temp, *incio; Os Operadores de Apontadores: Existem dois operadores especiais de ponteiro: o & e o *. O & um operador unrio que retorna o endereo de memria do seu operando. (Um operador unrio requer somente um operando.) Por exemplo, count_addr = &count; coloca em count_addr o endereo de memria da varivel count. Esse endereo a localizaao interna da varivel no computador. Ele nao faz nada com o valor de count. A operaao do operador & pode ser lembrada como "o retorno de endereo da varivel que a ele sucede". Portanto, a atribuiao dada pode ser determinada como "a varivel count_addr recebe o endereo de varivel count". Para entender melhor essa atribuiao, assuma que a varivel count esteja localizada no endereo 2000. Entao, apos a atribuiao, count_addr ter o valor de 2000. O segundo operador o *, e o complemento do operador &. Ele um operador unrio que retorna o valor da varivel localizada no endereo que o segue. Por exemplo, se count_addr contm o endereo de memria da varivel count, entao val = *count_addr; colocar o valor de count em val. Se count originalmente tem o valor 100, entao val ter o valor de 100, porque este o valor armazenado na localizaao 2000, que o endereo de memria que foi atribudo a count_addr. A operaao do operador * pode ser lembrada como "valor contido no endereo". Neste caso, entao, a declaraao pode ser lida como "val recebe o valor que est no endereo count_addr". Com relaao ao mapeamento de memria anterior, envolvendo as variveis mult ,sum, letter e os apontadores set e ptr1, se tivessemos feito a declaraao de uma outra varivel inteira, digamos, de nome teste, e fizssemos a seguinte declaraao: teste= *set; /* teste recebe o valor da varivel p/ a qual set aponta */ teste receberia o valor de sum que 32000. De mesma maneira se teste fosse declarada como float e fizssemos teste = *ptr1; teste receberia o valor de mult que 315.2. Infelizmente, o sinal de multiplicaao e o sinal "valor no endereo" sao os mesmos. Isso algumas vezes confunde iniciantes na linguagem C. Esses operadores nao se relacionam um com o outro. Tanto & como * tm precedncia maior que todos os outros operadores aritmticos, exceto o menos unrio, com o qual eles se igualam. Aqui est um programa que usa esses operadores para imprimir o nmero 100 no ecr: /* imprime 100 no ecr */

33

#include <stdio.h> main() { int *count_addr, count, val; count = 100; count_addr= & count; /* pega o endereo de count */ val=*count_addr; /*pega o valor que est no endereo count_addr*/ printf("%d", val); /*exibe 100*/ }

Tipo Base: Como o compilador sabe quantos bytes copiar em val do endereo apontado por count_addr? Mais genericamente, como o compilador transfere o nmero adequado de bytes em qualquer atribuiao usando um ponteiro? A resposta que o tipo base do ponteiro determina o tipo de dado que o compilador assume que o ponteiro est apontando. Nesse caso, uma vez que count_addr um ponteiro inteiro, 2 bytes de informaao sao copiados em val a partir do endereo apontado por count_addr. Se ele fosse um ponteiro double, entao 8 bytes teriam sido copiados. #include <stdio.h> /* Este programa nao funciona adequadamente */ main() { float x=10.1, y; int *p; p=&x; /* observar warning emitido pelo compilador */ y=*p; printf("%f",y); } Este programa nao atribuir o valor da varivel x para y. Tendo em vista que p declarado para ser um ponteiro para um inteiro, somente 2 bytes de informaao serao transferidos para y, nao os 4 bytes que normalmente formam um nmero de ponto flutuante. Expressoes com Apontadores: Atribuiao de Apontadores: Como qualquer varivel, um ponteiro pode ser usado no lado direito de uma declaraao para atribuir seu valor para um outro ponteiro. Por exemplo: #include<stdio.h> main() { int x; int *p1, *p2; x=101; p1=&x; p2=p1; printf("Na localizaao %p ", p2); /* imprime o valor hexadecimal do endereo */ /* de x e nao o valor de x */ printf("est o valor %d\n", *p2); /* agora imprime o valor de x */ } O endereo, em hexadecimal, de x exibido usando-se outro dos cdigos de formataao da funao printf(). O %p especifica que um endereo de ponteiro para ser exibido usando-se a notaao hexadecimal. Aritmtica com Apontadores: Somente duas operaoes aritmticas podem ser usadas com apontadores: adiao e subtraao. Para entender o que ocorre na aritmtica com apontadores, consideremos que p1 seja um ponteiro para um inteiro com o valor atual 2000.

34

Depois da expressao p1++; o contedo de p1 ser 2002, nao 2001. Cada vez que incrementado, p1 apontar para o prximo inteiro. O mesmo vlido para decrementos. Por exemplo, p1--; far com que a varivel p1 tenha o valor 1998, assumindo-se que ela tinha anteriormente o valor 2000. Cada vez que incrementado, um ponteiro apontar para a prxima localizaao de memria do seu tipo base. Cada vez que decrementado, apontar para a localizaao do elemento anterior. No caso de apontadores para caracteres, parecer ser uma aritmtica normal. Porm, todos os outros apontadores incrementarao ou decrementarao pelo tamanho do tipo dado para o qual apontam. Por exemplo, assumindo-se caracteres de 1 byte e inteiros de 2 bytes, quando um ponteiro caractere incrementado, seu valor incrementado de um; entretanto , quando um ponteiro inteiro incrementado, seu valor incrementado de dois bytes. Um ponteiro float seria increnentado de 4 bytes. Exemplo: char *chptr; int *iptr; float *flptr; chptr=0x3000; iptr=0x6000; flptr=0x9000; chptr++; /* ch=0x3001 */ iptr++; /* i=0x6002 */ flptr++; /* flptr=0x9004 */ Porm, voc nao est limitado a somente incrementar e decrementar. Pode-se inteiros para ou de apontadores. A expressao: p1=p1+9; far p1 apontar para o nono elemento do tipo base de p1 considerando-se que ele est apontando para o primeiro. Exemplo: double *p1; p1=0x1000; /* p1 = 4096 */ p1=p1+9 /* p1 = 0x1048 = 4096 + 9*8 */ Comparaao com Apontadores: possvel comparar dois apontadores em uma expressao relacional. Por exemplo, dados dois apontadores p e q, a seguinte declaraao perfeitamente vlida: if(p<q) printf("p aponta para um endereo de memria mais baixo que q\n"); Geralmente, comparaoes entre apontadores devem ser usadas somente quando dois ou mais apontadores estao apontando para um objeto em comum. Apontadores e Matrizes: H uma relaao entre apontadores e matrizes. Considere este fragmento de cdigo: char str[80], *p1; p1 = str; Aqui, p1 foi associado ao endereo do primeiro elemento da matriz em str. Na linguagem C, um nome de matriz sem um ndice o endereo para o comeo da matriz (em geral um ponteiro contm o endereo do adicionar ou subtrair

35

incio de uma rea de armazenamento ou transferncia de dados). O mesmo resultado - um ponteiro para o primeiro elemento da matriz str - pode ser gerado com a declaraao abaixo: p1=&str[0]; Entretanto, ela considerada uma forma pobre por muitos programadores em C. quinto elemento em str, pode escrever: str[4] ou *(p1+4) As duas declaraoes retornarao o quinto elemento. LEMBRE-SE: Matrizes comeam em ndice zero, entao um 4 usado para indexar str. Voc tambm pode adicionar 4 ao ponteiro p1 para obter o quinto elemento, uma vez que p1 aponta atualmente para o primeiro elemento de str. A linguagem C permite essencialmente dois mtodos para acessar os elementos de uma matriz. Isso importante porque a aritmtica de apontadores pode ser mais rpida do que a indexaao de matriz. Uma vez que a velocidade freqentemente importante em programaao, o uso de apontadores para acessar elementos da matriz muito comum em programas em C. Para ver um exemplo de como apontadores podem ser usados no lugar de indexaao de matriz, considere estes dois programas - um com indexaao de matriz e um com apontadores - , que exibem o contedo de uma string em letras minsculas. #include <stdio.h> #include <ctype.h> /* versao matriz */ main() { char str[80]; int i; printf("digite uma string em letra maiscula: "); gets(str); printf("aqui est a string em letra minscula: "); for(i=0; str[i]; i++) printf("%c", tolower(str[i])); } #include <stdio.h> #include <ctype.h> /* versao ponteiro */ main() { char str[80], *p; printf("digite uma string em letra maiscula: "); gets(str); printf("aqui est a string em letra minscula: "); p=str; /* obtm o endereo de str*/ while (*p) printf("%c", tolower(*p++)); } A versao matriz mais lenta que a versao ponteiro porque mais demorado indexar uma matriz do que usar o operador *. Algumas vezes, programadores iniciantes na linguagem C cometem erros pensando que nunca devem usar a indexaao de matrizes, j que apontadores sao muito mais eficientes. Mas nao o caso. Se a matriz ser acessada em ordem estritamente ascendente ou descendente, apontadores sao mais rpidos e fceis de usar. Entretanto, se a matriz ser acessada randomicamente, a indexaao da matriz pode ser uma melhor opao, pois geralmente ser tao rpida quanto a avaliaao de uma expressao complexa de apontadores, alm de ser mais fcil de codificar e entender. Indexando um Ponteiro: Se voc deseja acessar o

36

Em C, possvel indexar um ponteiro como se ele fosse uma matriz. Isso estreita ainda mais o relacionamento entre apontadores e matrizes. Por exemplo, este fragmento de cdigo perfeitamente vlido e imprime os nmeros de 1 at 5 no ecr: /* Indexando um ponteiro semelhante a uma matriz */ #include <stdio.h> main() { int i[5]={1, 2, 3, 4, 5}; int *p,t; p=i; for(t=0; t<5; t++) printf("%d ",*(p+t)); }

Em C, a declaraao p[t] idntica a *(p+t).

Apontadores e Strings: Uma vez que o nome de uma matriz sem um ndice um ponteiro para o primeiro elemento da matriz, o que est realmante acontecendo quando voc usa as funoes de string discutidas nos captulos anteriores que somente um ponteiro para strings passado para funao, e nao a prpria string. Para ver como isso funciona, aqui est uma maneira como a funao strlen() pode ser escrita:

strlen(char *s) { int i=0; while(*s){ i++; s++; } return i; } Quando uma constante string usada em qualquer tipo de expressao, ela tratada como um ponteiro para o primeiro carctere na string. Por exemplo, este programa perfeitamente vlido e imprime a frase "este programa funciona" no ecr: #include <stdio.h> main() { char *s; s= "este programa funciona"; printf(s) } Obtendo o Endereo de um elemento da Matriz: possvel atribuir o endereo de um elemento especfico de uma matriz aplicando-se o operador & para uma matriz indexada. Por exemplo, este fragmento coloca o endereo do terceiro elemento de x em p: p= &x[2]; Essa declaraao especialmente til para encontrar uma substring. Por exemplo, este programa imprimir o restante de uma string, que inserida pelo teclado, a partir do ponto onde o primeiro espao encontrado: #include <stdio.h> /* Exibe a string direita depois que o primeiro espao encontrado.*/ main() { char s[80]; char *p; int i;

37

printf("digite uma string: "); gets(s); /* encontra o primeiro espao ou o fim da string */ for (i=0; s[i] && s[i]!=' '; i++); p=&s[i]; printf(p); } Esse programa funciona, uma vez que p estar apontando para um espao ou para um nulo (se a string nao contiver espaos). Se h um espao, o restante da string ser impresso. Se h um nulo, a funao printf() nao imprime nada. Por exemplo, se "Ol a todos" for digitado, "a todos" ser exibido.

Matrizes de Apontadores: Podem existir matrizes de apontadores como acontece com qualquer outro tipo de dado. A declaraao para uma matriz de apontadores, do tipo int, de tamanho 10 : int *x[10]; Para atribuir o endereo de uma apontadores, voc deve escrever: x[2] = &var; Para encontrar o valor de var, voc deve escrever: *x[2]; Um uso bastante comum de matrizes de apontadores armazenar apontadores para mensagens de erros. Voc pode criar uma funao que emite uma mensagem, dado o seu nmero de cdigo, como mostrado na funao serror() abaixo: serror(int num) { char *err[]={ "nao possso abrir o arquivo\n", "erro de leitura\n", "erro de escrita\n", "falha na mdia\n" }; printf("%s",err[num]); } Como voc pode ver, a funao printf() chamada dentro da funao serror() com um ponteiro do tipo caractere, que aponta para uma das vrias mensagens de erro indexadas pelo nmero do erro passado para a funao. Por exemplo, se num passado com valor 2, a mensagem de erro de escrita exibida. Uma outra aplicaao interessante para matrizes de apontadores de caracteres inicializadas usa a funao de linguagem C system(),que permite ao seu programa enviar um comando ao sistema operacional. Uma chamada a system() tem esta forma geral: system("comando"); onde comando um comando do sistema operacional a ser executado, inclusive outro programa. Por exemplo, assumindo-se o ambiente DOS, esta declaraao faz com que o diretrio corrente seja exibido: system("DIR"); O programa seguinte implementa uma pequena interface com o usurio dirigida por menu que pode executar quatro comandos do DOS: DIR, CHKDSK, TIME e DATE. /* Uma interface simples do sistema do sistema operacional com o usurio dirigida por menu.*/ varivel ponteiro chamada var ao terceiro elemento da matriz de

38

#include <stdio.h> #include<stdlib.h> #include <conio.h> main() { /* cria uma matriz de strings */ char *comando[] = { "DIR", "CHKDSK", "TIME", "DATE" }; char ch; for(;;){ do { printf("1: diretrio\n"); printf("2: verifica o disco\n"); printf("3: atualiza a hora\n"); printf("4: atualiza a data\n"); printf("5: sair\n"); printf("\nopao: "); ch=getche(); } while ((ch<'1') || (ch>'5')); if(ch=='5')break; /* fim*/ /* executa o comando especfico */ printf("\n"); system(comando[ch-'1']); } } Qualquer comando do DOS, inclusive a chamada a outros programas pode ser implementada desta maneira. Simples Tradutor Ingls-Portugus: A primeira coisa que voc precisa a tabela ingls-portugus mostrada aqui, entretanto, voc livre para expand-la se assim o desejar. char trans[][20] = { "is", "", "this", "isto", "not", "nao", "a", "um", "book", "livro", "apple", "maa", "I", "eu", "bread", "pao", "drive", "dirigir", "to", "para", "buy", "comprar", "","" } Cada palavra em ingls colocada em par com a equivalente em portugus. Note que a palavra mais longa nao excede 19 caracteres. A funao main() do programa de traduao mostrada aqui junto com as variveis globais necessrias: char entrada[80]; char palavra[80]; char *p; main() { int loc;

39

printf("Informe a sentena em ingls; "); gets(entrada); p = entrada; /* d a p o endereo da matriz entrada */ printf(traduao rstica para o portugus: "); pega_palavra(); /* pega a primeira palavra */ /* Este o lao principal. Ele l uma palavra por vez da matriz entrada e traduz para o portugus.*/ do { /* procura o ndice da palavra em ingls em trans */ loc = verifica(palavra); /* imprime a palavra em portugus se uma correspondncia encontrada*/ if(loc!=-1) printf("%s ", trans[loc+1]); else printf("<desconhecida> "); pega_palavra(); /* pega a prxima palavra / } while(*palavra); /*repete at que uma string nula encontrada */ } O programa opera desta maneira: primeiro, o usurio solicitado a informar uma sentena em ingls. Essa sentena lida na string entrada. O ponteiro p , entao, associado ao endereo do incio de entrada e usado pela funao pega_palavra() para ler uma palavra por vez da string entrada e coloc-la na matriz palavra. O lao principal verifica, entao, cada palavra, usando a funao verifica(), e retorna o ndice da palavra em ingls ou -1 se a palavra nao estiver na tabela. Adicionando-se um a este ndice, a palavra correspondente em portugus encontrada. A funao verifica() mostrada aqui: /* Esta funao retorna a localizaao de uma correspondncia entre a string apontada pelo parmetro s e a matriz trans.*/ verifica(char *s) { int i; for(i=0; *trans[i]; i++) if(!strcmp(trans[i], s)) break; if(*trans[i]) return i; else return -1; } Voc chama verifica() com um ponteiro para a palavra em ingls e a funao retorna o seu ndice se ela estiver na tabela e -1 se nao for encontrada. A funao pega_palavra() mostrada a seguir. Da maneira como a funao foi concebida, as palavras sao delimitadas somente por espaos ou por terminador nulo. /* Esta funao ler a prxima palavra da matriz entrada. Cada palavra considerada como sendo separada por um espao ou pelo terminador nulo. Nenhuma outra pontuaao permita. A palavra retornada sr uma string de tamanho nulo quando o final da string entrada encontrado. */ pega_palavra() { char *q; /* recarrega o endereo da palavra toda vez que a funao chamada*/ q = palavra; /* pega a prxima palavra*/ while(*p && *p!=' ') { *q = *p; p++; q++; } if(*p==' ')p++; *q = '\0'; /* termina cada palavra com um terminador nulo */ } Assim que retorna da funao pega_palavra() a varivel global conter ou a prxima palavra em ingls na sentena ou um nulo. O programa completo da traduao mostrado aqui:

40

/* Um tradutor (bastante) simples de ingls para portugus.*/ /* Sugestao: Colocar Watches em: entrada<->p , palavra<->q , loc */ #include<stdio.h> #include<string.h> char trans[][20] = { "is", "", "this", "isto", "not", "nao", "a", "um", "book", "livro", "apple", "maa", "I", "eu", "bread", "pao", "drive", "dirigir", "to", "para", "buy", "comprar", "","" }; char entrada[80]; char palavra[80]; char *p; main() { int loc; printf("Informe a sentena em ingls: "); gets(entrada); p = entrada; /* d a p o endereo da matriz entrada */ printf("traduao rstica para o portugus: "); pega_palavra(); /* pega a primeira palavra */ /* Este o lao principal. Ele l uma palavra por vez da matriz entrada e traduz para o portugus.*/ do { /* procura o ndice da palavra em ingls em trans */ loc = verifica(palavra); /* imprime a palavra em portugus se uma correspondncia encontrada*/ if(loc!=-1) printf("%s ", trans[loc+1]); else printf("<desconhecida> "); pega_palavra(); /* pega a prxima palavra */ } while(*palavra); /*repete at que uma string nula encontrada */ } /* Esta funao retorna a localizaao de uma correspondncia entre a string apontada pelo parmetro s e a matriz trans.*/ verifica(char *s) { int i; for(i=0; *trans[i]; i++) if(!strcmp(trans[i], s)) break; if(*trans[i]) return i; /* se nao o fim da matriz trans retorna i */ else return -1; /* se , retorna -1 (desconhecida) */ } /* Esta funao ler a prxima palavra da matriz entrada. Cada palavra considerada como sendo separada por um espao ou pelo terminador nulo. Nenhuma outra pontuaao permitida. A palavra retornada ser uma string de tamanho nulo quando o final da string entrada encontrado. */ pega_palavra() { char *q; /* recarrega o endereo da palavra toda vez que a funao chamada*/ q = palavra; /* palavra global */ /* pega a prxima palavra: */

41

while(*p && *p!=' ') { /* p global */ *q = *p; p++; q++; } if(*p==' ')p++; *q = '\0'; /* termina cada palavra com um terminador nulo */ } Apontadores para Apontadores: Indireo simples: Ponteiro endereo Indireo mltipla: Ponteiro endereo Ponteiro endereo Varivel valor Varivel valor

Uma varivel que um ponteiro para um ponteiro deve ser declarada como tal. Isso feito colocando-se um asterisco adicional na frente do seu nome. Por exemplo, esta declaraao diz ao compilador que novobalano um ponteiro para um ponteiro do tipo float: float **novobalano; importante entender que novobalano nao um ponteiro para um nmero de ponto flutuante, mas, antes, um ponteiro para um ponteiro float. Para acessar o valor desejado apontado indiretamente por um ponteiro para um ponteiro, o operador asterisco deve ser aplicado duas vezes, como mostrado no pequeno exemplo a seguir: #include <stdio.h> main() { int x, *p, **q; x =10; p = &x; q = &p; printf("%d", **q); }

/* imprime o valor de x */

Aqui, p declarado como um ponteiro para um inteiro e q como um ponteiro para um ponteiro para um inteiro. A chamada a printf() imprimir o nmero 10 no ecr. Inicializando Apontadores: Depois que declarado, mas antes que tenha sido associado a um valor, um ponteiro conter um valor desconhecido. Se tentarmos usar um ponteiro antes de dar a ele um valor, provavelmente nao s travar o programa como tambm o sistema operacional - um tipo de erro muito desagradvel! Por convenao, um ponteiro que est apontando para nenhum lugar deve ter valor nulo. Entretanto, somente o fato de um ponteiro ter um valor nulo nao o torna "seguro". Se usarmos um ponteiro nulo no lado esquerdo de uma declaraao de atribuiao, correremos o risco de travar o programa e o sistema operacional. Como um ponteiro nulo considerado intil, podemos us-lo para tornar muitas rotinas de apontadores fceis de codificar e mais eficientes. Por exemplo, podemos usar um ponteiro nulo para marcar o final de uma matriz de apontadores. Se isso feito, a rotina que acessa aquela matriz saber que chegou ao fim quando um valor nulo encontrado.Esse tipo de abordagem ilustrado pelo lao for mostrado aqui:

42

/* Observe um nome assumindo o ltimo elemento de p como um nulo.*/ for(t=0; p[t]; ++t) if(!strcmp(p[t], nome)) break; O lao ser executado at que seja encontrada uma correspondncia ou um ponteiro nulo. Uma vez que o fim da matriz est marcado com um nulo, a condiao que controla o lao falhar quando ele for encontrado. uma prtica muito comum em programas profissionais escritos na linguagem C inicializar strings. Vimos dois exemplos anteriores neste captulo na seao de matrizes de apontadores. Outra variaao desse tema o seguinte tipo de declaraao de string: char *p="Al mundo\n"; Esse tipo de declaraao coloca o endereo da string "Al mundo" no ponteiro p. Por exemplo, o seguinte programa perfeitamente vlido: #include <stdio.h> #include <string.h> char *p="Al mundo"; main() { register int i; /* imprime a string normal e inversa*/ printf(p); for(t=strlen(p)-1; t>-1; t--) printf("%c",p[t]); } Problemas com Apontadores: Um problema com um ponteiro difcil de ser encontrado. O ponteiro por si s nao um problema; o problema que, cada vez que for realizada uma operaao usando-o, pode-se estar lendo ou escrevendo para algum lugar desconhecido da memria. Se voc ler, o pior que lhe pode acontecer ler informaao invlida, "lixo". Entretanto, se voc escrever para o ponteiro, estar escrevendo sobre outros pedaos de cdigo ou dados. Isso pode nao ser mostrado mais tarde na execuao do programa, levando-o a procurar pelo erro em lugar errado. Pode ser pouco ou nao certo sugerir que o ponteiro um problema. Esse tipo de erro tem feito com que programadores percam muitas noites de sono. Tendo em vista que erros com apontadores podem se transformar em pesadelos, voc deve fazer o mximo para nunca gerar um! Alguns dos erros mais comuns sao discutidos aqui, comeando com o exemplo clssico de erro com ponteiro: o ponteiro-nao-inicializado. Considere o seguinte: /* Este programa est errado. Nao o execute.*/ main() { int x, *p; x =10; *p = x; } Esse programa atribui o valor 10 a alguma localizaao desconhecida na memria. Ao ponteiro p nunca foi dado um valor; portanto ele contm um valor, "lixo". Embora o Turbo C emita uma mensagem de aviso neste exemplo, o mesmo tipo de problema surge quando um ponteiro est simplesmente apontado para um lugar indevido. Por exemplo, voc pode acidentalmente atribuir um ponteiro para um endereo errado. Esse tipo de problema freqentemente passa despercebido quando o seu programa muito pequeno, por causa da probabilidade de p conter um endereo "seguro" - um endereo que nao esteja no seu cdigo, na rea de dados ou no sistema operacional. Entretanto, medida que o seu programa cresce, a probabilidade de p apontar para algum lugar fundamental aumenta. Eventualmente, o seu programa pra de funcionar. Para evitar esse tipo de problema, certifique-se sempre de que um ponteiro esteja apontado para alguma posiao vlida antes que seja usado. Um segundo erro causado pelo mal-entendeido sobre como usar um ponteiro. Considere o seguinte:

43

#include <stdio.h> /* Este programa est incorreto. Nao o execute. */ main() { int x, *p; x = 10; p = x; printf("%d", *p); } A chamada funao printf() nao imprimir o valor de x, que 10, no ecr. desconhecido. O motivo para isso que a atribuiao p = x; est errada. Aquela declaraao atribui o valor 10 para o ponteiro p, que supostamente contm um endereo, nao um valor. Para tornar o programa correto, voc deve escrever: p = &x; Neste exemplo, o Turbo C avisar voc sobre o erro no programa.Entretanto, nem todos erros desse tipo geral podem ser verificados pelo compilador. 6) FUNOES: A forma Geral de uma Funao: A forma geral de uma funao : especificador-de-tipo nome-da-funao(declaraao de parmetros) { corpo da funao } O especificador de tipo determina o tipo de valor que a funao retornar usando a declaraao return. Ele pode ser qualquer tipo vlido. Se nenhum tipo especificado, entao, por definiao, a funao retorna um resultado inteiro. A lista de declaraao de parmetros uma lista de tipos de variveis e nomes, separados por vrgula e que receberao os valores dos argumentos quando a funao for chamada. Uma funao pode estar sem parmetro, caso em que a lista de parmetros estar vazia (argumento void). Contudo, mesmo se nao houver parmetros, os parnteses ainda sao requeridos. Imprimir algum valor

A Declaraao Return: A declaraao return tem dois importantes usos. Primeiro, ela far com que haja uma sada imediata da funao corrente. Isto , uma declaraao return far com que a execuao do programa retorne para o cdigo chamador assim que ele encontrado. Segundo, ela pode ser usada para retornar um valor. Esses dois recursos serao examinados aqui. Retornando de uma Funao: Existem duas maneiras de uma funao terminar a execuao e retornar ao seu chamador. A primeira quando a ltima declaraao na funao executada e, conceitualmente, o finalizador da funao } encontrado. Por exemplo, esta funao simples imprime uma string inversa no ecr: void pr_inverso(char *s) { register int t; for(t=strlen(s)-1; t>-1; t--) printf("%c", s[t]); } Uma vez que a string tiver sido exibida, nao h nada para a funao fazer a seguir, entao ela retorna ao lugar de onde foi chamada.

44

A segunda maneira de uma funao poder retornar pelo uso da declaraao return. A declaraao return pode ser usada sem qualquer valor associado a ela. Por exemplo, a seguinte funao imprime o resultado de um nmero elevado a uma potncia positiva. Se o expoente negativo, a declaraao return faz com que a funao termine antes que a chave final seja encontrada, mas nenhum valor retornado. void potncia(int base, int exp) { int i if(exp<0) return; /* nao pode calcular com expoente negativo */ i = 1; for( ; exp; exp--0 i = base * i); printf("A resposta : %d ", i); } Retornando um Valor: max(int a, int b) { int temp; if(a>b) temp = a; else temp = b; return temp; } Note que uma funao retorna um valor inteiro. Esse o tipo de retorno padrao de uma funao se nenhum outro explicitamente especificado na definiao da funao. possvel uma funao conter duas ou mais declaraoes return. Por exemplo, a funao max() melhor escrita como mostrado aqui: max(int a, int b) if(a>b) return a; else return b; } Outro exemplo de funao que usa duas declaraoes return a encontra_substr(), mostrada aqui. Ela retorna o ndice de incio de uma substring dentro de uma string ou -1, se nenhuma correspondncia encontrada. Sem o uso da segunda declaraao return, uma varivel temporria e cdigo extra precisariam ser usados. encontra_substr(char *sub, char *str) { register int t; char *p, *p2; for(t=0; str[t]; t++) { p = &str[t]; /* pega o ponto de incio de str */ p2 = sub; while(*p2 && *p2==*p) { /* enquanto nao for final de sub e for */ p++; /* verificado igualdade entre os caracteres */ p2++; /* de sub e str, avana. */ } if(!*p2) return t; /* se est no final de sub, entao a correspondncia foi encontrada*/ } return -1; } Se essa funao chamada com a substring contendo "dos" e a string principal contendo "Ol a todos", a funao retornar o valor 8. Todas as funoes, exceto aquelas declaradas para serem do tipo void, retornam um valor. Esse valor explicitamente especificado pela declaraao return ou desconhecido, se nenhuma declaraao return especificada. Isso significa que uma funao pode ser usada como um operando em qualquer expressao vlida em C. Portanto, cada uma das seguintes expressoes sao vlidas em C: x = abs(y);

45

if(max(x,y) > 100) printf("excelente"); for(ch=getchar(); isdigit(ch);)...; Entretanto, uma funao nao pode ser alvo de uma atribuiao. Uma declaraao como swap(x, y)=100 ; /* declaraao incorreta */ errada. O turbo C sinalizar com um erro e nao compilar um programa que contenha tal declaraao. Funoes Retornando Valores Nao-Inteiros: Quando o tipo de uma funao nao explicitamente declarado, ele automaticamente definido como int. Como exemplo introdutrio de uma funao retornando um tipo nao-inteiro, aqui est um programa que usa a funao sum(), que retorna um valor do tipo double que a soma de seus dois argumentos double: #include <stdio.h> double sum(double a, double b); /* declaraao inicial sempre necessria */ /* quando a funao nao retorna int */ main() { double primeiro, segundo; primeiro = 1023.23; segundo = 990.0; printf("%lf", sum(primeiro, segundo)); /* %lf o especificador p/ double */ } double sum(double a, double b) /*retorna um float */ { return a+b; } O prottipo antes de main() diz ao compilador que a funao sum() retornar um tipo de dado double em ponto flutuante. /* clculo da rea de um crculo */ #include <stdlib.h> #include <math.h> float area(float raio); /* prottipo*/ main() { float r,res; printf("Informe o raio: "); scanf("%f", &r); res=area(r); printf("A rea : %f\n", res); } float area(float raio) { return M_PI * pow(raio,2); /* M_PI amacro p/ o valor de pi em math.h */ } /* pow(x,y) igual ao x^y do BASIC e FORTRAN */ /* e definida em math.h */ Obs1) Qualquer funao em math.h devolve um valor double cujo valor mximo 1.7E308. Estamos atribuindo o retorno de *pow(raio,2) varivel float res (valor mximo = 3.4E38). Portanto devemos esperar overflow da funao area para raios maiores que 1.04E19 = sqrt(3.4E38/). A funao pow em si s acusar overflow para valores de retorno acima de 1.7E308 . Obs2) A declaraao dos argumentos no prottipo antes de main() opcional.

46

Funoes que retornam apontadores: Exemplos: /* testa correspondencia entre uma string e um caractere */ #include<stdio.h> char *corresp(); main() { char alfa[80]; char ch, *p; int loc; printf("\nDigite uma string: "); gets(alfa); printf("Digite um caractere: "); ch=getche(); p= corresp(ch,alfa); loc=strlen(alfa)-strlen(p)+1; /* d a localizaao da correspondencia */ if(p) printf("\nExiste correspondencia no %d caractere.",loc); /* existe correspondencia */ else printf("\nNenhuma correspondencia encontrada."); } /* Retorna um ponteiro p/ o primeiro caractere em s que corresponde a c: */ char *corresp(char c, char *s) { int count; count=0; while(c!=s[count]&&s[count] != '\0') count++; /* testa igualdade e nulo */ if(s[count]) return (&s[count]); /* se h correspondencia retorna ponteiro */ else return (char *) '\0'; /* caso contrrio retorna um ponteiro nulo */ } Funoes tipo void: Funoes que nao retornam valor sem especificaao de void, farao o compilador emitir warning "Function should return a value". Nenhuma outra consequencia haver. #include<stdio.h> void imprime_vertical(char *str); void main() { char alfa[80]; printf("Digite uma string: "); gets(alfa); printf("\n\n"); imprime_vertical(alfa); } void imprime_vertical(char *str) { while(*str) printf("%c\n",*str++); }

Blocos de cdigo e variveis locais:

47

void f(void) { char ch; printf("continuar (s/n)? "); ch=getche(); if(ch=='s'){ char s[80]; /* s s criada como varivel local de if se a resposta 's' */ printf("informe o nome: "); /* Obs: s s conhecido dentro do bloco de if */ gets(s); process1(s); /* executa algum processo sobre a string s */ } } Chamando uma funao por valor do argumento: (chamada por valor) /* imprime o valor de pi e de pi^2 */ #include<stdio.h> #include<math.h> double sqr(); void main(void) { double pi=M_PI; printf("%lf %lf",pi,sqr(pi)); } double sqr(x) /* quando sqr() chamada ela copia o valor de pi em x, deixan- */ /* do pi inalterado */ double x; { x=x*x; return x; } Chamando uma funao por referncia ao argumento: (chamada por referncia) /* programa p/ demonstraao da chamada de funoes por referncia */ #include <stdio.h> void troca(); main(void) { double x,y; printf("Digite dois nmeros: "); scanf("%lf %lf",&x,&y); printf("Valores iniciais: x=%lf y=%lf\n",x,y); troca(&x,&y);/* troca opera diretamente sobre os valores de x e y,alterando-os */ printf("Valores trocados: x=%lf y=%lf\n",x,y); }

48

void troca(a,b) double *a,*b; { double temp; temp=*a; /* guarda em temp o valor contido no endereo de x */ *a = *b; /* coloca em x o valor de y */ *b = temp; /* coloca x em y */ } Chamando funoes com argumentos do tipo matriz: Quando uma matriz usada como argumento para uma funao, somente o endereo da matriz passado para a funao e nao a matriz inteira. Em outras palavras quando chamamos uma funao com um argumento matriz, um ponteiro para o primeiro elemento da matriz passado para a funao. LEMBRE: Em C um nome de qualquer "array" (conjunto) sem ndice um ponteiro para o primeiro elemento do "array". Isto significa que a declaraao do argumento no prottipo da funao deve ser a de um ponteiro de tipo compatvel. Existem trs maneiras de se declarar um argumento em uma funao que ir receber um ponteiro de uma matriz. Primeiro, o ponteiro-argumento pode ser declarado como uma matriz do mesmo tipo e tamanho daquela usada como argumento na chamada da funao: /* programa demonstraao chamada de funoes c/ matrizes */ /* imprime nmeros 0 a 9 */ #include<stdio.h> void display(); main(void) { int t[10],i; for(i=0;i<10;++i) t[i]=i; display(t); } void display(int num[10]) { int i; for(i=0;i<10;i++)printf("%d ",num[i]); } Ainda que o argumento num seja declarado uma matriz de 10 elementos, o compilador o converter automaticamente em um ponteiro tipo int. Isto necessrio porque nenhum argumento a ser passado para uma funao pode ser uma matriz com todos os seus elementos: quando uma funao chamada em main() todas as variveis locais de main() sao empurradas para a pilha,em seguida sao empurrados os argumentos da funao chamada e por ltimo o endereo de retorno para main(). Entao o programa continua a execuao na funao chamada. Como a pilha tem tamanho limitado, bvia a necessidade deste artifcio. Uma vez que somente um ponteiro para a matriz ser passado para a funao, um argumento ponteiro deve estar declarado na funao para receber os argumentos enviados da main() atravs da pilha. A segunda maneira de declarar um parmetro sem tamanho, como mostrado aqui: void imprime(int num[ ]) { int i; for(i=0; i<10; i++) printf("%d ", num[ i ]); } Aqui,num declarado como uma matriz de inteiros de tamanho desconhecido. Uma vez que a linguagem C nao prov checagem de limites em matrizes, o tamanho real da matriz irrelevante para o parmetro (mas nao para o programa, obviamente). Esse mtodo de declaraao define a varivel num como um ponteiro inteiro. para uma matriz especificando-o como uma matriz

49

O modo final como a varivel num pode ser declarada, e a forma mais comum em programas profissionais escritos na linguagem C, um ponteiro, como mostrado aqui: void imprime(int *num) { int i; for(i=0; i<10; i++) printf("d ", num[i]); } Isso permitido j que qualquer ponteiro pode ser indexado usando-se num[ i ] como se fosse numa matriz, ou *(num + i). Reconhea que todos os trs mtodos de declaraao de parmetro matriz produzirao o mesmo resultado: um ponteiro. importante entender que, quando uma matriz usada como argumento de funao, seu endereo passado para a funao. Essa uma exceao para a convenao de passagem de parmetro por chamada de valor da linguagem C. Isso significa que o cdigo na funao estar operando sobre, e potencialmente alterando, o contedo atual da matriz usada para chamar a funao. Por exemplo, considere a funao imprime_maiscula(), que imprime o seu argumento string em letra maiscula. /* Imprime uma string em letra maiscula*/ #include<stdio.h> #include<ctype.h> void imprime_maiuscula(); main(void) { char s[80]; printf("informe uma string: "); gets(s); imprime_maiuscula(s); printf("\na string original alterada: %s", s); } void imprime_maiuscula(char *string) { register int t; for(t=0; string[t]; ++t) { string[t] = toupper(string[t]); printf("%c", string[t]); } } Depois de chamada funao imprime_maiscula(),o contedo da matriz s em main() ser mudado para letras maisculas. Se voc nao quer que isso acontea, pode escrever assim: /* Imprime uma string em letra maiscula */ #include <stdio.h> #include <ctype.h> void imprime_maiuscula(char *string); main(void) { char s[80]; printf("informe uma string: "); gets(s); imprime_maiuscula(s); printf("\na string original nao modificada: %s",s); } void imprime_maiuscula(char *string) { register int t; for(t=0; string[t]; ++t) {

50

printf("%c", toupper(string[t])); } } Nessa versao, o contedo da matriz s permanece inalterado uma vez que os seus valores sao os mesmos. Um exemplo clssico de passagens de matrizes em funoes encontrado na funao de biblioteca strcat(). Ainda que strcat(), na biblioteca padrao, seja algo diferente,a funao mostrada aqui lhe dar uma idia de como ela funciona. Para evitar confusao com a funao-padrao, ela ser chamada strcat2(). /* Demonstraao da funao strcat2() */ #include <stdio.h> char *strcat2(char *s1, char *s2); main(void) { char s1[80], s2[80]; printf("Informe duas strings:\n"); gets(s1); gets(s2); strcat2(s1, s2); printf("concatenado: %s", s1); } char *strcat2(char *s1, char *s2) { char *temp; temp=s1; /* primeiro, encontra o final de s1 */ while(*s1) s1++; /* adiciona s2*/ while(*s2) { *s1=*s2; s1++; s2++; } *s1='\0'; /* acrescenta o terminador nulo */ return temp; } A funao strcat2() deve ser chamada com duas matrizes de caracteres, que, por definiao, sao apontadores para caracteres. Logo na entrada, a funao strcat2() busca o final da primeira string, onde acrescenta a segunda string. Depois, colocado o terminador nulo na primeira string. /* retornando mais de um valor de uma funao: conversao polar <--> retangular p/ nmeros complexos complexos */ #include <stdlib.h> #include <math.h> double *rp(); main() { char sign; char op; double *zo; double zi[2]; double a,b; do{ printf("R->P (r) ou P->R (p)? "); op=tolower(getche()); printf("\n"); switch(op) { case 'r': printf("Z=Re+j*Im (entre no formato Re Im): ");

51

scanf("%lf %lf",&zi[0],&zi[1]); zo = rp(zi,op); printf("Z=Mag<Ang= %lf<%lf",zo[0],zo[1]); break; case 'p': printf("Z=Mag<Ang= (entre no formato Mag Ang): "); scanf("%lf %lf",&zi[0],&zi[1]); zo = rp(zi,op); if(zo[1]<0){sign='-';zo[1]=-zo[1];} else{sign='+';} printf("Z=Re+j*Im= %lf%cj*%lf",zo[0],sign,zo[1]); break; } }while(op!='r'&&op!='p'); } double *rp(zarg,op) double *zarg; char op; { double *zout; zout=(double *)malloc(2*sizeof(double)); /* para zout alocado 2*8 bytes */ if(!zout){exit(1);} /* se nao conseguiu alocar 16 bytes do heap, cai fora */ /* Observaao: obrigatrio o uso de malloc(). Se declarssemos zout como a matriz zout[2], perderamos seus dados no retorno de *rp() porque esta funao retorna o endereo de uma varivel que nao existe mais em main(). A nica maneira de manter um ponteiro existindo no retorno da main() usando malloc() que aloca permanentemente espao no heap e nao na pilha */ if(op=='r'){ zout[0]=sqrt((*zarg)*(*zarg)+*(zarg+1)*(*(zarg+1))); /* considerando zarg */ zout[1]=(180/M_PI)*atan2(*(zarg+1),(*zarg)); /* um ponteiro */ return zout; } if(op=='p'){ zout[0]= zarg[0]*cos((M_PI/180)*zarg[1]); /* considerando zarg uma matriz */ zout[1]= zarg[0]*sin((M_PI/180)*zarg[1]); return zout; } puts("Erro!"); exit(1); /* erro de lgica: cai fora */ } Obs1) O segmento de dados de 64Kb do modelo small dividido como segue:

Area Dinmica (heap)

dados alocados por malloc() ,liberados por free() ou realocados por realloc()

Pilha (stack)

variveis locais e argumentos de funoes

Dados do programa

variveis globais e static

52

Obs2) Uma varivel static uma varivel local que mantem seu valor entre chamadas de funoes, mas s conhecida dentro da funao que declarada. Obs3) sizeof() um operador de tempo de compilaao que retorna o tamanho em bytes to tipo de dado ou objeto usado como argumento. Os Argumentos argc, argv e env para a Funao main(): Algumas vezes, muito til passar informaoes para um programa quando ele executado. O mtodo geral passar informaoes para a funao main() por meio do uso de argumentos de linha de comando. Um argumento de linha de comando a informaao que segue o nome do programa na linha de comando do sistema operacional. Existem trs argumentos internos na funao main(). Os dois primeiros, argc e argv, sao usados para receber argumentos da linha de comando. O terceiro, env, usado para acessar os parmetros do ambiente do DOS que estao ativos quando o programa comea a execuao. Estes sao os nicos argumentos que main() pode ter. O argumento argc armazena o nmero de argumentos digitados na linha de comando, e deve ser declarado como inteiro. Ele sempre ser, no mnimo, o valor um, j que o prprio nome do programa considerado como o primeiro argumento. O argumento argv um ponteiro para uma matriz de strings. Cada elemento desta matriz unidimensional aponta para um argumento digitado na linha de comando. Todos os argumentos de linha de comando sao strings - qualquer nmeo digitado dever ser convertido por atof() , atoi() ou atol(). /* mostra os argumentos digitados e converte p/ double os dados numricos */ #include <stdio.h> #include <process.h> #include <math.h> /* necessario p/ atof() */ main(argc,argv) int argc; char *argv[]; /* equivalente a argv[][] */ { int i,j; char flag='v'/* indica: numero */; char str[80]; printf("argc=%d\n",argc); for(i=0;i<argc;i++){ strcpy(str,argv[i]); /* copia argv[i] p/ str, mas argv[i] poderia ser impresso */ printf("argv[%d]=%s\n",i,str); /* diretamente em printf como uma string */ } printf("\n Convertendo para double os argumentos numricos:\n\n"); for(i=0;i<argc;i++){ flag='v'; for(j=0;j<strlen(argv[i]);j++) { if(!isdigit(argv[i][j])&&argv[i][j]!='.'&&tolower(argv[i][j])!='e'){ flag='f';break;} } if(flag=='v'){ double x; /* x s existe neste bloco */ x=atof(argv[i]); printf("argv[%d]=%lf\n",i,x); } } }

53

Basicamente argc e argv sao usados para se obter os argumentos iniciais para o programa a ser executado. Normalmente o DOS permite at 128 caracteres na linha de comando. Normalmente usa-se estes argumentos para indicar um arquivo a ser lido ou escrito, ou entao uma opao (format a:, chkdsk /f, etc...). Note que cada argumento da linha de comando deve ser separado do seguinte por um espao ou uma tabulaao (nao sao vlidos a vrgula ou ponto e vrgula). Se for necessrio passar um argumento de linha de comando contendo espaos, deve-se coloc-lo entre aspas. O argumento env declarado da mesma maneira que argv. Ele um ponteiro para a matriz que contem as definioes do ambiente do DOS (variveis ambientais): /* mostra o ambiente do DOS */ #include<stdio.h> main(argc,argv,env) int argc; char *argv[]; char *env[]; { int i; for(i=0;env[i];i++)printf("%s\n",env[i]); } Retornando valores de main() para o DOS: (usado no comando IF ERRORLEVEL do DOS: zero->sucesso /* programa p/ executar comandos do DOS sequencialmente */ #include<stdio.h> #include<stdlib.h> main(int argc, char *argv[]) { int i; for(i=1; i<argc; i++) { if(system(argv[i])) { printf("%s falha\n", argv[i]); return -1 } } Funoes recursivas: /* /* /* /* determina o fatorial de um nmero inteiro RECURSIVAMENTE */ Vantagem: cdigo menor se comparado com a versao nao recursiva */ Desvantagem: roda ligeiramente mais lenta (operaao com pilha) */ Obs) colocar watches em n n-1 e resposta para acompanhamento */ nao_zero->erro)

#include<stdlib.h> #include<stdio.h> unsigned long int factr(); main() { unsigned long int n,f;

54

printf("Digite um nmero: "); scanf("%ld",&n); f=factr(n); printf("O fatorial de %ld %ld\n", n,f); } unsigned long int factr(n) unsigned long int n; { unsigned long int resposta; if(n==1||n==0)return(1); resposta = factr(n-1)*n; printf("%ld\n",resposta); /* opcional para visualizaao */ return(resposta); } /* determina o fatorial de um nmero inteiro NAO RECURSIVAMENTE */ #include<stdlib.h> #include<stdio.h> unsigned long int factr(); main() { unsigned long int n,f; printf("Digite um nmero: "); scanf("%ld",&n); f=factr(n); printf("O fatorial de %ld %ld\n", n,f); } unsigned long int factr(n) unsigned long int n; { unsigned long int resposta , t; for(t=1; t<n; t++) resposta = resposta*t; return(resposta); } Quando uma funao chama a si prpria, as novas variveis locais e os argumentos sao alocados na pilha, e o cdigo da funao executado com esses novos valores a partir do incio. Uma chamada recursiva nao faz uma nova cpia da funao. Somente os argumentos e as variveis sao novos. Quando cada chamada recursiva retorna, as antigas variveis locais e os parmetros sao removidos da pilha e a execuao recomea no ponto de chamada da funao dentro da funao dentro da funao. Tendo em vista que o local para os argumentos de funoes e para as variveis locais a pilha e que a cada nova chamada criado uma cpia destas variveis na pilha, possvel ocorrer overflow da pilha (stack overflow) e o programa trancar.

/* programa recursivo para geraao de som em tonalidade crescente at 1KHz. */ #include <stdio.h> #include <dos.h>

55

main() { int dur,salto; printf("Digite a duraao de cada ton em milissegundos: "); scanf("%d",&dur); printf("Digite o salto entre tons em Hz: "); scanf("%d",&salto); ton(1000,dur,salto); return 0; } ton(freq,duracao,salto) int freq,duracao,salto; { int i; if(freq>0) ton(freq-salto,duracao,salto); sound(freq); /* liga o som */ delay(duracao); /* aguarda duracao milissegundos */ nosound(); } 7) ENTRADA E SADA DO CONSOLE: Lendo e escrevendo caracteres: getch(), getche(), getchar() e putchar(): As funoes mais simples de E/S do console (teclado e vdeo) sao getche(), que l um caractere do teclado, e putchar(), que imprime um caractere no ecr. A funao getche() aguarda at que uma tecla seja pressionada e entao retorna o seu valor. A tecla pressionada ecoa na tela automaticamente. A funao putchar() escreve o seu argumento caractere no ecr na posiao corrente do cursor. Os prottipos sao: int getche(void); /* header: conio.h */ int putchar(int c); /* header: stdio.h */ Nao nos perturbemos com o fato de getche() retornar um inteiro; o byte de mais baixa ordem contem o caractere e quando atribue-se um inteiro a uma varivel char somente transferido o byte mais baixo. Tambm a funao putchar() embora seja declarada com um argumento inteiro, somente o byte mais baixo enviado tela. Uma versao mais til de getche() getch() que nao ecoa no ecr o caractere digitado. O prottipo idntico ao de getche().

/* inverte minsculas<-->maisculas */ #include<stdio.h> #include<conio.h> #include<ctype.h> main() { char ch; printf("Insira caracteres e digite um ponto para parar.\n"); do{ ch=getch(); if(islower(ch))putchar(toupper(ch)); else putchar(tolower(ch)); } while(ch!='.'); return 0; }

56

Uma versao antiga de getche(),ainda dirigida para ambientes UNIX, getchar() ,mas sofre de problemas com seu buffer intermedirio em ambientes DOS iterativos.No entanto,permite o retorno do caracter 'CR' (Carriage Return = 'Enter'). Lendo e escrevendo strings: gets() e puts() A funao gets() l uma string de caracteres entrada pelo teclado e coloca-a no endereo apontado pelo seu argumento ponteiro para caractere. Digita-se caracteres at que um 'Enter'(CR = Carriage Return) seja pressionado. O 'CR' nao toma parte da string; em vez disto um terminador nulo '\0' colocado no fim da sequencia de caracteres e a funao gets() retorna: Prottipo: char *gets(char *s); Header: stdio.h onde s uma matriz de caracteres que recebe os caracteres digitados. Observe que gets() retorna um ponteiro para s. A funao puts() escreve o seu argumento string para o ecr seguido de "\n". Prottipo: int puts(char *s); Header: stdio.h puts() reconhece os mesmos cdigos de barra invertida de printf(), como o \t de tabulaao. Uma chamada para puts() gera um cdigo executvel bem menor e mais rpido do que uma chamada a printf(). Entrada e sada formatada do console: scanf() e printf() O termo 'formatada' refere-se ao fato de que essas funoes podem ler e escrever dados em vrios formatos que estao sob seu controle. printf(): O prottipo de printf() int printf(char *string_de controle, lista_de_argumentos); com header em stdio.h. A string de controle consiste em dois tipos de tens. O primeiro tipo o conjunto de caracteres que o usurio deseja que sejam impressos no ecr. O segundo tipo o conjunto de especificadores de formato, que definem a maneira como os argumentos sao exibidos. Um especificador de formato comea com um sinal de porcentagem (%) e seguido pelo cdigo de formato. Deve haver o mesmo nmero de argumentos quantos forem os especificadores de formato, necessitando haver uma correspondncia de tipo entre especificador de formato e argumento na ordem de ocorrncia respectiva de cada um na string de controle e lista de argumentos: printf("Ol %c %d %s",'c',10,"todos!"); mostra: Ol c 10 todos!

Especificadores de formato de printf(): Especificador: %c %d %i %e %E %f %g %G %o %s %u %x Formato resultante no ecr: Um nico caractere Decimal inteiro Decimal inteiro Notaao cientfica ('e' minsculo: 6.22e17 ) Notaao cientfica ('E' maisculo: 6.22E17 ) Decimal ponto flutuante Usa %e ou %f - o que ocupar menos espao no ecr Igual a %g s que usa 'E' maisculo no caso de acionar %e Octal inteiro String de caracteres Decimal inteiro sem sinal Hexadecimal inteiro (usando letras minsculas)

57

%X %% %p %Fp %n

Hexadecimal inteiro (usando letras maisculas) Imprime o sinal % O endereo hexadecimal de 16 bits (deslocamento) para o qual aponta o argumento ponteiro associado dentro de um segmento de 64Kb (ponteiro near ). O endereo hexadecimal de 20 bits para o qual aponta o argumento-ponteiro associado no formato segmento:deslocamento (ponteiro far) . O argumento associado um ponteiro para um inteiro onde colocado o nmero de caracteres escritos at entao (Ver exemplo abaixo)

/* Imprime o numero de caracteres impresso por printf() */ #include<stdio.h> #include<conio.h> #include<ctype.h> main() { char str[80]; int i; gets(str); printf("%s %n",str,&i); /* imprime str e 2 espaos */ printf("%d",i); } Os especificadores de formato podem ter modificadores que especificam o tamanho do campo, o nmero de casas decimais e um indicador de justificaao a direita. Modificadores: Um inteiro colocado entre o sinal % e o caracterizador de formato atua como um especificador de largura mnima do campo. Este especificador preenche a sada com brancos ou zeros para assegurar que o campo esteja com pelo menos um certo comprimento mnimo. Se a string ou nmero maior que o mnimo, ser impressa por completo,mesmo se o mnimo for ultrapassado. O preenchimento padrao feito com espaos. Se houver intenao de preencher com zeros, colocar um 0 antes do especificador de comprimento do campo. Por exemplo, %05d preenche um nmero de menos de cinco dgitos com zeros, de modo que o tamanho total seja cinco. Para especificar o nmero de casas decimais impressas por um nmero em ponto flutuante, coloca-se um ponto decimal depois do especificador de tamanho de campo,seguido pelo nmero de casas decimais que se deseja mostrar.Por exemplo, %10.4f exibe um nmero de , no mnimo, dez caracteres de comprimento com quatro casas decimais. Quando isto aplicado a strings ou inteiros, o nmero seguinte ao ponto especifica o comprimento mximo do campo. Por exemplo, %5.7s exibe uma string de no mnimo cinco caracteres e no mximo sete. Se a string a ser impressa exceder sete caracteres, ser truncada. Por definiao, toda sada justificada direita: se a largura de campo maior que o dado impresso, o dado ser colocado na extremidade direita do campo. Pode-se forar a informaao a ser justificada esquerda colocando-se um sinal de menos (-) diretamente depois do %. Por exemplo, %-10.2f justificar a esquerda um nmero em ponto flutuante, com duas casas decimais, em um campo de 10 caracteres. Existem dois modificadores dos especificadores de formato que permitem printf() exibir inteiros short e long. Esses modificadores podem ser aplicados aos especificadores de tipo d,i,o,u,x. O modificador l indica que um dado tipo long ser impresso no ecr: %ld long int %lu unsigned long int O modificador h indica que um dado tipo short int ser impresso no ecr: %hu unsigned short int %hd short int %ho unsigned short int octal O modificador l tambm pode alterar os especificadores e,f,g para indicar que um double ser impresso:

58

%lf double em ponto flutuante %Lf long double em ponto flutuante %LE long double em notaao cientfica c/ 'E' maisculo /* Roda exemplos de especificadores de formato e campo de printf() */ #include<stdio.h> main() { puts("012345678901234567890"); printf("%-7.2f\n",123.234); printf("%7.2f\n",123.234); printf("%-5.2f\n",123.234); printf("%5.2f\n",3.324); printf("%10s\n","Al"); printf("%-10s\n","Al"); printf("%5.7s\n","123456789"); } scanf(): O prottipo de scanf() int scanf(char *string_de_de controle,lista_de_argumentos) com header em stdio.h. A string de controle consiste de trs conjunto de caracteres: - especificadores de formato - caracteres de espao em branco - caracteres de espao nao branco Os especificadores de formato de entrada sao precedidos pelo sinal % e dizem a scanf() qual tipo de dado deve ser lido em seguida. A lista de argumentos contm uma lista de apontadores para as variveisargumentos correspondentes em tipo, nmero e ordem de ocorrncia aos especificadores de formato na string de controle.

Especificadores de formato de scanf(): Especificador %c %d %i %e ou %E %f %g ou %G %o %s %u %x %p %Fp %n Dado a ser lido: Um nico caractere Decimal inteiro Decimal inteiro Notaao cientfica Decimal ponto flutuante L %e ou %f Octal inteiro String de caracteres Decimal inteiro sem sinal Hexadecimal inteiro L o valor de um ponteiro L o valor de um ponteiro far O argumento associado um ponteiro para um inteiro onde colocado o nmero de caracteres lidos at entao. Ver printf() Idntico ao caso sem o modificador l s que l double em vez de float Idntico ao caso sem o modificador l s que l long Idntico ao caso sem o modificador h s que l short

%le ou %lg %l seguido de d,i,o,u,x %h seguido de d,i,o,u x

59

Caracteres de espao em branco: Um caratere de espao em branco na string de controle faz com que a funao scanf() salte um ou mais caracteres em branco no buffer de entrada. Um caractere de espao em branco pode ser: - um espao ou - uma tabulaao ou - um newline ('Enter') Essencialmente, um caractere de espao em branco na string de controle far com que a scanf() leia, mas nao armazene, qualquer nmero de espaos em branco (incluindo zero, isto , nenhum) de caracteres de espao em branco at o primeiro caractere que nao seja um espao em branco. Por exemplo,"%d,%d" faz com que scanf() leia um inteiro,entao leia e descarte uma vrgula e, finalmente,leia um outro inteiro. Se o caractere especificado nao encontrado scanf() termina prematuramente e nao l o segundo inteiro. O programa a seguir usa o fato de que scanf() retorna o nmero de argumentos lidos com sucesso para demonstrar o efeito dos caracteres de espao nao branco na entrada de dados: /* testa caracteres de espao nao branco em scanf() */ /* Digitar dois inteiros separados por # e depois separados por qualquer */ /* outro cacarctere */ #include<stdio.h> main() { int i,j,c; printf("Digite dois nmeros inteiros separados por #: "); c=scanf("%d#%d",&i,&j);/* c recebe o nmero de argumentos lidos por scanf() */ if(c==2)printf("Voc digitou: %d e %d\n",i,j); else printf("Insucesso. Lido somente %d campo por scanf()!",c); } Um * colocado depois do % e antes do cdigo de formato ler suprimir sua atribuiao. Assim, scanf("%d%*c%d",&x,&y); dando-se a entrada 10!20, colocar o valor 10 em x descartando o caractere '!' e dar a y o valor 20. Os comandos de formato podem especificar um campo modificador de comprimento mximo. Esse modificador um nmero inteiro colocado entre o sinal % e o cdigo de comando de formato, que limita o nmero de caracteres lidos para qualquer campo. Por exemplo, para ler nao mais do que 20 caracteres em str, escreve-se: scanf("%20s",str); Se o buffer de entrada contiver mais do que 20 caracteres, scanf() termina a leitura e s atribui a str os 20 primeiros caracteres deixando os restantes no buffer. Uma chamada subsequente a scanf() comear a leitura do buffer de entrada onde ela terminou. Por exemplo, se digitarmos ABCDEFGHIJKLMNOPQRSTUVWXYZ em resposta chamada de scanf() deste exemplo, somente os 20 primeiros caracteres, at 'T', serao colocados em str. Na prxima chamada scanf() serao colocadas em str a string "UVWXYZ": /* demonstra modificador de comprimento mximo de campo de scanf() */ #include<stdio.h> main() { char str[80]; um dado de um tipo especificado, mas

60

puts("Digite abcdefghijklmnopqrstuvwxyz:"); scanf("%20s",str); puts(str); scanf("%20s",str); /* o programa nao parar nesta chamada a scanf() porque */ puts(str); /* ainda haver caracteres nao transferidos no buffer de entrada */ } Ainda que espaos, tabulaoes e newlines ('Enter') sejam usados como separadores de campos, quando da leitura de um nico caractere, esses ltimos sao lidos como qualquer outro caractere. Por exemplo, entrando no teclado com "x y" , scanf("%c%c%c",&a,&b,&c); far: a='x' , b=' ' e c='y' . Cuidado: se tivermos quaisquer outros caracteres na string de controle no exemplo anterior - incluindo espaos,tabulaoes e newlines - eles serao usados para correspondncia-e-descarte de caracteres de entrada.Qualquer caracter que corresponder descartado e se nao corresponder scanf() termina: /* digite xtyz e depois x yz */ #include<stdio.h> main() { char a,b,c; puts("Digite 'xtyz' na primeira tentativa e na segunda 'x yz'"); scanf("%ct%c%c",&a,&b,&c); printf("a=%c b=%c c=%c",a,b,c); } Outro exemplo: No programa a seguir, scanf() nao retornar at que seja digitado um caractere depois de digitar um caractere de espao em branco. Isso ocorre em razao de espao aps 0 %s, que instruiu a scanf() a ler e descartar espaos, tabulaoes e newlines: #include<stdio.h> main() { char str[80]; puts("Digite uma string um espao e um caractere. Depois nao obedea esta regra de entrada."); scanf("%s ",str); puts(str); } A funao scanf() inclui tambm o recurso chamado 'scanset'. Scanset define um conjunto de caracteres que serao correspondidos por scanf(). A funao scanf() continuar a ler caracteres enquanto eles estiverem no scanset. Assim que um caractere entrado nao corresponder a nenhum caractere do scanset, segue para o prximo especificador de formato (se existir). Um scanset definido colocando a lista de caracteres que se quer que seja examinada entre colchetes. Um sinal % deve anteceder o colchete inicial. Por exemplo, %[01234567] diz scanf() para ler somente os dgitos de 0 a 7. A varivel-argumento correspondente a scanset deve ser um ponteiro para uma matriz de caracteres (isto , uma string). /* analisa os dados de entrada e converte os nmeros para float */ #include<stdio.h> #include<math.h> main() { char s1[80]; char s2[80]; float x;

61

puts("Digite um nmero seguido de caracteres nao numricos: "); scanf("%[123456789.eE]%s",s1,s2); printf("Voce digitou o nmero %f\n",atof(s1)); printf("E a sequncia nao numrica foi %s.",s2); } Podemos especificar um intervalo dentro de um scanset usando um hfen. Por exemplo, o cdigo abaixo diz scanf() para aceitar os caracteres de A a Z: %[A-Z] Podemos tambm especificar mais de um intervalo dentro de um scanset. O programa abaixo por exemplo l dgitos e letras minsculas. Ele demonstra tambm o uso de especificador de campo mximo (78 no caso) para scanset: #include <stdio.h> main() { char str[80]; puts("Informe dgitos e letras minsculas:"); scanf("%78[a-z0-9]",str); puts(str); } Scanset permite tambm a especificaao da exclusao de conjuntos de caracteres mediante o especificador ^ colocado como primeiro caractere do scanset: #include <stdio.h> main() { char str[80]; puts("Informe dgitos e letras minsculas excluindo a,b,c:"); scanf("%[^a-b]",str); puts(str); } 8) ENTRADA E SADA DE DISCO: O ponteiro de arquivo: A linha comum que une o sistema de E/S de disco do Turbo C o ponteiro de arquivo. Um ponteiro de arquivo um ponteiro para uma rea na memria(buffer) onde estao contidos vrios dados sobre o arquivo a ler ou escrever, tais como o nome do arquivo, estado e posiao corrente. O buffer apontado pelo ponteiro de arquivo a rea intermediria entre o arquivo no disco e o programa. Este buffer intermedirio entre arquivo e programa chamado 'fluxo', e no jargao dos programadores comum falar em funoes que operam fluxos em vez de arquivos. Isto se deve ao fato de que um fluxo uma entidade lgica genrica, que pode estar associada a uma unidade de fita magntica, um disco,uma porta serial, etc. Adotaremos esta nomenclatura aqui. Um ponteiro de arquivo uma varivel-ponteiro do tipo FILE que definida em stdio.h. Para ler ou escrever em um arquivo de disco, o programa deve declarar uma (ou mais de uma se formos trabalhar com mais de um arquivo simultaneamente) varivel ponteiro de arquivo. Para obter uma varivel ponteiro de arquivo, usa-se uma declaraao semelhante a esta ao declararmos as demais variveis do programa: FILE *fp; onde fp o nome que escolhemos para a varivel (podia ser qualquer outro). -----------------------------------------------------------------------------As Funoes Mais Comuns do Sistema de Arquivo -----------------------------------------------------------------------------Funao Operaao fopen() Abre um fluxo

62

fclose() Fecha um fluxo putc() Escreve um caractere para um fluxo getc() L um carcatere para um fluxo fseek() Procura por um byte especificado no fluxo fprintf() para um fluxo aquilo que printf() para o console fscanf() para um fluxo aquilo que scanf() para o console feof() Retorna verdadeiro se o fim do arquivo encontrado ferror() Retorna verdadeiro se ocorreu um erro fread() L um bloco de dados de um fluxo fwrite() Escreve um bloco de dados para um fluxo rewind() Reposiciona o localizador de posiao de arquivo no comeo do arquivo remove() Apaga um arquivo -----------------------------------------------------------------------------Abrindo um Arquivo: A funao fopen() serve a dois propsitos. Primeiro, ela abre um fluxo para uso e liga um arquivo com ele. Segundo, retorna o ponteiro de arquivo associado quele arquivo. Mais freqentemente, e para o resto desta discussao, o arquivo um arquivo em disco. A funao fopen() tem este prottipo: FILE *fopen(char *nome_de_arquivo, char *modo); onde modo uma string contendo o estado desejado para abertura. O nome do arquivo deve ser uma string de caracteres que compreende um nome de arquivo vlido para o sistema operacional e onde possa ser includa uma especificaao de caminho (PATH). Como determinado, a funao fopen() retorna um ponteiro de arquivo que nao deve ter o valor alterado pelo seu programa. Se um erro ocorre quando se est abrindo um arquivo, fopen() retorna um nulo. Como a Tabela abaixo mostra, um arquivo pode ser aberto ou em modo texto ou em modo binrio. No modo texto, as seqncias de retorno de carro e alimentaao de formulrios sao transformadas em seqncias de novas linhas na entrada. Na sada, ocorre o inverso: novas linhas sao transformadas em retorno de carro e alimentaao de formulrio. Tais transformaoes nao acontecem em um arquivo binrio. -----------------------------------------------------------------------------Os valores legais para modo. -----------------------------------------------------------------------------Modo Significado "r" Abre um arquivo para leitura "w" Cria um arqivo para escrita "a" Acrescenta dados para um arquivo existente "rb" Abre um arquivo binrio para leitura "wb" Cria um arquivo binrio para escrita "ab" Acrescenta dados a um arquivo binrio existente "r+" Abre um arquivo para leitura/escrita "w+" Cria um arquivo para leitura/escrita "a+" Acrescenta dados ou cria um arquivo para leitura/escrita "r+b" Abre um arquivo binrio para leitura/escrita "w+b" Cria um arquivo binrio para leitura/escrita "a+b" Acrescenta ou cria um arquivo binrio para leitura/escrita "rt" Abre um arquivo texto para leitura "wt" Cria um arquivo texto para escrita "at" Acrescenta dados a um arquivo texto "r+t" Abre um arquivo-texto para leitura/escrita "w+t" Cria um arquivo texto para leitura/escrita "a+t" Acrescenta dados ou cria um arquivo texto para leitura/escrita ----------------------------------------------------------------------------Se voc deseja abrir um arquivo para escrita com o nome test voc deve escrever FILE *ofp; ofp = fopen ("test", "w"); Entretanto, voc ver freqentemente escrito assim: FILE *out;

63

if((out=fopen("test","w"))==NULL){ puts("nao posso abrir o arquivo\n"); exit(1); } A macro NULL definida em STDIO.H. Esse mtodo detecta qualquer erro na abertura do arquivo como um arquivo protegido contra escrita ou disco cheio, antes de tentar escrever nele. Um nulo usado porque no ponteiro do arquivo nunca haver aquele valor. Tambm introduzido por esse fragmento est outra funao de biblioteca: exit(). Uma chamada funao exit() faz com que haja uma terminaao imediata do programa, nao importando de onde a funao exit() chamada. Ela tem este prottipo (encontrado em STDLIB.H): void exit(int val); O valor de val retornado para o sistema operacional. Por convenao, um valor de retorno 0 significa trmino com sucesso para o DOS. Qualquer outro valor indica que o programa terminou por causa de algum problema (retornado para a varivel ERRORLEVEL em arquivos batch do DOS). Se voc usar a funao fopen() para abrir um arquivo, qualquer arquivo preexistente com o mesmo nome ser apagado e o novo arquivo iniciado. Se nao existirem arquivos com o nome, entao ser criado um. Se voc quiser acrescentar dados ao final do arquivo, deve usar o modo "a". Para se abrir um arquivo para operaoes de leitura necessrio que o arquivo j exista. Se ele nao existir, ser retornado um erro. Finalmente, se o arquivo aberto para escrita/leitura, as operaoes feitas nao apagarao o arquivo, se ele existir; entretanto, se nao existir, ser criado. Escrevendo um caractere: A funao putc() usada para escrever caracteres para um fluxo que foi previamente aberto para escrita pela funao fopen(). A funao declarada como int putc(int ch, FILE *fp); onde fp o ponteiro de arquivo retornado pela funao fopen() e ch, o caractere a ser escrito. Por razoes histricas, ch chamado formalmente de int, mas somente o caractere de mais baixa ordem usado. Se uma operaao com a funao putc() for satisfatria, ela retornar o caractere escrito. Em caso de falha, um EOF retronado. EOF uma macrodefiniao em STDIO.H que significa fim do arquivo. Lendo um caractere: A funao getc() usada para ler caracteres do fluxo aberto em modo de leitura pela funao fopen(). A funao declarada como int getc(FILE *fp) onde fp um ponteiro de arquivo do tipo FILE retornado pela funao fopen(). Por razoes histricas, a funao getc() retorna um inteiro, porm o byte de mais alta ordem zero. A funao getc() retornar uma marca EOF quando o fim do arquivo tiver sido encontrado ou um erro tiver ocorrido. Portanto, para ler um arquivo-texto at at que a marca de fim de arquivo seja mostrada, voc poder usar o seguinte cdigo: ch=getc(fp); while(ch!=EOF) { ch=getc(fp); } Usando a funao feof(): O sistema de arquivo do Turbo C tambm pode operar com dados binrios. Quando um arquivo aberto para entrada binria, possvel encontrar um valor inteiro igual marca de EOF. Isso pode fazer com que, na rotina anterior, seja indicada uma condiao de fim de arquivo, ainda que o fim de arquivo fsico nao tenha sido encontrado. Para resolver esse problema, foi incluida a funao feof() que usada para determinar o final de um arquivo quando da leitura de dados binrios. A funao feof() tem este prottipo:

64

int feof(FILE *fp); O seu prottipo est em STDIO.H. Ela retorna verdadeiro se o final do arquivo tiver sido encontrado; caso contrrio, retornado um zero. Portanto, a seguinte rotina l um arquivo binrio at que o final do arquivo seja encontrado. while(!feop(fp))ch=getc(fp); Obviamente, o mesmo mtodo pode ser aplicado tanto a arquivos textos como a arquivos binrios. Fechando um arquivo: A funao fclose() usada para fechar um fluxo que foi aberto por uma chamada funao fopen(). Ela escreve quaisquer dados restantes na rea intermediria (fluxo) no arquivo e faz um fechamento formal em nvel de sistema operacional do arquivo. Uma falha no fechamento de um arquivo leva a todo tipo de problemas, incluindo-se perda de dados, arquivos destrudos e a possibilidade de erros intermitentes no seu programa. Uma chamada funao fclose() tambm libera os blocos de controle de arquivo (FCB) associados ao fluxo e deixa-os disponveis para reutilizaao. Como voc provavelmente sabe, o sistema operacional limita o nmero de arquivos abertos que voc pode ter em um dado momento, de modo que pode ser necessrio fechar alguns arquivos antes de abrir outros.( Comando FILES= no autoexec.bat). A funao fclose() declarada como: int fclose(FILE*fp); onde fp o ponteiro do arquivo retornado pela chamada funao fopen(). Um valor de retorno igual a zero significa uma operaao de fechamento com sucesso; qualquer outro valor indica um erro. Voc pode usar a funao padrao ferror() (discutida a seguir) para determinar e reportar quaisquer problemas. Geralmente, o nico momento em que a funao fclose() falhar quando um disquete tiver sido prematuramente removido do drive ou nao houver mais espao no disco. ferror() e rewind(): A funao ferror() usada para determinar se uma operaao em um arquivo produziu um erro. Se um arquivo aberto em modo texto e ocorre um erro na leitura ou na escrita, retornado EOF. Voc usa a funao ferror() para determinar o que aconteceu. A funao ferror() tem o seguinte prottipo: int ferror(FILE*fp); onde fp um ponteiro vlido para um arquivo. ferror() retorna verdadeiro se um erro ocorreu durante a ltima operaao com o arquivo e falso, caso contrrio.Uma vez que cada operaao em arquivo determina uma condiao de erro, a funao ferror() deve ser chamada imediatamente aps cada operaao com o arquivo; caso contrrio, um erro pode ser perdido. O prottipo de funao ferror() est no arquivo STDIO.H. A funao rewind() recolocar o localizador de posiao do arquivo no incio do arquivo especificado como seu argumento. O seu prottipo : void rewind(FILE*fp); onde fp um ponteiro de arquivo. O prottipo para a funao rewind() est no arquivo STDIO.H. Usando fopen(), getc(), putc() e fclose(): /* Grava em disco dados digitados no teclado at teclar $ */ #include <stdio.h> #include <stdlib.h> main(int argc, char *argv[]) { FILE *fp; char ch; if(argc!=3) { puts("Voc se esqueceu de informar o nome do arquivo e o modo!"); exit(1);

65

} if((fp=fopen(argv[1], argv[2]))==NULL){ /* experimentar "wb"/"wt" e testar newline */ puts("O arquivo nao pode ser aberto!"); exit(1); } do { /* getche() nao funcionaria aqui porque getchar() */ ch=getchar(); /* pega o caracteres do teclado vai armazenando no buffer at o newline. */ if(EOF==putc(ch, fp)){ puts("Erro ao escrever no arquivo!"); break; } } while (ch!='$'); fclose(fp); return 0; /* cdigo de retorno de final OK p/ o DOS. OPCIONAL */ } /*L arquivos e exibe-os no ecr.*/ #include <stdio.h> #include <stdlib.h> main(int argc, char *argv[]) { FILE *fp; char ch; if(argc!=3) { puts("Voc se esqueceu de informar o nome do arquivo e o modo!"); exit(1); } if((fp=fopen(argv[1], argv[2]))==NULL){ /* experimentar "rb"/"rt" e testar newline */ puts("O arquivo nao pode ser aberto!"); exit(1); } ch= getc(fp); /* l um caractere */ while(ch!=EOF){ putchar(ch); /* imprime no ecr */ ch=getc(fp); } return 0; } getw() e putw(): Funcionam exatamente como putc() e getc() s que em vez de ler ou escrever um nico caractere, lem ou escrevem um inteiro. Prottipos: int putw(int i, FILE *fp); int getw(FILE *fp); Header: stdio.h Exemplo: putw(100,saida); escreve o inteiro 100 no arquivo apontado por saida. fgets() e fputs(): Funcionam como gets() e puts() s que lem e escrevem em fluxos. Prottipos: char *fputs(char *str, FILE *fp);

66

char *fgets(char *str, int tamanho, FILE *fp); Obs) fgets() l uma string de um fluxo especificado at que um newline seja lido OU tamanho-1 caracteres tenham sido lidos. Se um newline lido, ele far parte da string (diferente de gets()). A string resultante termina com um nulo. Header: stdio.h fread() e fwrite(): Lem e escrevem blocos de dados em fluxos. Prottipos: unsigned fread(void *buffer, int num_bytes, int count, FILE *fp); unsigned fwrite(void *buffer, int num_bytes, int count, FILE *fp); Header: stdio.h No caso de fread(), buffer um ponteiro para uma rea da memria que receber os dados lidos do arquivo. No caso de fwrite(), buffer um ponteiro para uma rea da memria onde se encontram os dados a serem escritos no arquivo. Buffer usualmente aponta para uma matriz ou estrutura(a ser visto adiante). O nmero de bytes a ser lido ou escrito especificado por num_bytes. O argumento count determina quantos tens (cada um tendo num_bytes de tamanho) serao lidos ou escritos. O argumento fp um ponteiro para um arquivo de um fluxo previamente aberto por fopen(). A funao fread() retorna o nmero de tens lidos, que pode ser menor que count caso o final de arquivo (EOF) seja encontrado ou ocorra um erro. A funao fwrite() retorna o nmero de tens escritos, que ser igual a count exceto na ocorrncia de um erro de escrita. Quando o arquivo for aberto para dados binrios, fread() e fwrite() podem ler e escrever qualquer tipo de informaao. O programa a seguir escreve um float em um arquivo de disco chamado teste.dat: #include<stdio.h> #include<stdlib.h> #include<math.h> main() { FILE *fp; float f=M_PI; if((fp=fopen("teste.dat","wb"))==NULL){ puts("Nao posso abrir arquivo!"); exit(1); } if(fwrite(&f, sizeof(float), 1, fp)!=1) puts("Erro escrevendo no arquivo!"); fclose(fp); return 0; } Como o programa anterior ilustra, a rea intermediria de armazenamento pode ser (e frequentemente ) simplesmente uma varivel. Uma das aplicaoes mais teis de fread() e fwrite() o armazenamento e leitura rpidos de matrizes e estruturas (estrutura uma entidade lgica a ser vista adiante) em disco: /* escreve uma matriz em disco */

67

#include<stdio.h> #include<stdlib.h> main() { FILE *fp; float exemplo[10][10]; int i,j; if((fp=fopen("exemplo.dat","wb"))==NULL){ puts("Nao posso abrir arquivo!"); exit(1); } for(i=0; i<10; i++){ for(j=0; j<10; j++){ exemplo[i][j] = (float) i+j; /* lei de formaao dos elementos da matriz */ } } /* o cdigo a seguir grava a matriz inteira em um nico passo: */ if(fwrite(exemplo, sizeof(exemplo), 1, fp)!=1) { puts("Erro ao escrever arquivo!");exit(1);} fclose(fp); return 0; } /* l uma matriz em disco */ #include<stdio.h> #include<stdlib.h> main() { FILE *fp; float exemplo[10][10]; int i,j; if((fp=fopen("exemplo.dat","rb"))==NULL){ puts("Nao posso abrir arquivo!"); exit(1); } /* o cdigo a seguir l a matriz inteira em um nico passo: */ if(fread(exemplo, sizeof(exemplo), 1, fp)!=1) { puts("Erro ao ler arquivo!");exit(1);} for(i=0; i<10; i++){ for(j=0; j<10; j++) printf("%3.1f ", exemplo[i][j]); printf("\n"); } fclose(fp); return 0; } Acesso randmico a arquivos: fseek()

68

A funao fseek() indica o localizador de posiao do arquivo. Prottipo: int fseek(FILE *fp, long numbytes, int origem); Header: stdio.h - fp um ponteiro para o arquivo retornado por uma chamada fopen(). - numbytes (long int) o nmero de bytes a partir da origem at a posiao corrente. - origem uma das seguintes macros definidas em stdio.h: Origem comeo do arquivo posiao corrente fim do arquivo Nome da Macro SEEK_SET SEEK_CUR SEEK_END 0 1 2 Valor numrico

Portanto, para procurar numbytes do comeo do arquivo, origem deve ser SEEK_SET. Para procurar da posiao corrente em diante, origem SEEK_CUR. Para procurar numbytes do final do arquivo de trs para diante, origem SEEK_END. O seguinte cdigo l o 235 byte do arquivo chamado test: FILE *fp; char ch; if((fp=fopen("teste","rb"))==NULL){ puts("Nao posso abrir arquivo!"); exit(1); } fseek(fp,234,SEEK_SET); ch=getc(fp); /* l um caractere na posiao 235 */ A funao fseek() retorna zero se houve sucesso ou um valor nao-zero se houve falha no posicionamento do localizador de posiao do arquivo. /* utilitrio de vizualizaao de disco usando fseek() */ /* visualiza setores de 128 de bytes do disco e apresenta em ASCII e hexadecimal */ /* digite -1 para deixar o programa */ #include <stdio.h> #include <ctype.h> /* isprint() necessita */ #include <stdlib.h> #define TAMANHO 128 /* TAMANHO = 128 */ char buf[TAMANHO]; /* global */ void display(); main(int argc, char *argv[]) { FILE *fp; long int setor, numlido; if(argc!=2){puts("Uso: tmp nome_do_arquivo");exit(1);} /* esqueceu de digitar */ /* o nome de arquivo */ if((fp=fopen(argv[1],"rb"))==NULL){ puts("Nao posso abrir arquivo!"); exit(1); } for(;;) { printf("Informe o setor (-1 p/ terminar): ");

69

scanf("%ld",&setor); if(setor<0) break; if(fseek(fp, setor*TAMANHO , SEEK_SET)){puts("Erro de busca!");exit(1);} if((numlido=fread(buf,1,TAMANHO,fp)) != TAMANHO){puts("EOF encontrado!");} display(numlido); } return 0; } void display(long int numlido) { long int i,j; for(i=0; i<=numlido/16 ; i++) {/* controla a indexaao de linhas: 0 a 8 (128/16=8) */ for(j=0; j<16; j++) printf("%2X ", buf[i*16+j]); /* imprime 16 col. em hexa */ printf(" "); /* separa mostrador hexa do ascii */ for(j=0; j<16; j++) { /* imprime 16 col. em ascii: (s imprimveis) */ if(isprint(buf[i*16+j]))printf("%c",buf[i*16+j]); else printf("."); } printf("\n"); } } Os Fluxos-Padroes: Toda vez que um programa comea a execuao, sao abertos 5 fluxos padroes: stdin (aponta p/ o teclado se nao redirecionado pelo DOS. Ex: MORE < FILE.TXT) stdout (aponta p/ o ecr se nao redirecionado pelo DOS. Ex : DIR > PRN) stderr (recebe mensagens de erro - aponta para o ecr) stdprn (aponta p/ a impressora) stdaux (aponta p/ a porta serial) Para entender o funcionamento destes fluxos, note que putchar() definida em stdio.h como: #define putchar(c) putc(c,stdout) e a funao getchar() definida como #define getchar() getc(stdin) Ou seja, estes fluxos permitem serem lidos e escritos como se fossem fluxos de arquivos. Toda entrada de dado de um program feita por stdin e toda sada por stdout: /* localiza uma palavra especificada pela linha de comando em um arquivo redirecionado para stdin, mostra a linha e seu nmero */ /* rodar no prompt do DOS o seguinte comando: tmp argv < tmp.c */ #include<string.h> /* strstr() necessita */ #include<stdio.h> main(int argc, char *argv[]) { char string[128]; int line=0; while(fgets(string, sizeof(string),stdin))

70

{ ++line; if(strstr(string, argv[1])) printf("%02d %s",line,string); } } O fluxo stderr usado para mostrar mensagens de erro no ecr, quando a sada do programa esta redirecionada p/ outro dispositivo que nao seja o ecr: #include <stdio.h> main(int argc, char *argv[]) { FILE *fp; int count; char letter; if(argc!=2){puts("Digite o nome do arquivo!");exit(1);} if(!(fp = fopen(argv[1],"w"))){ fputs("Erro na abertura do arquivo",stderr); /* 2 opao: puts("Erro na abertura do arquivo"); -> vai p/ stdout,isto , tela ou outro dispositivo */ exit(1); } else for(letter='A';letter<='Z'; letter++) putc(letter, fp); fclose(fp); } Se a sada deste programa (stdout) for redirecionada para algum outro arquivo, a mensagem de erro forosamente aparecer no ecr porque estamos escrevendo em stderr. fprintf() e fscanf(): Comportam-se como printf() e scanf() s que escrevem e lem de arquivos de disco. Todos os cdigos de formato e modificadores sao os mesmos. Prottipo: int fprintf(FILE *fp, char *string_de_controle, lista_de_argumentos); int fscanf(FILE *fp, char *string_de_controle, lista_de_argumentos); Header: stdio.h Embora fprintf() e fscanf() sejam a maneira mais fcil de ler e escrever tipos de dados nos mais diversos formatos, elas nao sao as mais eficientes em termos de tamanho de cdigo resultante e velocidade. Quando o formato de leitura e escrita for de importncia secundria, deve-se dar preferncia a fread() e fwrite(). /* imprime os quadrados de 0 a 10 no arquivo quad.dat no formato nmero - quadrado */ #include<stdio.h> #include<stdlib.h> main() { int i; FILE *out; if((out=fopen("quad.dat","wt"))==NULL){ /* sempre texto c/ fscanf() e fprintf() */ puts("Nao posso abrir arquivo!"); exit(1); } for (i=0; i<=10; i++){ fprintf(out,"%d %d\n", i , i*i); } fclose(out);

71

} Deixaremos o exemplo de uso de fscanf() para o prximo captulo (Alocaao Dinmica). Apagando arquivos: remove() int remove(char *nome_arquivo); Retorna zero em caso de successo e nao zero se falhar. 9) ALOCAAO DINMICA DA MEMRIA: Existem trs funoes bsicas de gerenciamento dinmico da memria: malloc() free() realloc() aloca determinado tamanho de memria libera a memria alocada por malloc() p/ ser reutilizada encolhe ou expande o bloco de memria alocado por malloc(), se necessrio move-o para caber no espao disponvel

Prottipos: (header em stdlid.h) void *malloc (unsigned numbytes); malloc() retorna um ponteiro void, isto , ele pode apontar p/ o que se desejar: uma matriz, estrutura, nmero. numbytes() normalmente calculado por sizeof(). Se nao houver memria suficiente retorna um ponteiro nulo. void free(void *p); free() libera a memria alocada a partir do ponteiro gerado por malloc(). NUNCA use free() p/ liberar o bloco de memria apontada por um ponteiro que nao tenha sido gearado por malloc(), ou a memria usada ser perdida. void *realloc(void *p, unsigned tamanho); realloc() altera o tamanho da memria alocada por p para o tamanho especificado por 'tamanho'. Se nao houver memria suficiente retorna um ponteiro nulo e libera (executa free()) o bloco original. /* l um arquivo numrico de nome a especificar no formato de 2 colunas */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> #include<alloc.h> main(argc,argv) int argc; char *argv[]; { FILE *in; float *x, *y; float tempx, tempy; int i, teste,numlido; if(argc!=2){puts("Digite o nome do arquivo!");exit(1);} if((in=fopen(argv[1],"rt"))==NULL){ /* sempre texto c/ fscanf() e fprintf() */ puts("Nao posso abrir arquivo!"); exit(1); } x=(float*)malloc(sizeof(float)); if(!x){LBLMEM:puts("Memria insuficiente!");exit(1);} y=(float*)malloc(sizeof(float)); if(!y){goto LBLMEM;}

72

i=0; while(1){ teste=fscanf(in,"%f%f",&tempx,&tempy); if(teste==-1)break; /* testa EOF: fscanf() retorna -1 quando encontra-o */ *(x+i)=tempx; /* para fugir de erro de linkagem do compilador - BUG da versao 1.0 do Turbo C++ */ *(y+i)=tempy; i++; x=realloc(x,((unsigned)(i+1))*sizeof(float)); if(!x){LBLALOC:puts("Erro de alocaao!");exit(1);} y=realloc(y,((unsigned)(i+1))*sizeof(float)); if(!y){goto LBLALOC;} } fclose(in); numlido=i-1; for(i=0;i<=numlido;i++)printf("%f %f\n",*(x+i),*(y+i)); } 10) ESTRUTURAS: Em C, uma estrutura uma coleao de variveis referenciada sob um nome, provendo um meio de manter informaoes relacionadas juntas. Uma declaraao de estrutura permite que se se crie variveis de estruturas. As variveis que compreendem uma estrutura sao chamadas de elementos da estrutura: struct endereco { char nome[30]; char rua[40]; char cidade[20]; char estado[3]; unsigned long int cep; }; Obs1) O nome endereco chamada etiqueta da estrutura Obs2) At aqui nao foi alocada nenhuma memria struct endereco info_adr; /* declara a varivel info_adr como sendo do tipo estrutura endereco */ /* somente agora alocada memria */ Outra maneira, alocando memria j na declaraao da estrutura: struct endereco { char nome[30]; char rua[40]; char cidade[20]; char estado[3]; unsigned long int cep; }info_adr, binfo, cinfo; /*Isto definir um tipo de estrutura chamada endereco e declarar as variveis info_adr, binfo, cinfo como sendo deste tipo.*/ Se voc necessita somente de uma varivel de um determinado tipo, a etiqueta nao necessria: struct { char nome[30]; char rua[40]; char cidade[20]; char estado[3]; unsigned long int cep; }info_adr; /* declara uma varivel chamada info_adr como definida pela estrutura que a

73

precedeu */ A forma geral de uma declaraao de estrutura : struct nome_do_tipo_da_estrutura { tipo nome_de_elemento_1; tipo nome_de_elemento_2; tipo nome_de_elemento_3; . . . tipo nome_de_elemento_N; }variveis_estrutrura; onde o nome do tipo da estrutura (etiqueta) ou as variveis da estrutura pode ser omitido, mas nao os dois. Referenciando os membros da estrutura: operador ponto nome_da_varivel_estrutura.nome_do_elemento Exemplo: info_adr.cep=91720260; /* atribui */ printf("%lu", info_adr.cep); /* imprime cep no ecr */ printf("%s",info_adr.nome); /* imprime nome */ for(t=0;info_adr.nome[t];++t)putchar(info_adr.nome[t]); /* dem caracter por caractere */

Matrizes de estruturas: struct endereco info_adr[100]; /* cria 100 conjuntos de variveis que sao */ /* organizadas como definido na estrutura */ /* endereco */ printf("%lu", info_adr[2].cep); /* imprime no ecr o cep da estrutura 3 da */ /* matriz de 100 estruturas */ Matrizes de estruturas sao muito comuns em planilhas de clculo tipo Lotus, em que se armazena uma grande quantidade de dados encadeados. Um exemplo de uso de estruturas: uma lista de endereos simples Para ajudar a mostrar como as estruturas e matrizes de estruturas sao usadas, um programa simples de listas de endereos que usa uma matriz de estruturas para armazenar a informaao do endereo ser desenvolvido. As funoes neste programa interagem com as estruturas e seus elementos para ilustrar o uso de estruturas. Neste exemplo, a informaao que ser armazenada inclui: nome rua cidade estado cep Para definir a estrutura de dados bsica, endereco, que guardar esta informaao, voc poder escrever: struct endereo{ char nome[40]; char rua[40]; char cidade[30]; char estado[3];

74

char cep[10]; }info_adr[tamanho]; Note que essa estrutura usa um string para armazenar o cdigo do cep em vez de um inteiro sem sinal.Essa abordagem acomoda tanto os cdigos postais com letras como aqueles com nmeros, como os usados pelo Canad e outros pases. A matriz info_adr contm TAMANHO estruturas do tipo endereco, onde TAMANHO pode ser definido para satisfazer uma necessidade especfica. Fica aqui a sugestao que se tente reescrever este programa utilizando as funoes de alocaao dinmica do Turbo C no sentido de automatizar a alocaao de memria para a matriz info_adr. A primeira funao necessria ao programa main() mostrada aqui: main(void) { char opcao; inic_list(); for(;;) { opcao=menu(); switch(opcao) { case 'i': insere(); break; case 'e': exibe(); break; case 'c': carrega(); break; case 's': salva(); break; case't': return 0; } } } Primeiro, a funao inic_list() prepara a matriz de estrutura para uso, colando um caractere nulo no primeiro byte do campo nome de cada estrutura na matriz. O programa assume que uma varivel estrutura nao est em uso se o campo nome est vazio. A funao inic_list() escrita como mostrado aqui: /* Inicializa a matriz info_adr.*/ void inic_list(void) { register int t; for(t=0; t<TAMANHO; t++) *info_adr[t].nome='\0'; /* um nome com comprimento zero significa vazio*/ } A funao menu() exibir as opoes e retornar opao do usurio: /* Obtm uma seleao no menu.*/ menu(void) { char ch; do{ printf("(I)nsere\n"); printf("(E)xibe\n"); printf("(C)arrega\n"); printf("(S)alva\n"); printf("(T)ermina\n\n"); printf("sua opao:"); ch= getche(); printf("\n"); } while(!strchr("iecst", tolower(ch)));

75

return tolower(ch);

Essa funao faz uso de outra funao de biblioteca do Turbo C, strchr(), que tem este prottipo: char *strchr(char *str,char ch); Essa funao pesquisa na string apontada por str uma ocorrncia do caractere em ch. Se o caractere encontrado, um ponteiro para ele retornado. Por definiao, um valor verdadeiro. Contudo, se nenhuma correspondncia for encontrada, um nulo retornado, que por definiao falso. Essa funao usada neste programa para ver se o usurio informa uma opao vlida. A funao insere() solicita ao usurio informar os dados e entao coloca a informaao passada na primeira estrutura livre. Se a matriz estiver cheia, a mensagem "lista cheia" impressa no ecr. /*Insere nomes na lista.*/ void insere (void) { register int i; /* encontra a primeira estrutura livre*/ for(i+0; i<TAMANHO; i++) if(!*info_adr[i].nome)break; /* i ser igual a tamanho se a lista estiver cheia.*/ if(i==TAMANHO){ printf("lista cheia\n"); return; } /* Insere a informaao*/ printf("nome: "); gets(info_adr[i].nome); printf("rua: "); gets(info_adr[i].rua); printf("cidade: "); gets(info_adr[i].cidade); printf("estado: "); gets(info_adr[i].estado); printf("cep: "); gets(info_adr[i].cep); } As rotinas salva() e carrega() mostradas aqui sao usadas para salvar e carregar o banco de dados da lista de endereos. Note a pouca quantidade de cdigo em cada rotina por causa do poder das funoes fread() e fwrite(): /* Salva a lista.*/ void salva(void) { FILE *fp; register int i; if((fp=fopen("listend.dat","wb"))==NULL){ printf("o arquivo nao pode ser aberto\n"); return; } for(i=0; i<TAMANHO; i++) if(*info_adr[i].nome) if(fwrite(&info_adr[i], sizeof(struct endereco), 1, fp)!=1) printf("erro de escrita no arquivo\n"); fclose(fp);

76

/* Carrega o arquivo.*/ void carrega (void) { FILE *fp; register int i; if((fp=fopen("listend.dat","rb"))==NULL){ printf("o arquivo nao pode ser aberto\n"); return;

inic_list(); for(i=0; i<TAMANHO; i++) if(fread(&info_adr[i], sizeof(struct endereco), 1, fp)!=1){ if(feof(fp)){ fclose(fp); return; } printf("erro de leitura no arquivo\n"); } } As duas rotinas confirmam uma operaao com sucesso no arquivo checando o valor retornado pela funao fread() ou fwrite(). Ainda, a funao carrega() deve checar explicitamente o fim do arquivo pelo uso da funao feof(), uma vez que a funao fread() retorna o mesmo valor se o fim do arquivo encontrado ou se tiver ocorrido um erro. A funao final que o programa necessita exibe(). Ela imprime a lista de endereos inteira no ecr: /* Exibe a lista.*/ void exibe(void) { register int t; for(t=0; t<TAMANHO; t++){ if(*info_adr[t].nome){ printf("Nome: %s\n", info_adr[t].nome); printf("Rua: %s\n", info_adr[t].rua); printf("Cidade: %s\n", info_adr[t].cidade); printf("Estado: %s\n", info_adr[t].estado); printf("CEP: %s\n\n", info_adr[t].cep); } } } A listagem completa do programa Lista de Endereos mostrada aqui: /* uma lista de endereos simples utilizando uma matriz de estruturas */ #include<conio.h> #include<stdio.h> #include<ctype.h> #include<string.h> #include<stdlib.h> #define TAMANHO 100 struct endereco{ char nome[40]; char rua[40]; char cidade[30]; char estado[3]; char cep[10]; }info_adr[TAMANHO]; void insere(void), inic_list(void), exibe(void); void salva(void), carrega(void);

77

int menu(void); main(void) { char opcao; inic_list(); for(;;) { opcao=menu(); switch(opcao) { case 'i': insere(); break; case 'e': exibe(); break; case 'c': carrega(); break; case 's': salva(); break; case't': return 0; } }

/* Inicializa a matriz info_adr.*/ void inic_list(void) { register int t; for(t=0; t<TAMANHO; t++) *info_adr[t].nome='\0'; /* um nome com comprimento zero significa vazio*/ } /* Obtm uma seleao no menu.*/ menu(void) { char ch; do{ printf("(I)nsere\n"); printf("(E)xibe\n"); printf("(C)arrega\n"); printf("(S)alva\n"); printf("(T)ermina\n\n"); printf("sua opao:"); ch= getche(); printf("\n"); } while(!strchr("iecst", tolower(ch))); return tolower(ch); } /*Insere nomes na lista.*/ void insere (void) { register int i; /* encontra a primeira estrutura livre*/ for(i=0; i<TAMANHO; i++) if(!*info_adr[i].nome)break; /* i ser igual a tamanho se a lista estiver cheia.*/ if(i==TAMANHO){ printf("lista cheia\n"); return; } /* Insere a informaao*/ printf("nome: ");

78

gets(info_adr[i].nome); printf("rua: "); gets(info_adr[i].rua); printf("cidade: "); gets(info_adr[i].cidade); printf("estado: "); gets(info_adr[i].estado); printf("cep: "); gets(info_adr[i].cep); } /* Exibe a lista.*/ void exibe(void) { register int t; for(t=0; t<TAMANHO; t++){ if(*info_adr[t].nome){ printf("Nome: %s\n", info_adr[t].nome); printf("Rua: %s\n", info_adr[t].rua); printf("Cidade: %s\n", info_adr[t].cidade); printf("Estado: %s\n", info_adr[t].estado); printf("CEP: %s\n\n", info_adr[t].cep); } } } /* Salva a lista.*/ void salva(void) { FILE *fp; register int i; if((fp=fopen("listend.dat","wb"))==NULL){ printf("o arquivo nao pode ser aberto\n"); return;

} for(i=0; i<TAMANHO; i++) if(*info_adr[i].nome) if(fwrite(&info_adr[i], sizeof(struct endereco), 1, fp)!=1) printf("erro de escrita no arquivo\n"); fclose(fp); } /* Carrega o arquivo.*/ void carrega (void) { FILE *fp; register int i; if((fp=fopen("listend.dat","rb"))==NULL){ printf("o arquivo nao pode ser aberto\n"); return; } inic_list(); for(i=0; i<TAMANHO; i++) if(fread(&info_adr[i], sizeof(struct endereco), 1, fp)!=1) { if(feof(fp)){ fclose(fp); return; } printf("erro de leitura no arquivo\n"); }

79

Atribuindo Estruturas: #include<stdio.h> main(void) { struct exemplo { int i; double d; } um,dois; um.i=10; um.d=98.6; dois=um; /* atribui a estrutura um a estrutura dois */ printf("%d %lf", dois.i, dois.d); return 0; } Obs) Nao possvel atribuir uma estrutura a outra se elas sao de tipos Passando estruturas para funoes: Passando elementos de estrutura para funoes: Seja struct { char x; int y; float z; char s[10]; } exemplo; Chamando funoes por valor: func(exemplo.x); /* passa p/ func() o valor do char em x */ func2(exemplo.y); /* passa p/ func2() o valor do int em y */ func3(exemplo.z); /* passa p/ func3() o valor do float em z */ func4(exemplo.s); /* passa p/ func4() o endereo da string em s */ func(exemplo.s[2]); /* passa p/ func() o valor do char em s[2] */ Chamando funoes por referncia: func(&exemplo.x); /* passa p/ func() o endereo do char em x */ func2(&exemplo.y); /* passa p/ func2() o endereo do int em y */ func3(&exemplo.z); /* passa p/ func3() o endereo do float em z */ func4(exemplo.s); /* passa p/ func4() o endereo da string em s */ func(&exemplo.s[2]); /* passa p/ func() o endereo do char em s[2] */ Observe que o operador & precede o nome da varivel estrutura, nao o nome do elemento individual. Note tambm que o elemento string s j significa um endereo, assim o operador & nao requerido. Passando estruturas inteiras para funoes: Quando uma estrutura usada como argumento para uma funao, a estrutura inteira passada usandose o mtodo padrao de chamada por valor. Isto significa, obviamente, que qualquer mudana feita no contedo de uma estrutura dentro da funao para qual ela passada, nao afeta a estrutura usada como argumento. diferentes.

80

A consideraao mais importante para se ter em mente quando se usa uma estrutura como argumento que os tipos devem ser compatveis na chamada e na declaraao da funao. Este programa declara os argumentos arg e parm para serem do mesmo tipo de estrutura: #include<stdio.h> /* define um tipo de estrutura: */ struct exemplo { int a,b; char ch; }; /* nenhuma memria alocada at aqui */ void f1(); main(void) { struct exemplo arg; /* aloca memria para arg */ arg.a=1000; f1(arg); return 0; } void f1(parm) struct exemplo parm; { printf("%d",parm.a); } Apontadores para estruturas: Existe uma desvantagem maior em passar tudo, at a mais simples estrutura para funoes: a sobrecarga da pilha implicando em perda de tempo na transferncia de dados entre uma funao chamadora e funao chamada. Alm do perigo bvio do overflow da pilha (stack overflow). Em estruturas simples, com poucos elementos, esse esforo nao tao importante, mas se muitos elementos sao usados e se algum dele matriz, a velocidade de execuao pode ser completamente degradada ao utilizarmos chamada de funoes por valor. A soluao passar para a funao chamada somente um ponteiro para a estrutura argumento. Quando um ponteiro para estrutura passado para uma funao, somente o endereo da estrutura passado para a funao: somente o endereo da estrutura empurrado para a pilha. Isto significa que uma chamada de funao extremamente rpida pode ser executada. Tambm, como estaremos referenciando a estrutura argumento em si e nao uma cpia na pilha, a funao chamada ter condioes de modificar o contedo dos elementos da estrutura argumento usada na chamada. Para encontrar o endereo de uma varivel estrutura, o operador & colocado antes do nome da varivel estrutura: struct sal { float saldo; char nome[80]; } cliente; struct sal *ptr; /* declara um ponteiro para estruturas tipo sal */ ptr= &cliente; /* coloca o endereo de cliente em ptr */ Para acessar o elemento saldo atravs de ptr temos duas sintaxes: float x,y; x= (*ptr).saldo /* atribui a x o saldo de cliente */ y= ptr->saldo /* atribui a y o saldo de cliente */

81

A segunda sintaxe de acesso usa o chamado operador seta e universalmente preferida em relaao a primeira. Seguem dois exemplos de programas, um chamando funoes por valor e outro por referncia a estruturas de dados. Ambos utilizam a palavra reservada #typedef que define um novo nome para um tipo de dado j existente. Nao confundir com #define que atribui a uma sequncia de caracteres um identificador: #define ERRO "Impossvel abrir arquivo\n" /* atribui o identificador ERRO */ /* sequncia de caracteres que o segue */ puts(ERRO); /-------/ #typedef struct cliente_tipo { float divida; int atraso; char nome[40]; char tipo_divida[80]; } cliente; cliente lista_clientes[NUM_CLIENTES] /* define uma matriz de estruturas do */ /* tipo cliente chamada lista_clientes */ /* Obs) cliente nao uma varivel do tipo cliente_tipo mas sim um outro nome para a struct cliente_tipo ! */ puts(lista_clientes[4].nome);/*imprime no ecr o 5 nome da lista de clientes*/ /* Programa de operaoes simples c/ complexos para demonstraao */ /* da passagem de estruturas p/ funoes com chamada por valor */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> typedef struct z { /* a etiqueta z opcional */ double re; double im; } cpx; typedef struct zplus { /* a etiqueta zplus opcional */ double double double double cpxp cpxp cpxp cpxp cpxp re; im; mag; ang;

} cpxp;

soma(); subt(); mult(); divi(); polar();

main() { char op,sign; cpx z1,z2; cpxp zout;

82

do{ printf("\nQual operaao com complexos desejada? [+ - * /] "); op=getche(); }while(op!='+'&&op!='-'&&op!='*'&&op!='/'); puts("\n\nInforme dois complexos Z1=R1+j*I1 e Z2=R2+j*I2 entrando"); puts("na seguinte forma: R1 I1 R2 I2 "); scanf("%lf %lf %lf %lf",&z1.re,&z1.im,&z2.re,&z2.im); switch(op) { case '+': zout = soma(z1,z2); break; case '-': zout = subt(z1,z2); break; case '*': zout = mult(z1,z2); break; case '/': zout = divi(z1,z2); break; default: puts("Erro!"); exit(1); } if(zout.im<0){sign='-'; zout.im= -(zout.im);} /* sign d o sinal da */ else{sign='+';} /* parte imaginria de */ /* zout no formato da printf */ printf("\nZ1 %c Z2 = %lf %c j*%lf =",op,zout.re,sign,zout.im); printf(" %lf<%lf",zout.mag,zout.ang); } cpxp soma(arg1,arg2) cpx arg1,arg2; { cpxp res; res.re = arg1.re + arg2.re; res.im = arg1.im + arg2.im; res = polar(res); return res;

cpxp subt(arg1,arg2) cpx arg1,arg2; { cpxp res; res.re = arg1.re - arg2.re; res.im = arg1.im - arg2.im; res = polar(res); return res;

cpxp mult(arg1,arg2) cpx arg1,arg2; { cpxp res;

83

res.re = (arg1.re)*(arg2.re)-(arg1.im)*(arg2.im); res.im = (arg1.im)*(arg2.re)+(arg1.re)*(arg2.im); /* (a+jb)(c+jd) = (ac-bd) + j(bc+ad) */ res = polar(res); return res;

} cpxp divi(arg1,arg2) cpx arg1,arg2; { cpxp res;

res.re = (arg1.re)*(arg2.re)+(arg1.im)*(arg2.im); res.im = (arg1.im)*(arg2.re)-(arg1.re)*(arg2.im); res.re = res.re/(arg2.re*arg2.re + arg2.im*arg2.im); res.im = res.im/(arg2.re*arg2.re + arg2.im*arg2.im); /* (a+jb)/(c+jd) = ((ac+bd) + j(bc-ad))/(c^2 + d^2) */ res = polar(res); return res;

cpxp polar(arg) cpxp arg; { cpxp res; res.mag = sqrt((arg.re)*(arg.re)+(arg.im)*(arg.im)); res.ang =(180/M_PI)*atan2(arg.im,arg.re); res.re = arg.re; res.im = arg.im; return res; } /----------/ /* Programa de operaoes simples c/ complexos para demonstraao */ /* da passagem de estruturas p/ funoes com chamada por referncia */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> typedef struct z { /* a etiqueta z opcional */ double re; double im;

} cpx;

typedef struct zplus { /* a etiqueta zplus opcional */ double double double double cpxp cpxp cpxp cpxp re; im; mag; ang;

} cpxp; soma(); subt(); mult(); divi();

84

cpxp polar(); main() { char op,sign; cpx z1,z2; cpxp zout; do{ printf("\nQual operaao com complexos desejada? [+ - * /] "); op=getche(); }while(op!='+'&&op!='-'&&op!='*'&&op!='/'); puts("\n\nInforme dois complexos Z1=R1+j*I1 e Z2=R2+j*I2 entrando"); puts("na seguinte forma: R1 I1 R2 I2 "); scanf("%lf %lf %lf %lf",&z1.re,&z1.im,&z2.re,&z2.im); switch(op) { case '+': zout = soma(&z1,&z2); break; case '-': zout = subt(&z1,&z2); break; case '*': zout = mult(&z1,&z2); break; case '/': zout = divi(&z1,&z2); break; default: puts("Erro!"); exit(1); } if(zout.im<0){sign='-'; zout.im= -(zout.im);} /* sign d o sinal da */ else{sign='+';} /* parte imaginria de */ /* zout no formato da printf */ printf("\nZ1 %c Z2 = %lf %c j*%lf =",op,zout.re,sign,zout.im); printf(" %lf<%lf",zout.mag,zout.ang); } cpxp soma(arg1,arg2) cpx *arg1,*arg2; { cpxp res; res.re = arg1->re + arg2->re; res.im = arg1->im + arg2->im; res = polar(&res); return res;

cpxp subt(arg1,arg2) cpx *arg1,*arg2; { cpxp res;

85

res.re = arg1->re - arg2->re; res.im = arg1->im - arg2->im; res = polar(&res); return res; } cpxp mult(arg1,arg2) cpx *arg1,*arg2; { cpxp res; res.re = (arg1->re)*(arg2->re)-(arg1->im)*(arg2->im); res.im = (arg1->im)*(arg2->re)+(arg1->re)*(arg2->im); /* (a+jb)(c+jd) = (ac-bd) + j(bc+ad) */ res = polar(&res); return res;

cpxp divi(arg1,arg2) cpx *arg1,*arg2; { cpxp res; res.re = (arg1->re)*(arg2->re)+(arg1->im)*(arg2->im); res.im = (arg1->im)*(arg2->re)-(arg1->re)*(arg2->im); res.re = res.re/(arg2->re*arg2->re + arg2->im*arg2->im); res.im = res.im/(arg2->re*arg2->re + arg2->im*arg2->im); /* (a+jb)/(c+jd) = ((ac+bd) + j(bc-ad))/(c^2 + d^2) */ res = polar(&res); return res;

} cpxp polar(arg) cpxp *arg; { cpxp out; out.mag = sqrt((arg->re)*(arg->re)+(arg->im)*(arg->im)); out.ang =(180/M_PI)*atan2(arg->im,arg->re); out.re = arg->re; out.im = arg->im; return out; } Matrizes dentro de estruturas: struct x { int a[10][10]; /* matriz de 100 inteiros */ float b; }y; valor = y.a[3][7]; /* atribui a valor o elemento (3,7) da matriz a da estrutura y */ Estruturas dentro de estruturas (estruturas aninhadas): struct endereco { char nome[30]; char rua[40]; char cidade[20]; char estado[3]; unsigned long int cep; };

86

struct empregado { struct endereco residencia; /* residencia uma struct do tipo endereco */ float salario; /* aninhada na struct funcionario que do */ } funcionario; /* tipo empregado */ funcionario.residencia.cep = 90500657; /* atribui o cep 90500657 para o campo cep da residencia do funcionario */ Resumindo: Os elementos de cada estrutura sao referenciados da esquerda para a direita do mais externo para o mais interno. 11) CAMPOS DE BITS: Ao contrrio de muitas outras linguagens, a linguagem C tem um modo prprio de acessar um ou mais bits dentro de um byte ou palavra. Isto pode ser til por diversas razoes: 1) se o espao limitado podese armazenar muitas variveis Boolean (V/F) em um byte ; 2) certas interfaces de dispositivos transmitem informaoes codificadas em bits dentro de um byte ; 3) certas rotinas de codificaao de dados necessitam acessar os bits dentro de um byte. Um dos mtodos que a linguagem C usa para acessar bits baseia-se na estrutura chamada campo-bit. Um campo-bit na realidade, somente um tipo de membro de estrutura que define o quao grande, em bits, cada elemento deve ser. A forma geral de uma declaraao campo-bit feita como segue: struct nome_do_tipo_estrutura { tipo nome_1: tamanho; tipo nome_2: tamanho; tipo nome_3: tamanho; . . . tipo nome_N: tamanho; } Um campo-bit deve ser declarado como int unsigned ou signed. Campos-bit de tamanho 1 devem ser declarados como unsigned, j que um nico bit nao pode ter sinal. Exemplo aplicativo: A funao biosequip() retorna uma lista dos perifricos instalados no computador por meio de um valor de 16 bits codificado como mostrado aqui: int biosequip(void); /* header em bios.h */

----------------------------------------------------------------------------Bit: Perifrico: ----------------------------------------------------------------------------0 obriga o boot atravs do drive de disco flexvel 1 coprocessador matemtico presente 2,3 tamanho da ram na placa mae: 0,0: 16Kb 0,1: 32Kb 1,0: 48Kb 1,1: 64Kb 4,5 modo inicial de vdeo: 0,0: nao usado 0,1: colorido ou BW 40x25 1,0: colorido ou BW 80x25 1,1: monocromtico 80x25 6,7 nmero de drives de disco flexvel: 0,0: 1 0,1: 2 1,0: 3 1,1: 4

87

8 9,10,11

chip DMA instalado nmero de portas seriais: 0,0,0: 0 0,0,1: 1 0,1,0: 2 0,1,1: 3 1,0,0: 4 1,0,1: 5 1,1,0: 6 1,1,1: 7 12 joystick instalado 13 impressora serial instalada (somente PCjr) 14,15 nmero de impressoras: 0,0: 0 0,1: 1 1,0: 2 1,1: 3 ----------------------------------------------------------------------------Isto pode ser representado como um campo-bit, usando-se esta estrutura:

struct perif { unsigned boot_floppy: 1; unsigned tem_coproc: unsigned ram_placa: 2; unsigned modo_video: 2; unsigned drive_floppy: 2; unsigned DMA: 1; unsigned portas: 3; unsigned joystick: 1; unsigned sem_uso: 1; unsigned num_impressoras: } eq;

1;

2;

O programa seguinte usa este campo-bit para exibir o nmero de drives de discos flexveis e o nmero de portas seriais: #include<stdio.h> #include<bios.h> main() { struct perif { unsigned boot_floppy: 1; unsigned tem_coproc: unsigned ram_placa: 2; unsigned modo_video: 2; unsigned drive_floppy: 2; unsigned DMA: 1; unsigned portas: 3; unsigned joystick: 1; unsigned sem_uso: 1; unsigned num_impressoras: 2; } eq; int *i; i=(int *)&eq; *i= biosequip(); printf("%d drives \n", eq.drive_floppy+1); /* (0,0) = 1 drive */ printf("%d portas \n", eq.portas); return 0; } O endereo eq atribudo ao ponteiro para inteiro i e o valor de retorno da funao biosequip() atribudo a eq por meio deste ponteiro. Isto necessrio, uma vez que a linguagem C nao permitir que um inteiro seja um ponteiro para uma estrutura.

1;

88

Como se pode ver por este exemplo, cada campo-bit associado usando-se o operador Entretanto, se a estrutura referenciada por meio de um ponteiro, deve-se usar o operador seta.

ponto.

Nao necessrio nomear cada campo-bit. Isto torna fcil encontrar o bit que se quer, rejeitando os nao usados. Por exemplo, o campo sem_uso pode ser deixado sem nome, como mostrado: struct perif { unsigned boot_floppy: 1; unsigned tem_coproc: unsigned ram_placa: 2; unsigned modo_video: 2; unsigned drive_floppy: 2; unsigned DMA: 1; unsigned portas: 3; unsigned joystick: 1; unsigned: 1; unsigned num_impressoras: } eq;

1;

2;

Variveis do tipo campo-bit tm certas restrioes. Nao se pode obter oendereo de uma varivel campobit. As variveis campo-bit nao podem ser colocadas em matrizes. Nao se pode ultrapassar o limite dos inteiros. Se a mquina usa a regra memria_baixa-byte_baixo (DEC e National) ou memria_baixa-byte_alto (Intel e Motorola) afeta o modo de endereamento dos campos. Isto implica que qualquer cdigo que usa os campos-bit pode ter alguma dependncia de mquina. Pode-se misturar membros normais de estruturas com elementos campos-bit. Por exemplo: struct emp { struct addr endereco; float salario; unsigned situacao: 1; /* demitido ou em atividade */ unsigned contrato: 1; /* horista ou mensalista */ unsigned deducoes: 3; /* deducoes: IR,IAPAS,...(at 2^3=8) */ }; Esta estrutura define um registro de empregados que usa somente um byte para armazenar tres pedaos de informaao: situaao, contrato e deduoes. Sem o uso do campo-bit estas informaoes poderiam requerer 3 bytes. 12) UNIONS: Em C uma union uma localizaao de memria usada por muitos declaraao de uma union similar a de uma estrutura, como mostrado: union u_tipo { int i; char ch; }; Assim como com estruturas, esta declaraao nao declara qualquer varivel. Pode-se declarar uma varivel tanto colocando-se o nome dela no fim da declaraao como pelo uso de uma declaraao em separado. Para declarar uma varivel union chamada cnvt, do tipo u_tipo usando a declaraao dada anteriormente escreve-se: union u_tipo cnvt; Em cnvt, o inteiro i e o caractere ch compartilham a mesma localizaao de memria. Obviamente i ocupa 2 bytes e ch ocupa 1 byte. A figura a seguir mostra como i e ch compartilham o mesmo endereo: tipos de variveis diferentes. A

Byte

Byte 1

ch

89

i Quando uma varivel union declarada o compilador aloca automaticamente um rea de memria grande o suficiente para armazenar o tipo da varivel de maior tamanho da union. Para acessar um elemento de uma union, usa-se a mesma sintaxe de estruturas: os operadores ponto e seta. Quando se opera diretamente na union, usa-se o operador ponto. Se a varivel union acessada por meio de um ponteiro, usa-se o operador seta. Por exemplo, para atribuir o inteiro 10 ao elemento i de cnvt deve-se escrever cnvt.i = 10; Unioes sao usadas frequentemente quando sao necessrios conversoes de tipo de dados nao realizveis com um cast simples, uma vez que elas permitem que se considere uma regiao de memria de mais de uma maneira. Por exemplo, a funao de biblioteca putw() escrever a representaao binria de um inteiro para um arquivo em disco. Ainda que existam muitas maneiras de codificar esta funao, a que mostrada aqui usa uma varivel union. Primeiro criada uma union compreendida por um inteiro e por uma matriz de caracteres de 2 bytes: union pw { int i; char ch[2]; }; Agora a funao putw() pode ser escrita usando-se a union: void putw(union pw word,FILE *fp) { putc(word.ch[0],fp); /* escreve 1 metade */ putc(word.ch[1],fp); /* escreve 2 metade */ } Ainda que chamada com um inteiro, putw() pode usar putc() para escrever um inteiro para um arquivo em disco apontado por fp. O programa a seguir combima unioes com campos-bit para exibir o cdigo ASCII em binrio gerado quando se pressiona uma tecla. A union permite funao getche() atribuir o valor de uma tecla uma varivel tipo char enquanto o campo-bit usado para exibir os bits individuais. /* exibe os cdigos ASCII em binrio para caracteres */ /* um campo-bit ser decodificado */ #include<stdio.h> #include<conio.h> struct byte { int a: 1; int b: 1; int c: 1; int d: 1; int e: 1; int f: 1; int g: 1; int h: 1; }; union bits { char ch; struct byte bit; }ascii; void decode(union bits b); main() {

90

do{ ascii.ch=getche(); printf(": "); decode(ascii); }while(ascii.ch!='q'); /* encerra se digitado 'q' */ return 0; } /* exibe a sequencia de bits para cada caractere: */ void decode(union bits b) { if(b.bit.h)printf("1 "); else printf("0 "); if(b.bit.g)printf("1 "); else printf("0 "); if(b.bit.f)printf("1 "); else printf("0 "); if(b.bit.e)printf("1 "); else printf("0 "); if(b.bit.d)printf("1 "); else printf("0 "); if(b.bit.c)printf("1 "); else printf("0 "); if(b.bit.b)printf("1 "); else printf("0 "); if(b.bit.a)printf("1 "); else printf("0 "); printf("\n"); } Para encerrar, vamos ver como uma union pode prover um meio de carregar um inteiro em um campo de bit. No programa anterior que determinava o nmero de drives e portas seriais, tivemos que usar um artifcio devido linguagem C nao permitir que um inteiro aponte para uma estrutura. Contudo, este programa cria uma union que contm um inteiro e um campo de bit. Quando a funao biosequip() retorna a lista de perifricos codificada como um inteiro, este atribudo para o inteiro no campo de bit. Entretanto, o programa est livre para usar o campo-bit quando for reportar os resultados. /* Mostra o nmero de acionadores de disco flexvel e portas usando union */ #include<stdio.h> #include<bios.h> main() { struct perif { unsigned boot_floppy: 1; unsigned tem_coproc: 1; unsigned ram_placa: 2; unsigned modo_video: 2; unsigned drive_floppy: 2; unsigned DMA: 1; unsigned portas: 3; unsigned joystick: 1; unsigned sem_uso: 1; unsigned num_impressoras: 2; }; union { struct perif eq; unsigned i; } eq_union; eq_union.i= biosequip(); printf("%d drives \n", eq_union.eq.drive_floppy+1); /* (0,0) = 1 drive */ printf("%d portas \n", eq_union.eq.portas); return 0;

91

} 13) OPERADORES LGICOS DE BIT: Diferente de muitas outras, a linguagem C suporta um arsenal completo de operadores de bit. J que a linguagem C foi projetada para tomar o lugar da linguagem Assembly em muitas tarefas de programaao, era importante que ela tivesse a habilidade de suportar todas (ou, no mnimo a maior parte) as operaoes que podem ser feitas em Assembler. Operaoes bit-a-bit referem-se a testar, indicar ou movimentar os bits em um byte que corresponde aos tipos char, int ou long em C. Operadores de bit nao podem ser usados com os tipos float, double, long double, void ou outro tipo mais complexo. Operadores de bit: -----------------------------------------------------------------------------Operador Aao -----------------------------------------------------------------------------a&b cada bit de a AND cada bit de b a|b cada bit de a NOR cada bit de b a^b cada bit de a XOR cada bit de b ~a NOT a (complemento de 1) v>>b v deslocado b bits p/ a direita v<<b v deslocado b bits p/ a esquerda -----------------------------------------------------------------------------Exemplos: /* converte para binrio */ #include<stdio.h> void exibe_binario(); main() { int i; Linit: printf("Digite um inteiro: "); scanf("%d",&i); if(i>256){puts("O inteiro deve ser menor que 256!");goto Linit;} puts("\nRepresentaao binria do inteiro: "); exibe_binario(i); return 0; } /*Exibe os bits dentro de um byte: */ void exibe_binario(int i) { register int t; for(t=128;t>0;t=t/2) if(i&t) printf("1 "); else printf("0 "); printf("\n"); } /* Desloca um bit dentro de um byte e mostra */ #include <stdio.h> void exibe_binario(); main() { int i=1, t; for(t=0;t<8;t++){ exibe_binario(i); i=i<<1; } printf("\n");

92

for(t=0;t<8;t++){ i=i>>1; exibe_binario(i); } return 0; } /*Exibe os bits dentro de um byte*/ void exibe_binario(int i) { register int t; for(t=128;t>0;t=t/2) if(i&t) printf("1 "); else printf("0 "); printf("\n"); } /* codifica e decodifica uma string */ #include<stdio.h> main() { char str[80]; int i; puts("Digite uma string: "); gets(str); for(i=0;i<strlen(str);i++) str[i]=~str[i]; /* NOT em cada um dos 8 bits de */ /* cada char que compoe str */ puts("A string codificada/decodificada : "); puts(str); return 0; } 14) FUNOES DE CONTROLE DO ECR: O controle do ecr do Turbo C dividido em duas partes: modo texto e modo grfico. As funoes de modo texto gerenciam a aparncia da tela quando a placa de vdeo est selecionada para um modo de texto e as funoes grficas usam o ecr quando a placa de vdeo est selecionada para um modo de exibiao grfica. Ambos os modos sao selecionveis via software, por comandos especficos que veremos adiante. 14.1) AS FUNOES BSICAS DE MODO TEXTO: Antes de explorar algumas das mais importantes funoes de controle e manipulaao de tela no modo texto, h alguns pontos a considerar. Primeiro, todas as funoes de modo texto no Turbo C requerem a presena do arquivo-cabealho conio.h em qualquer programa que as use. Segundo, as funoes de modo texto consideram que o ecr est em um modo texto, nao grfico. J que esse modo o padrao quando da inicializaao (boot) do PC, isto nao deve ser problema. Brevemente aprenderemos a fixar os diversos modos de vdeo. Janelas de Texto: Muitas das rotinas do modo texto do Turbo C operam em uma janela, nao no ecr. Felizmente, a janela padrao a tela inteira, de modo que nao necessrio se preocupar com a criaao de qualquer janela especial para usar as rotinas de modo texto (e grfico). Entretanto, importante entender os conceitos bsicos das janelas, a fim de se obter o mximo das funoes de tela do Turbo C.Uma janela uma rea retangular que o seu programa usa para enviar mensagens para o usurio. Ela pode ser tao grande quanto o ecr inteira ou tao pequena quanto uns poucos carcateres. Em programas sofisticados, nao incomum a tela ter muitas janelas ao mesmo tempo - uma para cada tarefa ealizada pelo programa.

93

O Turbo C permite definir a localizaao e as dimensoes de uma janela. Depois de se definir uma janela, as rotinas que manipulam texto afetam somente a janela definida e nao o ecr inteira. Por exemplo, a funao clrscr() apaga a janela ativa, nao o ecr inteira (a menos,obviamente,que,como por definiao, a janela ativa seja o ecr inteira). Adiconalmente, todas as coordenadas de posiao sao relativas janela ativa em vez de tela. Um dos aspectos mais importantes desse assunto que o Turbo C automaticamente previne as sadas de irem alm dos limites de uma janela. Se alguma sada ultrapassar um limite, somente a parte que couber ser exibida, o resto iniciado na prxima linha. Para janelas de texto, as coordenadas do canto superior esquerdo sao (1,1). As funoes de tela modo texto que usam coordenadas aplicam-nas relativas janela ativa nao tela. Assim, se existem duas janelas, o canto superior esquerdo de ambas considerado a localizaao (1,1) quando alguma est ativa. Para o momento, nao nos preocuparemos com a criaao de qualquer janela, simplesmente usaremos a janela padrao, o ecr inteira. Limpando a janela: Uma das funoes mais comuns de manipulaao de tela a que limpa a janela ativa (e somente a janela ativa): void clrscr(void); Posicionando o cursor: Para posicionar o cursor na janela ativa usa-se: void gotoxy(int x, int y); Aqui, x e y especificam as coordenadas em relaao janela ativa para onde o cursor posicionado. Se ambas as coordenadas estao fora do limite, nenhuma aao realizada, contudo isso nao considerado um erro. Exemplo: /*Demonstraao das funoes clrscr() e gotoxy()*/ #include <conio.h> #include <stdio.h> #include <string.h> char mensagem[]="Funoes de Tela Modo Texto Sao Um Tdio!"; main() { register int x,y; char *p; clrscr(); p=mensagem; for(x=1;x<=strlen(mensagem); x++){ for (y=1; y<=12; y++){ gotoxy (x,y); delay(50); /* espera 0.05 segundos */ printf("%c",*p); /* imprime de cima para baixo */ gotoxy (x, 25-y); printf("%c",*p); /* imprime de baixo para cima */ } p++; } getche(); gotoxy(1,25); return 0; } Apagando at o fim da linha:

94

A funao clreol() limpa da esquerda para a direita, uma linha a partir da posiao corrente do cursor at o final da janela. til quando se requisita informaao do usurio: void clreol(void); Pode-se combinar esta funao com gotoxy para exibir uma mensagem de aviso nas coordenadas especificadas, aps ter limpado a linha. til para manter o ecr livre de mensagens perdidas: #include <conio.h> #include <stdio.h> void aviso(); main() { clrscr(); aviso("isto um aviso",1,10); getch(); aviso("isto tambm", 1, 10); getch(); return 0; } void aviso(char *s, int x, int y) { gotoxy(x,y); clreol(); printf(s); } Apagando e Inserindo Linhas: Quando se tem o ecr cheia de texto e se quer remover (nao simplesmente apagar) uma ou mais linhas, reposicionando as linhas de texto abaixo da removida (scroll) ou quando se quer criar uma linha em branco onde se quer inserir um texto usa-se delline() e insline(): void delline(void); void insline(void); Exemplo: #include <conio.h> #include <stdio.h> main() { int x, y; clrscr(); /* Preenche o ecr inteira com algumas linhas: */ for(y=1; y<25; y++) for(x=1; x<80; x++){ gotoxy(x,y); printf("%c",(y-1)+'A'); } getch(); /*apaga todas as linhas de dois em dois a partir dos B's: */ /* e rola o ecr para cima */ for(y=2;y<26;y+=2){ gotoxy(1,y); delline(); } return 0;

95

} Variaao do programa anterior: #include <conio.h> #include <stdio.h> main() { int x, y; clrscr(); /* Preenche o ecr inteira com algumas linhas: */ for(y=1; y<25; y++) for(x=1; x<80; x++){ gotoxy(x,y); printf("%c",(y-1)+'A'); } getch(); /*apaga todas as linhas alternadas e insere uma linha em branco no lugar*/ for(y=2;y<26;y+=2){ gotoxy(1,y); delline(); insline(); } return 0; } Criando Janelas: Para se criar window(): janelas de qualquer tamanho e em qualquer lugar , desde que caibam no ecr, usa-se

void window(int esquerda, int acima, int direita, int abaixo); Se alguma coordenada invlida, window nao realiza qualquer aao. /* demonstra window() */ #include <conio.h> #include <stdio.h> main() { system("cls"); /*limpa o ecr */ gotoxy(2,3); printf("coord (2,3) do ecr"); window(10,10,60,15); gotoxy(2,3); printf("coord (2,3) da janela"); } Observe que as coordenadas usadas para chamar window sao absolutas do ecr e nao relativas janela correntemente ativa. Isto significa que mltiplas janelas podem ser abertas sem terem coordenadas uma relativa outra, isto , nao sao consideradas aninhadas. No entanto, uma janela pode superpor-se outra janela. /*Um programa de demonstraao de janela de texto*/ #include <conio.h> #include <stdio.h>

96

void borda(); main() { clrscr(); /*Desenha uma borda em volta do ecr*/ borda(1,1,79,25); /*Cria a primeira janela*/ window(3,2,40,9); borda(3,2,40,9); gotoxy(3,2); printf("Primeira janela"); delay(700); /* espera 0.7 segundos */ /*Cria uma segunda janela*/ window(30,10,60,18); borda(30,10,60,18); gotoxy(3,2); printf("Segunda janela"); delay(700); /* espera 0.7 segundos */ gotoxy(5,4); printf("Ainda na segunda janela"); delay(700); /* espera 0.7 segundos */ /*Retorna para a primeira janela*/ window(3,2,40,9); gotoxy(3,3); printf("De volta primeira janela"); delay(700); /* espera 0.7 segundos */ /*Demonstra sobreposiao de janelas*/ window(5,5,50,15); borda(5,5,50,15); gotoxy(2,2); printf("Esta janela sobrepoe s duas primeiras"); getch(); return 0; } /*Desenha uma borda em torno de uma janela de texto*/ void borda(int iniciox, int inicioy, int fimx, int fimy) { register int i; gotoxy(1,1); for(i=0; i<=fimx-iniciox; i++) putch('*'); /* putch() escreve um char p/ a */ /* janela ativa */ gotoxy(1, fimy-inicioy); for(i=0; i<=fimx-iniciox; i++) putch('*'); for(i=2; i<fimy-inicioy; i++) { gotoxy(1,i); putch('*'); gotoxy(fimx-iniciox+1,i); putch('*'); } } Algumas funoes de E/S em janelas:

97

Quando se usa a janela padrao, que o ecr inteira, nao importa se sao usadas as funoes de E/S padrao (como printf, etc..) ou as funoes de E/S de janela. No entanto, se estivermos usando uma janela pequena em algum lugar do ecr, seremos obrigados a usar as funoes de janela. ----------------------------------------------------------------------------Funao: Uso: ----------------------------------------------------------------------------cprintf() como printf() s que escreve para a janela ativa cputs() como puts() s que escreve para a janela ativa putch() escreve um char para a janela ativa getche() l um caractere de dentro da janela ativa cgets() l uma string de dentro da janela ativa ----------------------------------------------------------------------------Quando se usa uma funao de janela o texto de entrada ou sada se quebra automaticamente no limite da janela ativa: /* demonstraao de algumas das funoes de janela de texto */ #include<conio.h> #include<stdio.h> void borda(); main() { clrscr(); /*cria uma janela*/ window(3,2,20,9); borda(3,2,20,9); gotoxy(2,2); cprintf("Esta linha quebrar na borda da janela "); printf("Esta linha nao quebrar na borda da janela"); getch(); return 0; } /*Desenha uma borda em torno de uma janela de texto*/ void borda(int iniciox, int inicioy, int fimx, int fimy) { register int i; gotoxy(1,1); for(i=0; i<=fimx-iniciox; i++) putch('*'); /* putch() escreve um char p/ a */ /* janela ativa */ gotoxy(1, fimy-inicioy); for(i=0; i<=fimx-iniciox; i++) putch('*'); for(i=2; i<fimy-inicioy; i++) { gotoxy(1,i); putch('*'); gotoxy(fimx-iniciox+1,i); putch('*'); } } Outra caracterstica importante das funoes de janela de texto que elas nao sao redirecionveis pelo DOS nem a nvel de entrada nem a nvel de sada. Experimente redirecionar a sada do programa anterior para um arquivo e observe o ecr e arquivo(use type). Modos Texto: O padrao PC-IBM para modo de vdeo umo ecr com 80 colunas por 25 linhas. Dependendo da placa de vdeo que um computador dispoe possvel selecionar vrios modos de vdeo para texto pela funao textmode():

98

void textmode(int modo); ____________________________________________________________________________ modo (macro): inteiro equivalente: descriao: _____________________________________________________________________________ BW40 0 40 col. PB C40 1 40 col. colorido BW80 2 80 col. 16 tons cinza C80 3 80 col. colorido MONO 7 80 col. monocromtico LASTMODE -1 modo anterior (original) _____________________________________________________________________________ Exibindo textos em cores: (obviamente necessrio um vdeo colorido) A funao textcolor(cor) muda a cor dos textos subsequentes para a cor especificada pelo argumento cor. S funciona se for usada uma das funoes de janela (printf() nao funciona): void textcolor(int cor); Obs: 0<=cor<=15 Do mesmo modo, podemos definir uma cor de fundo para os textos: void textbackground(int cor); Obs: 0<=cor<=7 importante observar que estas funoes s alteram sadas de texto aps sua chamada no programa. _____________________________________________________________________________ cor(macro): inteiro equivalente: _____________________________________________________________________________ BLACK 0 BLUE 1 GREEN 2 CYAN 3 RED 4 MAGENTA 5 BROWN 6 LIGHTGREY 7 DARKGREY 8 LIGHTBLUE 9 LIGHTGREEN 10 LIGHTCYAN 11 LIGHTRED 12 LIGHTMAGENTA 13 YELLOW 14 WHITE 15 BLINK 128 _____________________________________________________________________________ /* este programa define todas as combinaoes de cores para textos */ #include<conio.h> main() { register int ft,fd; textmode(C80); for(ft=BLUE;ft<=WHITE;ft++){ for(fd=BLACK;fd<=LIGHTGRAY;fd++){

99

textcolor(ft); textbackground(fd); delay(70); cprintf("Isto um teste "); } cprintf("\n\r"); } textcolor(WHITE); textbackground(BLACK); cprintf("pronto"); textmode(LASTMODE); return 0; } Para fazer com que um texto pisque, deve-se fazer o OR da cor com textcolor(GREEN | BLINK); 14.2) UMA INTRODUAO AO SISTEMA GRFICO DO TURBO C: Acima de Tudo uma Janela: Como as funoes de controle de tela do modo texto, todas as funoes grficas operam por meio de uma janela. Na terminologia do Turbo C, uma janela grfica chamada de porta de visualizaao, mas uma porta de visualizaao tem essencialmente as mesmas caractersticas das janelas de texto. A nica diferena real entre uma janela e uma porta de visualizaao que o canto superior esquerdo de uma porta de visualizaao a coordenada(0,0) e nao (1,1) como em uma janela. Por definiao, umo ecr inteira uma porta de visualizaao, contudo, pode-se criar portas de visualizaao de outras dimensoes, como veremos mais tarde. importante ter em mente que todas as sadas grficas sao relativas porta de visualizaao corrente, que nao necessariamente a mesma que o ecr. Inicializando a Placa de Vdeo: Antes que qualquer das funoes grficas possam ser usadas, necessrio colocar a placa de vdeo em um dos modos grficos. Obviamente o computador deve ter uma placa grfica de vdeo, caso contrrio, nenhuma das funoes e comandos vistos daqui por diante serao vlidos. Por definiao, a grande maioria dos sistemas PC-IBM inicializam (boot) o computador no modo texto 80 colunas. J que este nao o modo grfico, as funoes grficas nao podem funcionar nele. Para inicializar a placa de vdeo em um modo grfico, usa-se a funao initgraph(): void far initgraph(int far *driver,int far *modo, char far *path); A funao initgraph() carrega um driver grfico correspondente ao nmero na memria apontado por driver. Sem um driver grfico carregado na memria, nenhuma funao grfica pode operar. O argumento modo aponta para o inteiro que especifica um dos modos de vdeo disponvel no driver grfico.O argumento path especifica em que local da estrutura de diretrios encontra-se o arquivo do driver grfico utilizado. Se nenhum path especificado, o diretrio de trabalho corrente pesquisado. Os drivers grficos estao contidos nos arquivos *.BGI, que devem estar disponveis no sistema. _______________________________________________ Drive Grfico Inteiro Equivalente ----------------------------------------------------------------------------DETECT 0 CGA 1 MCGA 2 EGA 3 EGA64 4 EGAMONO 5 IBM8514 6 HERCMONO 7 ATT400 8 VGA 9 PC3270 10 ----------------------------------------------------------------------------128 (BLINK):

100

Quando se usa DETECT, initgraph detecta automaticamente o tipo de placa e monitor de vdeo presente no sistema e seleciona o modo de vdeo com a melhor resoluao. Isso faz com que os valores dos argumentos driver e modo sejam fixados automaticamente. O valor de modo deve ser um dos modos grficos mostrados na tabela abaixo. ___________________________________________________________________________ Placa Vdeo Modo Inteiro Equivalente Resoluao (driver *.BGI) _____________________________________________________________________________ CGA CGAC0 0 320x200 CGAC1 1 320x200 CGAC2 2 320x200 CGAC3 3 320x200 CGAHI 4 640x200 MCGA MCGAC0 0 320x200 MCGAC1 1 320x200 MCGAC2 2 320x200 MCGAC3 3 320x200 MCGAMED 4 640x200 MCGAHI 5 640x480 EGA EGALO 0 640x200 EGAHI 1 640x350 EGA64 EGA64LO 0 640x200 EGA64HI 1 640x350 EGAMONO EGAMONOHI 3 640x350 HERC HERCMONOHI 0 720x348 ATT400 ATT400C0 0 320x200 ATT400C1 1 320x200 ATT400C2 2 320x200 ATT400C3 3 320x200 ATT40CMED 4 640x200 ATT400CHI 5 640x400 VGA VGALO 0 640x200 VGAMED 1 640x250 VGAHI 2 640x480 PC3270 PC3270HI 0 720x350 IBM8514 IBM8514LO 0 640x480 IBM8514HI 1 1024x768 __________________________________________________________________________ Por exemplo, para fazer pontos de resoluao, use: 7 #include<graphics.h> int driver, modo; driver=CGA; modo=CGAC0; initgraph(&driver,&modo,""); /*assume o arquivo CGA.BGI no diretrio corrente*/ Deixando o modo grfico temporariamente e definitivamente: #include<graphics.h> int driver, modo; driver=CGA; modo=CGAC0; initgraph(&driver,&modo,"");/*assume o arquivo CGA.BGI no diretrio corrente*/ { faz algo no modo grfico } restorecrtmode() /* volta p/ texto temporariamente */ { faz algo no modo texto com que o sistema seja inicializado no modo grfico CGA 4 cores 320x200

101

} setgraphmode(modo) /* volta ao modo grfico dado por modo */ { faz algo no modo grfico novamente } closegraph() /* sai definitivamente do modo grfico: esvazia a memria do driver *.BGI */ Cores e Paletas: (EGA/VGA): O tipo de placa de vdeo conectada ao seu sistema determina os tipos e cores disponveis quando se est em modo grfico. A tendncia atual a linha de monitores VGA (onde se incluem os EGA). A CGA 4 cores d acesso a 4 cores por paleta e 4 paletas para escolher.Comparado com os demais monitores, o CGA de uma limitaao mpar no que diz respeito a apresentaao de modo grfico colorido. Nao cobriremos cores no CGA. Nos monitores EGA, VGA e SVGA quando se quer estabelecer uma cor para as diversas funoes de desenho do Turbo C sobre uma cor de fundo faz-se: setcolor(cor); /* d a cor do traado de uma funao de desenho subsequentemente chamada */

setbkcolor(cor_de_fundo); /* d a cor de fundo do traado de uma funao de desenho subsequentemente chamada */ onde cor e cor_de_fundo sao dados pelas tabelas abaixo: _____________________________________________________________________________ cor inteiro equivalente: _____________________________________________________________________________ EGA_BLACK 0 EGA_BLUE 1 EGA_GREEN 2 EGA_CYAN 3 EGA_RED 4 EGA_MAGENTA 5 EGA_BROWN 20 EGA_LIGHTGREY 7 EGA_DARKGREY 56 EGA_LIGHTBLUE 57 EGA_LIGHTGREEN 58 EGA_LIGHTCYAN 59 _____________________________________________________________________________ cor inteiro equivalente: _____________________________________________________________________________ EGA_LIGHTRED 60 EGA_LIGHTMAGENTA 61 EGA_YELLOW 62 EGA_WHITE 63 _____________________________________________________________________________ ____________________________________________________________________________ cor_de_fundo inteiro equivalente: _____________________________________________________________________________ BLACK BLUE GREEN CYAN RED MAGENTA BROWN 1 2 3 5 6 4 0

102

LIGHTGREY 7 DARKGREY 8 LIGHTBLUE 9 LIGHTGREEN 10 LIGHTCYAN 11 LIGHTRED 12 LIGHTMAGENTA 13 YELLOW 14 WHITE 15 _____________________________________________________________________________ As Funoes Bsicas de Desenho: As funoes grficas mais fundamentais sao aquelas que desenham um ponto, uma linha e um crculo. Essas funoes sao chamadas de putpixel(), line() e circle(), respectivamente. Os seus prottipos sao mostrados aqui: void far putpixel(int x, int y, int cor); void far line(int comeox, int comeoy, int fimx, int fimy); void far circle(int x, int y, int raio); A funao putpixel() escreve a cor especificada na localizaao determinada por x e y. A funao line() desenha uma linha a partir da localizaao especificada por (comeox, comeoy) at (fimx, fimy), na cor corrente estabelecida por setcolor(). A cor de desenho padrao branco. A funao circle() desenha um crculo de raio igual a raio, na cor estabelecida por setcolor(), com centro determinado por (x,y). Se qualquer das coordenadas crculo dentro do limite. estiver fora dos limites, ser exibida somente a parte (se houver) do

/*Demonstraao de pontos, linhas ,crculos e sada de texto*/ #include<graphics.h> #include<conio.h> main() { int driver,modo; register int i; driver=DETECT; initgraph(&driver,&modo,""); setcolor(EGA_LIGHTGREEN); line(0,0,200,150); setcolor(EGA_LIGHTRED); line(50,100,200,125); /*Desenha alguns pontos*/ for(i=0;i<319;i+=10)putpixel(i,100,EGA_YELLOW); /*Desenha alguns crculos*/ setcolor(EGA_LIGHTBLUE); circle(50,50,35); setcolor(EGA_LIGHTMAGENTA); circle(100,160,100); sleep(1); /* espera 1 segundo */ cleardevice(); /* limpa o ecr grfica */ /* escreve uma string de texto iniciando no meio do ecr grafica */ setbkcolor(LIGHTBLUE); /* cor de fundo azul claro */ setcolor(EGA_LIGHTRED); outtextxy(getmaxx()/2,getmaxy()/2,"Isto um teste"); /* imprime a string iniciando no meio do ecr */

103

getch(); closegraph(); return 0; } Preenchendo uma rea: Para preencher qualquer forma geomtrica fechada, usa-se a funao floodfill(): void far floodfill(intx, inty, int cor_da_borda); Para usar essa funao chame-a com as coordenadas(x,y) de um ponto dentro da figura e a cor da linha que forma a borda da figura fechada. importante certificar-se de que o objeto a ser preenchido est completamente fechado, caso contrrio, a rea fora da figura ser preenchida tambm. O que preencher o objeto determinado pelo padrao e cor corrente de preenchimento. Por definiao, a cor de fundo definida por setbkcolor() usada. Entretanto, possvel mudar o modo como os objetos sao preenchidos usando-se a funao setfillstyle(): void far setfillstyle(int padrao, int cor); Onde padrao graphics.h. definido pelo tpico 'constants, data types and global variables' tem 'fill_pattern' em

A Funao Rectangle: A funao rectangle desenha uma caixa definida na cor de desenho corrente: pelas coordenadas (esquerda,acima) e (direita,baixo)

void far rectangle(int esquerda, int acima, int dreita, int abaixo) /*programa de demonstraao de cor, retngulos e padroes de preenchimento*/ #include<graphics.h> #include<conio.h> main() { int driver, modo; register int i; driver=VGA; modo=VGAMED; initgraph(&driver,&modo,""); /*Desenha uma borda no ecr*/ setcolor(EGA_RED); rectangle(0,0,639,349); sleep(1); /* espera 1 segundo */ /* desenha diagonal */ setcolor(EGA_YELLOW); line(0,0,639,349); sleep(1); /* desenha um retangulo */ setcolor(EGA_LIGHTGREEN); rectangle(100,100,300,200); sleep(1); /* preenche o retangulo anterior */ setfillstyle(SOLID_FILL,BLUE); floodfill(110,110,EGA_LIGHTGREEN); sleep(1);

104

/* desenha uma linha */ setcolor (EGA_BLUE); line(50,200,400,125); sleep(1); /*Desenha o setor superior em 'V'*/ setcolor (EGA_LIGHTRED); for(i=0;i<640;i+=3) line(320,174,i,0); sleep(1); /* desenha alguns circulos */ setcolor(EGA_LIGHTGREEN); circle(50,50,35); setfillstyle(SOLID_FILL,MAGENTA); /* preenche */ floodfill(50,50,EGA_LIGHTGREEN); setcolor(EGA_LIGHTMAGENTA); /* nao preenche este */ circle(320,175,100); setcolor(EGA_LIGHTCYAN); /* nao preenche este */ circle(100,100,200); sleep(1); /*Desenha um alvo*/ setcolor(EGA_GREEN); circle(500,250,90); setfillstyle(SOLID_FILL,GREEN); floodfill(500,250,EGA_GREEN); /* preenche crculo mais externo */ setcolor (EGA_RED); circle (500,250,60); setfillstyle(SOLID_FILL,RED); floodfill(500,250,EGA_RED); setcolor (EGA_GREEN); circle (500,250,30); setfillstyle(SOLID_FILL,GREEN); floodfill(500,250,EGA_GREEN); setcolor (EGA_RED); circle (500,250,10); setfillstyle(SOLID_FILL,RED); floodfill(500,250,EGA_RED); getch(); closegraph(); } Criando Portas de Visualizaao: possvel criar portas de visualizaao em modo grfico de maneira similar forma como se cria janelas no modo texto. Como afirmado anteriormente, todas as sadas grficas sao relativas s coordenadas da porta de visualizaao ativa. Isto significa que as coordenadas do canto superior esquerdo da porta sao (0,0), nao interessando onde a porta de visualizaao esteja. A funao usada para criar uma porta de visualizaao grfica chama-se setviewport(): void far setviewport(int esquerda, int acima, int direita, int abaixo, int clipflag); Para us-la especifica-se as coordenadas do canto superior esquerdo e do canto inferior direito da porta. Se o parmetro clipflag diferente de zero, ser realizado um truncamento automtico da sada que pode exceder os limites da porta de visualizaao. Isto evita consumo intil da memria de vdeo. Com clipping desabilitado possvel a sada ir alm da porta de visualizaao. Pode-se encontrar as dimensoes da porta de visualizaao corrente usando-se a funao getviewsetings():

105

void far getviewsettings(struct viewporttype far *info); A estrutura viewporttype definida em graphics.h como mostrado aqui: struct viewporttype{ int left, top, right, bottom; int clipflag; } Os campos left, top, right e bottom armazenam as coordenadas do canto superior esquerdo e canto superior direito da porta de visualizaao.Quando o clipflag zero, nenhum truncamento da sada ocorre caso esta ultrapasse os limites da porta. O uso mais importante da funao getviewsettings permitir que os programas se ajustem automaticamente s dimensoes da tela ditadas pela placa de vdeo presente no sistema. Consultando a estrutura viewporttype o programa pode saber as dimensoes da porta de visualiza ao corrente e fazer o ajuste necessrio em suas variveis internas para que a rea grfica caia no ecr. /*Demonstraao de portas de visualizaao*/ #include<graphics.h> #include<stdlib.h> #include<conio.h> main() { int driver, modo; struct viewporttype porta; int largura, altura; register int i; driver=DETECT; initgraph(&driver, &modo,""); /*Obtm as dimensoes da porta.*/ getviewsettings(&porta); /*Desenha linhas verticais na primeira porta.*/ for (i=0;i<porta.right;i+=20) line(i,porta.top,i,porta.bottom); /*Cria uma nova porta no centro da porta.*/ altura=porta.bottom/4; largura=porta.right/4; setviewport(largura,altura,largura*3,altura*3,1); getviewsettings(&porta); /*Desenha linhas horizontais dentro da segunda porta.*/ for(i=0;i<porta.right;i+=10)line(0,i,porta.bottom,i); getch(); closegraph(); return 0; } 15) APNDICE I: TABELA DE CARACTERES ASCII E GERADOR

106

/* Gerador da tabela ASCII */ #include<stdlib.h> main() { unsigned int i; for(i=0;i<33;i++){ if(i==26)printf("26 EOF"); else printf("%d %c",i,i); printf(" "); } getch(); puts("\n"); for(i=33;i<=255;i++){ printf("%d %c",i,i); printf("\t"); } } 16) APNDICE II: PROGRAMA DEMONSTRATIVO DE JANELAS POP-UPS /* window.c: testa operacoes com janelas no modo texto */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> #include<string.h> void cwindow();

107

void main() {

int i; system("cls"); cwindow(1,1,80,25,edge1,LIGHTCYAN,CYAN); /* desenha janela de fundo CYAN */ /* e borda LIGHTCYAN tipo edge1 */ cwindow(5,5,30,15,NULL,LIGHTBLUE,BLUE); /* janela c/ fundo BLUE sem borda */ textcolor(LIGHTBLUE);/* proxima sada de texto em cor LIGHTBLUE */ cputs("Iniciando em (1,1) desta janela. Note que os caracteres ficam confinados internamente, embora a largura da linha exceda a da janela."); cwindow(40,10,70,15,edge2,BLACK,GREEN); /*janela: fundo GREEN, borda BLACK tipo edge2 */ textcolor(YELLOW); /* prox. sada de texto em YELLOW */ gotoxy(5,2);cputs("Iniciando em (5,2) desta janela."); cwindow(10,20,40,25,edge1,LIGHTMAGENTA,MAGENTA);/*janela: fundo MAGENTA, borda LIGHTMAGENTA tipo edge1 */ textcolor(LIGHTRED);/* prox sada texto em LIGHTRED */ gotoxy(6,1);cputs("CONTAGEM REGRESSIVA:"); for(i=10;i>=0;i--){ gotoxy(15,3);cprintf("%02d",i); /* sada para a janela ativa! */ sleep(1); } cwindow(5,15,35,20,edge3,LIGHTRED,RED);/*janela: fundo RED, borda LIGHTRED tipo edge3 */ textcolor(WHITE); /* prox sada texto em WHITE */ gotoxy(13,3);cputs("BUM!"); sleep(1); cwindow(50,5,75,10,edge1,LIGHTBLUE|BLINK,BLUE);/* fundo BLUE, borda LIGHTBLUE piscando tipo edge1 */ textcolor(YELLOW|BLINK);/* prox sada texto em YELLOW piscando */ gotoxy(4,2); cprintf("Digite sua idade:"); cscanf("%d",&i); /* pega dado da janela ativa */ gotoxy(4,3); textcolor(WHITE);/* prox sada texto em WHITE */ cprintf("Voce tem %d anos!",i); sleep(2); system("cls"); /* limpa e apaga o ecr */ } void cwindow(lft,up,rgt,dwn,edge,edgecolor,backgrcolor) int lft,up,rgt,dwn; int edgecolor,backgrcolor; char *edge;

108

{ register int i,j; char line[80]=""; char tmp[2]; union REGS inregs,outregs; struct text_info ti;

window(lft+1,up+1,rgt-1,dwn-1); /* define area da janela */ textbackground(backgrcolor); /* define cor de fundo */ clrscr(); /* limpo ecr e pinta fundo */ window(1,1,80,25); /* redefine: janela = janela total */ /* coloca borda */ if(edge!=NULL){ /* define linhas horizontais da borda */ tmp[0]=edge[5]; tmp[1]='\0'; j= lft+1; while(j<=rgt-1){ strncat(line,tmp,1); j++; } /* imprime lados horizontais da borda */ textcolor(edgecolor); gotoxy(lft+1,up); cprintf("%s",line); gotoxy(lft+1,dwn); cprintf("%s",line); /* imprime lados verticais da borda */ i=up+1; while(i<=dwn-1) { gotoxy(lft,i); putch(edge[4]); gotoxy(rgt,i); putch(edge[4]); i++; } /* coloca caracteres nos 4 cantos da janela */ gotoxy(lft,up);putch(edge[0]); gotoxy(rgt,up);putch(edge[1]); gotoxy(lft,dwn);putch(edge[3]); /* truque p/ evitar rolagem do caso tela = tela total (80x25) */ /* quando escrevendo o caracter do canto inferior direito da janela */ gotoxy(rgt,dwn); gettextinfo(&ti); /* pega text info e bota na struct ti */ inregs.h.ah=0x09;/* int 10h, funao 09h: escreve char na pos. cursor */ /* c/ atributo especificado e NAO avana cursor */ /* That's the trick! */ inregs.h.al=edge[2]; /* char a ser escrito */ inregs.h.bh=0; /* pagina ativa */

109

inregs.h.bl=ti.attribute; /* atributo definido na struct ti */ inregs.x.cx=1; /* fator de repetiao */ int86(0x10,&inregs,&outregs); } window(lft+1,up+1,rgt-1,dwn-1); /* reduz em uma unidade as dimensoes */ /* da janela ativa p/ evitar transbordo */ /* alem da borda definida quando a linha */ /* a ser impressa excede tamanho da janela */

REGS (union) is defined in union REGS { struct WORDREGS x; struct BYTEREGS h; }; BYTEREGS and WORDREGS are defined in <DOS.H> Structures for storing byte and word registers struct BYTEREGS { unsigned char al, ah, bl, bh; unsigned char cl, ch, dl, dh; }; struct WORDREGS { unsigned int ax, bx, cx, dx; unsigned int si, di, cflag, flags; }; int86 is defined in <DOS.H> <DOS.H>

int int86(int intno, union REGS *inregs, union REGS *outregs); text_info is defined in Current text window information. Used by the gettextinfo function. struct text_info { unsigned char winleft; unsigned char wintop; unsigned char winright; unsigned char winbottom; unsigned char attribute; unsigned char normattr; unsigned char currmode; left window coordinate top window coordinate right window coordinate bottom window coordinate text attribute normal attribute current video mode: BW40, BW80, C40, C80, or C4350 unsigned char screenheight; text screen's height unsigned char screenwidth; text screen's width unsigned char curx; x-coordinate in current window unsigned char cury; y-coordinate in current window }; <CONIO.H>

110

17) APNDICE III: PROGRAMA DEMONSTRATIVO DE CONTROLE DO MOUSE EM MODO TEXTO /* Programa simples p/ identificaao sonora de cones c/ Mouse IBM */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> #include<graphics.h> #include<alloc.h> #include<string.h> #define #define #define #define #define #define #define #define #define #define #define NOT_MOVED 0 RIGHT 1 LEFT 2 UP 3 DOWN 4

WLFT 1 WUP 1 WDWN WUP+HEIG HEIG 2 /* altura da janela internamente sem gap */ HINIT 5 /* pos. horiz. 1a. janela desenhada na linha */ SPACE 4 /* largura de cada janela sem string e sem o gap */ /* e tambem espao entre janelas - 1 */ #define HGAP 2 /* aprox. gap entre caracter e lados horiz da janela (dep. par ou impar) */ #define VGAP 2 /* aprox gap entre lados vert. da janela (dep. par ou impar) */ void void void void void void void void void storewincoord(); wedge(); cwindow(); cmouses(); mouse_position(); mode(); cursor_on(); cursor_off(); mouse_reset();

typedef struct wincoord { int lf; int up; int rg; int dw; } wico; main() { wico wc[42]; char *notes[]={"D","R","Mi","F","Sol","L","Si"}; int freq[]={65,73,82,87,98,110,123,131,147,165,175,196,220,247,262,294,330,349,392,440,494,523,587,65 9,698,784,880,988,1047,1175,1319,1397,1568,1760,1976,2093,2349,2637,2794,3136,3520,3951}; int register i; int x,y; int wup,wdwn,wlft,wrgt,nwin; int tcolor,bcolor,ecolor; int noit; system("cls"); textmode(0x03); /* 16 color text 80x25, virtual screen 640x200, mouse cursor 8x8 pixel */ /* ver tabela abaixo */

111

mouse_reset(); /* init mouse */ wup=WUP; wdwn=WDWN; wlft=WLFT+HINIT; bcolor=BLUE; ecolor=LIGHTBLUE; tcolor=WHITE; nwin=0;noit=0; while(nwin<42){ /* 6 octaves with 7 pitch */ if(nwin!=0){wlft=wrgt+SPACE;} /* pos inicial prox janela na linha horiz de janelas */ wrgt=wlft+strlen(notes[noit])+2*HGAP; if(wrgt>80){ wup=wup+HEIG+VGAP; wdwn=wdwn+HEIG+VGAP; wlft=WLFT+HINIT; wrgt=wlft+strlen(notes[noit])+2*HGAP; } storewincoord(wlft,wup,wrgt,wdwn,&wc[nwin]); /* stores window coordinates */ cwindow(wlft,wup,wrgt,wdwn,bcolor,ecolor); window(wlft+1,wup+1,wrgt-1,wdwn-1); textcolor(tcolor); gotoxy((wrgt-wlft)/2,(wdwn-wup)/2); cputs(notes[noit]); bcolor++;ecolor++; if(abs(ecolor-bcolor)==8){bcolor++;} /* avoid edge color = background color */ if(bcolor%8==0||bcolor==BLACK){bcolor++;} /* avoid background color BLACK */ nwin++;noit++; if(nwin%7==0){noit=0;} /* if remainder zero -> end of octave */ } window(1,1,80,25); /* back to original window */ textcolor(YELLOW); cursor_on(); do{ mouse_position(&x,&y); /* show mouse position */ gotoxy(1,25);cprintf("Virtual Coords: %03d %03d",x,y); if(!leftb_pressed()){nosound();} else { for(i=0;i<42;i++){ if(x<=wc[i].rg&&x>=wc[i].lf&&y>=wc[i].up&&y<=wc[i].dw){sound(freq[i]);break;} } } /* looping up till right button pressed */ } while(!rightb_pressed()); system("cls"); cursor_off(); textmode(LASTMODE); return 0; } /* selects videomode - nao usada -> igual a textmode() */ void mode(mode_code) int mode_code; {

112

union REGS r; r.h.al = mode_code; r.h.ah = 0; int86(0x10,&r, &r); } /************************************************/ /* mouse interfacing functions /************************************************/ /* turn-off mouse cursor */ void cursor_off() { int fnum; fnum=2; /* cursor off */ cmouses(&fnum,&fnum,&fnum,&fnum); } /* turn-on mouse cursor */ void cursor_on() { int fnum; fnum=1; /* cursor on */ cmouses(&fnum,&fnum,&fnum,&fnum); } /* returns V if rightbutton pressed, F otherwise */ rightb_pressed() { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); return arg2&2; } /* returns V if leftbutton pressed, F otherwise */ leftb_pressed() { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); return arg2&1; } /* return mouse cursor coordinates */ void mouse_position(x,y) int *x,*y; { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); *x=arg3; *y=arg4; } /* init mouse */ void mouse_reset() { int fnum,arg2,arg3,arg4; fnum=0; /* init mouse */ cmouses(&fnum,&arg2,&arg3,&arg4); if(fnum!=-1){puts("No mouse driver installed!");exit(1);} /*if(arg2!=2){puts("Two buttons mouse required!");exit(1);}*/ } void cmouses(fnum,arg2,arg3,arg4) int *fnum,*arg2,*arg3,*arg4; { */

113

union REGS regs; regs.x.ax = *fnum; regs.x.bx = *arg2; regs.x.cx = *arg3; regs.x.dx = *arg4; int86(0x33, &regs, &regs); *fnum=regs.x.ax; *arg2=regs.x.bx; *arg3=regs.x.cx; *arg4=regs.x.dx; } /************************************************/ /* miscellaneous functions /************************************************/ void cwindow(int lft,int up,int rgt,int dwn,int back,int edge) { register int i,j; window(lft,up,rgt,dwn); textcolor(back); textbackground(back); for(j=1;j<=dwn-up+1;j++){ gotoxy(1,j); for(i=0;i<=rgt-lft;i++){putch(' ');} } textcolor(edge); wedge(lft,up,rgt,dwn); } void wedge(int sx,int sy,int ex,int ey) */

void storewincoord(wlft,wup,wrgt,wdwn,sptr) int wlft,wup,wdwn; wico *sptr; { /* text screen (1,1,80,25); virtual screen (0,0,640,200); mouse cursor 8x8 */ /* xt=1 -> xv=0 e xt=80 -> xv=640-8=632*/ /* yt=1 -> yv=0 e yt=25 -> xv=200-8=192*/ /* 640-8 & 200-8 -> cursor mouse = 8x8 pixels */ /* coordinates transformation: virtual = 8*text - 8 */ sptr->lf=8*wlft-8; sptr->up=8*wup-8; sptr->rg=8*wrgt-8; sptr->dw=8*wdwn-8; }

114

18) APNDICE IV: PROGRAMA DEMONSTRATIVO DE CONTROLE DO MOUSE EM MODO GRFICO /* Programa de desenho simples p/ Mouse IBM e monitor VGA */ /* EGAVGA.bgi deve estar no diretorio corrente */ #include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> #include<graphics.h> #include<alloc.h> #define ICONRIGHT 26 #define ICONDOWN 10 #define #define #define #define #define void void void void void void NOT_MOVED 0 RIGHT 1 LEFT 2 UP 3 DOWN 4

cmouses(); mouse_position(); mouse_motion(); cursor_on(); cursor_off(); mouse_reset();

main() { char deltax, deltay; int x,y,xa,ya; int graphdriver,graphmode,ncolor; char *idcolor[]={"Pr","Az","Vd","Ci","Vm","Mg","CzC","Mr","CzE","AzC","VdC","CiC","VmC","MgC","Am","Br"}; int color[]={EGA_BLACK,EGA_BLUE,EGA_GREEN,EGA_CYAN,EGA_RED,EGA_MAGENTA,EGA_LIGHTGRAY,EGA_B ROWN,EGA_DARKGRAY,EGA_LIGHTBLUE,EGA_LIGHTGREEN,EGA_LIGHTCYAN,EGA_LIGHTRED,EGA_LIGHTM AGENTA,EGA_YELLOW,EGA_WHITE}; directvideo=1; graphdriver=VGA; graphmode=VGAHI; initgraph(&graphdriver,&graphmode,""); ncolor=WHITE; setcolor(color[ncolor]); setlinestyle(SOLID_LINE,0,NORM_WIDTH); mouse_reset(); /* init mouse */ cursor_on(); outtextxy(2,2,idcolor[ncolor]); /* writes idcolor on icon */ rectangle(0,0,ICONRIGHT,ICONDOWN); /* draw icon rectangle */ setlinestyle(SOLID_LINE,0,THICK_WIDTH); do{ mouse_position(&x,&y); /* get mouse position */ if(!leftb_pressed()){xa=x;ya=y;moveto(x,y);} /* tracks mouse cursor if leftb not pressed */

115

mouse_motion(&deltax,&deltay); /* get mouse motion */ /* if right button pressed outside icon: */ if(rightb_pressed()&&(x>=ICONRIGHT||y>=ICONDOWN)){cursor_off();cleardevice();cursor_on();goto Licon;} /* if right button pressed with cursor inside icon: */ if(rightb_pressed()&&x<ICONRIGHT&&y<ICONDOWN){ ncolor++; if(color[ncolor]>EGA_WHITE)ncolor=0; Licon: setcolor(EGA_BLACK); cursor_off(); outtextxy(2,2,"___"); /* clean icon area with ascii 219 */ if(color[ncolor]==EGA_BLACK||color[ncolor]==EGA_DARKGRAY)setcolor(EGA_WHITE); else{setcolor(color[ncolor]);} /* avoid darkgray and black on icon */ outtextxy(2,2,idcolor[ncolor]); /* writes idcolor on icon */ setlinestyle(SOLID_LINE,0,NORM_WIDTH); rectangle(0,0,ICONRIGHT,ICONDOWN); /* draw icon rectangle */ setlinestyle(SOLID_LINE,0,THICK_WIDTH); cursor_on(); setcolor(color[ncolor]); /* set new color */ delay(200); /* wait 0.2 seconds */ } /* draws if left button is pressed: */ if(leftb_pressed()&&(deltax||deltay)){cursor_off();linerel(x-xa,y-ya);cursor_on();xa=x;ya=y;} } while(!(leftb_pressed() && rightb_pressed())); /* looping up till both buttons pressed together */ cleardevice(); closegraph(); cursor_off(); return 0; } /************************************************/ /* mouse interfacing functions /************************************************/ /* turn-off mouse cursor */ void cursor_off() { int fnum; fnum=2; /* cursor off */ cmouses(&fnum,&fnum,&fnum,&fnum); } /* turn-on mouse cursor */ void cursor_on() { int fnum; fnum=1; /* cursor on */ cmouses(&fnum,&fnum,&fnum,&fnum); } /* returns V if rightbutton pressed, F otherwise */

*/

116

rightb_pressed() { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); return arg2&2; } /* returns V if leftbutton pressed, F otherwise */ leftb_pressed() { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); return arg2&1; } /* return mouse cursor coordinates */ void mouse_position(x,y) int *x,*y; { int fnum,arg2,arg3,arg4; fnum=3; /* get button position and status */ cmouses(&fnum,&arg2,&arg3,&arg4); *x=arg3; *y=arg4; } /* return path direction */ void mouse_motion(deltax,deltay) char *deltax,*deltay; { int fnum,arg2,arg3,arg4; fnum=11; /* get movement direction */ cmouses(&fnum,&arg2,&arg3,&arg4); if(arg3>0){ *deltax=RIGHT;} else if(arg3<0) {*deltax=LEFT;} else {*deltax=NOT_MOVED;} if(arg4>0){ *deltay=DOWN;} else if(arg4<0) {*deltay=UP;} else {*deltay=NOT_MOVED;} } /* init mouse */ void mouse_reset() { int fnum,arg2,arg3,arg4; fnum=0; /* init mouse */ cmouses(&fnum,&arg2,&arg3,&arg4); if(fnum!=-1){puts("No mouse driver installed!");exit(1);} /*if(arg2!=2){puts("Two buttons mouse required!");exit(1);}*/ } void cmouses(fnum,arg2,arg3,arg4) int *fnum,*arg2,*arg3,*arg4; { union REGS regs; regs.x.ax = *fnum; regs.x.bx = *arg2; regs.x.cx = *arg3; regs.x.dx = *arg4; int86(0x33, &regs, &regs); *fnum=regs.x.ax;

117

*arg2=regs.x.bx; *arg3=regs.x.cx; *arg4=regs.x.dx; }

19) APNDICE V: PROGRAMA DEMONSTRATIVO DE USO DE ESTRUTURAS PARA DETERMINAAO DO CENTRO GEOMETRICO DE UM POLIEDRO SLIDO /*Calcula Centro Geometrico - Versao 1*/ #include <stdio.h> struct coord{ float x; float y; float z; }; void main() { int i,teste; FILE *fp; struct coord vert[10]; float *cg; fp=fopen("vertices.dat","rt"); if(!fp){puts("nao posso abrir arquivo");exit(1);} for(i=0;i<10;i++){ teste=fscanf(fp,"%f%f%f",&vert[i].x,&vert[i].y,&vert[i].z); if(teste==-1){puts("achei EOF!");exit(1);} } cg=med(vert); /* o nome de qualquer rea de armazenamento o endereo */ /* inicial da rea na memria!!! */ printf("%f %f %f",cg[0],cg[1],cg[2]); } float *med(V) struct coord *V; { int i; float *cg; cg=(float *)malloc(3*sizeof(float)); if(!cg){puts("memoria insuficiente");exit(1);} cg[0]=cg[1]=cg[2]=0; for(i=0;i<10;i++) { cg[0]=cg[0]+V[i].x; /* V um ponteiro para uma MATRIZ DE ESTRUTURAS e nao */ cg[1]=cg[1]+V[i].y; /* para uma VARIAVEL DE ESTRUTURA !!! */ cg[2]=cg[2]+V[i].z; } cg[0]=cg[0]/10; cg[1]=cg[1]/10; cg[2]=cg[2]/10; return cg; } /*Calcula Centro Geometrico - Versao 2 */

118

#include <stdio.h> struct coord{ float x[10]; float y[10]; float z[10]; }; float *med(); void main() { int i,teste; FILE *fp; struct coord vert; float *cg; fp=fopen("vertices.dat","rt"); if(!fp){puts("nao posso abrir arquivo");exit(1);} for(i=0;i<10;i++){ teste=fscanf(fp,"%f%f%f",&vert.x[i],&vert.y[i],&vert.z[i]); if(teste==-1){puts("achei EOF!");exit(1);} } cg=med(&vert); /* vert uma VARIAVEL DE ESTRUTURA com matrizes */ /* como membros e nao uma MATRIZ DE ESTRUTURAS!!! */ printf("%f %f %f",cg[0],cg[1],cg[2]); } float *med(V) struct coord *V; { int i; float *cg; cg=(float *)malloc(3*sizeof(float)); if(!cg){puts("memoria insuficiente");exit(1);} cg[0]=cg[1]=cg[2]=0; for(i=0;i<10;i++) { cg[0]=cg[0]+V->x[i]; /* V um PONTEIRO PARA UMA VARIAVEL DE ESTRUTURA */ cg[1]=cg[1]+V->y[i]; /* e x um membro da VARIAVEL DE ESTRUTURA apontada */ cg[2]=cg[2]+V->z[i]; /* por V o qual uma MATRIZ !!! */ } cg[0]=cg[0]/10; cg[1]=cg[1]/10; cg[2]=cg[2]/10; return cg; } O arquivo VERTICES.DAT deve conter as coordenadas dos vrtices do poliedro (no caso, 10 vrtices) no formato numrico de 3 colunas, cada uma representando respectivamente x,y,e z. Por exemplo: 0 1 2 3 4 5 6 7 8 9 7 6 5 4 3 2 1 0 8 9 9 8 7 6 5 4 3 2 1 0

119

20) APNDICE VI: PROGRAMA DEMONSTRATIVO DO USO DE APONTADORES FAR PARA ESCRITA DIRETA NA MEMORIA DE VIDEO #include <dos.h> #include <stdio.h> #include <graphics.h> int main(void) { int gd, gm, i; unsigned int far *screen; detectgraph(&gd, &gm); if (gd == HERCMONO) screen = (unsigned int *) MK_FP(0xB000, 0); else screen = (unsigned int *) MK_FP(0xB800, 0); for (i=0; i<26; i++) screen[i] = 0x0700 + ('a' + i); return 0;

21) APNDICE VII: PROGRAMA DEMONSTRATIVO DO USO DE ALOCAO DINMICA DE MEMORIA PARA OPERAOES SIMPLES COM MATRIZES BIDIMENSIONAIS /*Programa demonstra operacoes de soma , multiplicao e I/O de matrizes*/ /* Utilizando tcnicas de alocaao dinamica de memoria */ #include #include #include #include #include #include #include <stdio.h> <stdlib.h> <ctype.h> <math.h> <conio.h> <time.h> <dos.h>

void matmult(unsigned,unsigned,unsigned,double **,double **,double **); void matsum(unsigned,unsigned,double **,double **,double **); void matfprint(unsigned, unsigned, double **, char *); void matprint(unsigned,unsigned,double **); double **matinit(unsigned,unsigned); void main(void) { double **A; double **B; double **C; unsigned int ai,aj,bi,bj; register unsigned int i,j;

120

double dtemp; char op; puts("\nEste programa realiza A [op] B, sendo A e B matrizes e [op] as operaoes + ou *."); do{ printf("Operaao[*/+]?"); op=getche(); printf("\n"); if(toupper(op)=='F'){puts("Fim.");exit(0);} }while(op!='*'&&op!='+'); LBL01: printf("Nmero de linhas da matriz A?"); scanf("%u",&ai); printf("Nmero de colunas da matriz A?"); scanf("%u",&aj); A=matinit(ai,aj); for(i=0;i<=ai-1;i++){ printf("Entre linha %d da matriz A:\n",i+1); for(j=0;j<=aj-1;j++){ printf("A[%u,%u]? ",i+1,j+1); scanf("%lG",&dtemp); A[i][j]=dtemp; } } printf("Nmero de linhas da matriz B?"); scanf("%u",&bi); printf("Nmero de colunas da matriz B?"); scanf("%u",&bj); if(op=='*'&&aj!=bi){ putch(7);/* ascii BEL */ puts("\nNmero de colunas de A deve ser igual ao nmero de linhas de B!"); goto LBL01; } if(op=='+'&&(ai!=bi||aj!=bj)){ putch(7);/* ascii BEL */ puts("\nDimensoes de A e B devem ser idnticas!"); goto LBL01; } B=matinit(bi,bj); for(i=0;i<=bi-1;i++){ printf("Entre linha %d da matriz B:\n",i+1); for(j=0;j<=bj-1;j++){ printf("B[%u,%u]? ",i+1,j+1); scanf("%lG",&dtemp); B[i][j]=dtemp; } } if(op=='*'){ C=matinit(ai,bj); matmult(ai,aj,bj,A,B,C); matprint(ai,bj,C); matfprint(ai,bj,C,"matprod.dat"); } else{ C=matinit(ai,aj); matsum(ai,aj,A,B,C); matprint(ai,aj,C); matfprint(ai,aj,C,"matsoma.dat"); } }

121

double **matinit(l,c) unsigned int l,c; { double **a; register unsigned int i; a=(double **)malloc(l*sizeof(double)); if(!a) {LMEM:puts("Memria insuficiente!");exit(1);} for(i=0;i<=l-1;i++){ a[i]=(double *)malloc(c*sizeof(double)); if(!a[i]) goto LMEM; } return a; } void matprint(l,c,mat) unsigned int l,c; double **mat; { register unsigned int i,j; for(i=0;i<=l-1;i++){ for(j=0;j<=c-1;j++){ printf("%5.2lG\t",mat[i][j]); } printf("\n"); } } void matfprint(l,c,mat,arq) unsigned int l,c; double **mat; char arq[80]; { register unsigned int i,j; FILE *out; if((out=fopen(arq,"w"))==NULL) {printf("Impossvel abrir arquivo %s!",arq);exit(1);} for(i=0;i<=l-1;i++){ for(j=0;j<=c-1;j++){ fprintf(out,"%5.2lG\t",mat[i][j]); } fprintf(out,"\n"); } fclose(out); } void matmult(unsigned l1,unsigned c1,unsigned c2,double ** M1,double **M2,double **MP) { register unsigned int i,j,k; /* k-> c1=l2 */ for(i=0;i<=l1-1;i++){ /* i e j indexam MP */ for(j=0;j<=c2-1;j++){ MP[i][j]=0;/* inicializa (zera) MP[i][j] */ for(k=0;k<=c1-1;k++){ MP[i][j]+=M1[i][k]*M2[k][j]; } } } }

122

void matsum(unsigned l,unsigned c,double **M1,double **M2,double **MS) { register unsigned int i,j; for(i=0;i<=l-1;i++){ for(j=0;j<=c-1;j++){ MS[i][j]=M1[i][j]+M2[i][j]; } } }

21) APNDICE VIII: PROGRAMA EXEMPLO PARA GERAAO DE GRAFICOS CARTESIANOS NO ECR A PARTIR DE UM ARQUIVO NUMERICO. /* l um arquivo numrico de nome a especificar no formato de 2 colunas */ /* e plota no ecr */ /* Compilar no modelo de memria SMALL */ /* Necessita ser compilado e linkado atravs de arquivo de projeto o qual referencia */ /* este arquivo e os arquivos LITT.OBJ e EGAVGA.OBJ obtidos com o utilitrio */ /* BGIOBJ.EXE do diretrio /BGI */

123

#include<dos.h> #include<stdio.h> #include<stdlib.h> #include<ctype.h> #include<math.h> #include<conio.h> #include<alloc.h> #include<graphics.h> #define #define #define #define #define #define MONITOR VGA MODO VGAHI MAXX 639 /* largura do ecr em pixels */ MAXY 479 /* altura do ecr em pixels */ MME 5 /* Metade do tamanho dos Marcadores de Escala */ BUFSIZE 4096 /* Tamanho do buffer de arquivos */

void main(argc,argv) int argc; char *argv[]; { FILE *in; float *x; float *y; float delta,accum, topx, topy, botx, boty, Ax, Bx, Ay, By; int i, teste,numlido,th,tw; int gdriver, gmode, errorcode; char num[80]; if(argc!=2){puts("Digite o nome do arquivo!");exit(1);} if((in=fopen(argv[1],"rt"))==NULL){ /* sempre texto c/ fscanf() e fprintf() */ puts("Nao posso abrir arquivo!"); exit(1); } if (setvbuf(in, NULL, _IOFBF, BUFSIZE) != 0) /* cria e conecta buffer a 'in' */ puts("Nao posso estabelecer buffer para arquivo a ser lido!"); x=(float *)malloc(sizeof(float)); if(!x){LBLMEM:puts("Memria insuficiente!");exit(1);} y=(float *)malloc(sizeof(float)); if(!y){goto LBLMEM;} i=0; while(1){ teste=fscanf(in,"%f%f",x+i,y+i); if(teste==-1)break; /* testa EOF: fscanf() retorna -1 quando encontra-o */ i++; x=realloc(x,((unsigned)(i+1))*sizeof(float)); if(!x){LBLALOC:puts("Memria insuficiente!\nNao posso ler mais dados!");exit(1);} y=realloc(y,((unsigned)(i+1))*sizeof(float)); if(!y){goto LBLALOC;} } fclose(in); numlido=i-1; /* deteta top e bottom de x e y */ topx=topy=botx=boty=0; for(i=0;i<=numlido;i++){ if(x[i]>topx){topx=x[i];} if(x[i]<botx){botx=x[i];} if(y[i]>topy){topy=y[i];} if(y[i]<boty){boty=y[i];}

124

} gdriver=MONITOR; gmode=MODO; /* inicializa modo grafico */ errorcode = registerbgidriver(EGAVGA_driver); /* registra driver EGAVGA */ /* reporta qualquer erro de registro */ if (errorcode < 0)goto L_EGR; errorcode = registerbgifont(small_font); /* registra fonte 'Small_Font' */ /* reporta qualquer erro de registro */ if (errorcode < 0)goto L_EGR; initgraph(&gdriver, &gmode, ""); /* le resultado da inicializaao */ errorcode = graphresult(); if (errorcode != grOk) /* ocorreu um erro */ { L_EGR: printf("Erro grfico: %s\n", grapherrormsg(errorcode)); printf("Pressione qualquer tecla para parar."); getch(); exit(1);/* retorna p/ DOS com codigo de erro 1(nao erro = 0)*/ } /* Transfomaao Linear CoordReal ---> CoordVideo: CoordVideo = A*CoordReal + B Sejam topx e topy os valores maximos das coordenadas reais x e y e sejam botx e boty os valores minimos das coordenadas reais x e y. Sejam MAXX e MAXY respectivamente os valores inteiros mximos das coordenadas de vdeo (MAXX=639 e MAXY=479 p/ monitor VGA em modo VGAHI). Para a coordenada Vertical Y temos o seguinte sistema de equaoes: 0 = Ay*topy + By MAXY = Ay*boty + By Resolvendo p/ Ay e By temos: Ay=MAXY/(boty-topy) By=MAXY/(1-boty/topy) Para a coordenada horizontal X temos o seguinte sistema de equaoes: 0 = Ax*botx + Bx MAXX = Ax*topx + Bx Resolvendo p/ Ax e Bx temos: Ax=MAXX/(topx-botx) Bx=MAXX/(1-topx/botx) */ /* Calcula Ax,Bx,Ay e By */

125

Ax=MAXX/(topx-botx); Ay=MAXY/(boty-topy); if(botx==0){Bx=0;} /* Para evitar divisao por zero */ else{Bx=MAXX/(1-topx/botx);} if(topy==0){By=0;} /* Para evitar divisao por zero */ else{By=MAXY/(1-boty/topy);} /* Faz a transformaao de coordenadas */ for(i=0;i<=numlido;i++){ x[i]=Ax*x[i]+Bx; y[i]=Ay*y[i]+By; } /* plota no ecr */ settextstyle(SMALL_FONT, HORIZ_DIR, 4); setcolor(EGA_LIGHTBLUE); for(i=0;i<=numlido-1;i++){ line((int)x[i],(int)y[i],(int)x[i+1],(int)y[i+1]); } /* desenha linha vertical e horizontal da escala */ setcolor(EGA_YELLOW); line(MAXX/2,0,MAXX/2,MAXY); setcolor(EGA_LIGHTRED); line(0,MAXY/2,MAXX,MAXY/2); /* desenha 11 marcadores numricos na escala vertical */ setcolor(EGA_YELLOW); th=textheight("W"); tw=textwidth("W"); delta=(topy-boty)/10; for(i=0,accum=topy;i<MAXY;i+=(MAXY+1)/10,accum-=delta){ /* coloca os 10 1's marcadores */ sprintf(num,"%4.2G",accum); if(i!=(MAXY+1)/2){outtextxy(MAXX/2+MME+tw/2,i,num); line(MAXX/2-MME,i,MAXX/2+MME,i);} else{outtextxy(MAXX/2+MME+tw/2,i-2*th,num);} /* para evitar superposiao c/ eixo horizontal */ line(MAXX/2-MME,MAXY,MAXX/2+MME,MAXY); /* coloca o 11 ltimo marcador */ sprintf(num,"%4.2G",accum); outtextxy(MAXX/2+MME+tw,MAXY-th,num); /* desenha 11 marcadores numricos na escala horizontal */ setcolor(EGA_LIGHTRED); delta=(topx-botx)/10; for(i=0,accum=botx;i<MAXX;i+=(MAXX+1)/10,accum+=delta){ /* coloca os 10 1's marcadores */ sprintf(num,"%-4.2G",accum); /* justifica a esquerda */ if(i!=(MAXX+1)/2){ line(i,MAXY/2-MME,i,MAXY/2+MME); if(i!=0){outtextxy(i,MAXY/2+MME+th,num);} else{outtextxy(i,MAXY/2+MME+th,num);} } else{outtextxy(i+tw/2,MAXY/2+MME+th,num); /* para evitar superposiao c/ vertical */ line(i-1,MAXY/2-MME,i-1,MAXY/2+MME);

126

} } line(MAXX,MAXY/2-MME,MAXX,MAXY/2+MME); /* coloca o 11 ltimo marcador */ sprintf(num,"%4.2G",accum); outtextxy(MAXX-textwidth(num),MAXY/2+MME+th,num); getch(); cleardevice(); closegraph(); }

22) REFERNCIAS BIBLIOGRAFICAS: [1] - Turbo C - Guia do Usurio Herbert Schildt Ed. Mc Graw Hill [2] - Turbo C Avanado Herbert Schildt Ed. Mc Graw Hill [3] - Dominando o Turbo C Stan Kelly-Bootle Ed. Cincia Moderna [4] - MSDOS Avanado - Guia Microsoft p/ programao em C Ray Duncan Makron Books

127

Você também pode gostar