Skip to content

Aula extra sobre alguns tricks e especificadores de formato na linguagem C

Notifications You must be signed in to change notification settings

andrefakhoury/especificadores-de-formato-c

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

13 Commits
 
 
 
 
 
 
 
 

Repository files navigation

Especificadores de formato em C... e algumas coisas a mais

Nessa aula, falarei principalmente sobre especificadores de formato para as funções scanf e printf da linguagem C - e também vou citar mais algumas funções ou truques relacionados a isso, e que acho úteis.

Formatando números inteiros

Em C, podemos declarar um int da seguinte forma:

int dec = 10; // dec = 10
int oct = 010; // oct = 8
int hex = 0x10; // hex = 16

Ou seja, se colocarmos um valor sem 0 a esquerda, ele será convertido para a base 10. Se colocarmos 0 antes do número, ele será convertido para a base 8. Se colocarmos 0x antes do número, ele será convertido para a base 16.

Também existe uma extensão do GCC que faz a mesma coisa para binário, e podemos escrever o seguinte código:

int bin = 0b10; // bin = 2

Caracteres especiais \n e \r

Os dois caracteres servem para (normalmente) a mesma coisa - pular/quebrar uma linha. Porém, normalmente, ouvimos falar apenas do \n, e deixamos o \r de lado - o que não é muito bom.

Na verdade, a existência de \n ou \r depende bastante do sistema que está sendo utilizado.

q?

É meio difícil explicar isso de forma completa e simples, então vou copiar parafrasear parte dessa resposta do stack overflow.

  • em sistema Unix (e em baseados nele), o \n é o caractere de fim de linha, e o \r não representa nada em especial;

  • como consequência, em C e na maioria das linguagens, \n é o padrão de fim de linha;

  • em sistemas Mac antigos (pré-OSX), \r era o end-of-line;

  • no Windows (e vários outros SO's antigos), o código de end-of-line é \r\n, nessa ordem.

  • no Windows, com o \r é possível retornar ao início da linha atual - ou seja, printf("abcdef\r123"); imprime na tela 123def.

Só uso linux, preciso me preocupar com \r?

Eu até diria que não... mas sim heheh

Digamos que estou utilizando um grandioso sistema de correção automática de códigos (*cof* run.codes *cof*) - nele, as vezes, a entrada (ou até mesmo a saída esperada) pode conter um \r. Por isso, é bom tratar ele juntamente do \n, para evitar problemas.

Principais funções para scan e print em C

OBS: Podemos ver a documentação das funções no linux escrevendo man <nome_da_função> no terminal - porém, o man serve para todas as funções do linux, e as vezes as funções tem o mesmo nome em C e em bash, por exemplo. Aí, pode-se escrever man 3 <nome_da_função> no terminal, que pega a seção 3 do manual, que possui as funções do C.

scanf

Lê dados da entrada padrão (stdin).

int n;
scanf("%d", &n);

Retorno: se bem sucedido, retorna o número de itens corretamente lidos. Em caso de final de entrada, retorna EOF (end-of-file).

printf

Escreve dados na saída padrão (stdout).

int n = 10;
printf("%d", n); // imprime 10

Retorno: se bem sucedido, retorna o número de caracteres corretamente impressos. Se algum erro ocorreu, retorna um valor negativo.

stdin e stdout

Basicamente, são os streams de entrada e saída padrão de dados.

Ou seja, se eu quero executar um programa e ler/escrever dados diretamente no terminal, utilizam-se stdin e stdout.

sscanf

Lê dados de uma string - basicamente um scanf que utiliza uma string, e não o terminal.

char entrada[] = "10 20";
int n, k;

sscanf(entrada, "%d%d", &n, &k);
// n = 10, k = 20

sprintf

Escreve dados em uma string - basicamente um printf que 'imprime' em uma string, e não no terminal.

char saida[1024];
int n = 10;
sprintf(saida, "%d", n);
// saida = 10

getchar

Lê e retorna um caractere do stdin.

char c = getchar();

puts

Imprime uma string no stdout - sem opção de formatação.

char s[] = "oi";
puts(s);

Outras

Existem ainda outras funções de leitura e escrita, algumas específicas para algumas coisas e tal. Porém, algumas não são tão úteis no geral, enquanto outras necessitam de um certo conhecimento de arquivos em C para serem entendidas, o que não é o objetivo dessa aula. Porém, vou citar algumas apenas para desencargo de consciência: fgets, gets, fgetc, fputc, fscanf, fprintf, fread, ...

Buffer

Basicamente, buffer é uma região de memória utilizada para armazenar dados temporariamente. Portanto, quando se escreve algo em um teclado, esses dados primeiramente vão para um buffer de entrada, e depois são utilizados por alguém (por exemplo, pelo scanf do seu programa em C).

Buffer de Entrada

O buffer de entrada, como o nome sugere, armazena os dados de entrada (vindos, por exemplo, do teclado) temporariamente, para depois ser utilizado.

Buffer de Saída

O buffer de saída, como o nome sugere, armazena os dados de saída temporariamente, para depois ser utilizado (por exemplo, sendo impresso na tela do terminal).

fflush

Em C, é possível forçar a limpeza de certa stream (por exemplo, stdint ou stdout).

No caso da stream ser de saída (como stdout), o fflush força a escrita de todos os dados do buffer.

No caso da stream ser de entrada (como stdin), o fflush descarta todos os dados do buffer de entrada que foram buscados mas ainda não consumidos pela função.

Por exemplo, os seguintes trechos de código executam de maneira distinta:

for (int i = 1; i <= 5; i++) {
	printf("%d", i);
	sleep(1);
}
// espera tudo, imprime tudo junto

for (int i = 1; i <= 5; i++) {
	printf("%d", i);
	fflush(stdout);
	sleep(1);
}
// imprime e espera, imprime e espera...

for (int i = 1; i <= 5; i++) {
	printf("%d\n", i);
	sleep(1);
}
// imprime e espera, imprime e espera...

Especificadores de Formato

Vou falar rapidão aqui sobre especificadores de formato da linguagem C, utilizados, principalmente, nas funções de scanf e printf.

Bom começamos do básico. Podemos formular essa pequena tabela, que por si só já consegue nos dar uma boa noção das coisas.

Format specifier Description Supported Data Types
%c Character char unsigned char
%d Signed Integer short unsigned short int long
%e or %E Scientific notation of float values float double
%f Floating point float
%g or %G Similar as %e or %E float double
%hi Signed Integer (Short) short
%hu Unsigned Integer (Short) unsigned short
%i Signed Integer short unsigned short int long
%l or %ld or %li Signed integer long
%lf Floating point double
%Lf Floating point long double
%lu Unsigned integer unsigned int unsigned long
%lli or %lld Signed integer long long
%llu Unsigned integer unsigned long long
%o Octal representation of Integer short unsigned short int unsigned int long
%p Address of pointer to void void* void *
%s String char *
%u Unsigned Integer unsigned int unsigned long
%x or %X Hexadecimal representation of Unsigned Integer short unsigned short int unsigned int long
%n Prints nothing
%% Prints % character

Alguns exemplos no scanf

Exemplos do scanf - valeu Braulio

Diferença entre %i e %d

Eles são a mesma coisa quando usados no output, como no printf.

printf("%d\n", 020); //Saida: 16
printf("%i\n", 020); //Saida: 16

Porém, quando utilizadas como especificadoras de input, como no scanf, as máscaras são diferentes. %d faz a leitura de um integer como um número decimal normal, porem o %i, além de ler um número decimal, também converte a entrada para octal ou hexadecimal, de acordo com os prefixos 0 (octal) e 0x (hexadecimal).

Ou seja, suponha a entrada 020 no código abaixo:

scanf("%d", &n); // n = 20
scanf("%i", &n); // n = 16

Espaçamento no print

Podemos formatar o print de um número de ponto flutuante (float) da maneira que quisermos - em relação à casas depois da vírgula e coisas assim.

Por exemplo, se eu quiser imprimir um número com 2 casas depois da vírgula do seguinte modo: printf("%.<numr. casas>f", <valor>).

float f = 3.141516;
printf("%.2f", f);
//saida: 3.14

O número de casas depois da vírgula também pode ser especificado por uma variável: printf("%.*f", <numr. casas>, <valor>)

printf("%.*f", 2, 3.141516);
//saida: 3.14

Também podemos definir o espaçamento que certa variável terá - por exemplo, se eu quero alinhar uma variável à direita de um bloco de n caracteres: printf("%<n>d");. Pode ser utilizado tanto em números inteiros quanto em outros tipos de variável.

printf("|%3d|\n", 10);     // saída: | 10|
printf("|%5.2f|\n", 3.14); // saída: | 3.14|

Também é possível alinhar à esquerda colocando-se um hífen antes:

printf("|%-3d|\n", 10);     // saída: |10 |
printf("|%-5.2f|\n", 3.14); // saída: |3.14 |
printf("|%-5s|\n", "oi");   // saída: |oi   |

Leitura de strings - meio complicado, mas bora lá

A leitura de strings e/ou caracteres podem ser atrapalhados por coisas simples, como espaços em branco. É por isso que é necessário saber qual a formatação da entrada que o seu programa vai receber - por exemplo, se existe quebra de linha entre os dados, se tem espaços em branco entre strings, etc.

Quando se utiliza um scanf("%c", &caractere);, o primeiro caractere que estiver no buffer de entrada será lido e atribuído à variável caractere - inclusive espaços, '\n' e afins.

Se um scanf(" ") for utilizado, todos os espaços em branco que estão antes de algo que não seja um espaço em branco serão ignorados. Ou seja, o combo scanf("%c ", &caractere); garante que, após a leitura de caractere, espaços extra serão ignorados.

Outra coisa interessante para se notar é que o especificador %s lê caracteres até que seja encontrado algum espaço em branco, como ' ', '\n' e afins - e esses caracteres continuam no buffer após a leitura.

Ou seja, caso seja fornecida a entrada string1 string2, e se deseja armazenar as duas palavras em strings distintas, deve-se ignorar o caractere de espaço ' ' do meio. Para fazer isso, utiliza-se o comando scanf("%s %s", str1, str2);.

Mas e se eu quiser ler uma string até determinado caractere específico? Por exemplo, eu desejo que a entrada anterior string1 string2 seja salva na mesma string. Para isso, pode-se utilizar a seguinte máscara para ler até o final da linha:

scanf("%[^\n\r]", str);

Sendo que o [^] representa que deve ser feita a leitura até que chegue em algum dos caracteres determinados - nesse caso, '\n' ou '\r'. Quando algum dos caracteres é encontrado, o scanf para a leitura.

Outro exemplo é se eu quiser ler uma string até que seja encontrado um caractere de $:

scanf("%[^$]", str);
// exemplo de entrada: abc def aaa$
// conteudo de str: abc def aaa

Porém, quando o scanf para a leitura, o buffer continua com o caractere delimitador. Ou seja, no exemplo anterior, o primeiro caractere presente no buffer após a leitura seria o '$'. Caso eu queira ignorar esse caractere, posso fazer o seguinte:

scanf("$");

Isso é útil quando se quer ler um conjunto de strings, e deseja-se que os caracteres delimitadores sejam ignorados. Por exemplo, se eu quero ler 2 strings que estão da seguinte forma:

string1$string2

O seguinte código estaria incorreto:

char str1[32], str2[32];
scanf("%[^$] %s", str1, str2);

Pois a str2 armazenaria a string $string2, sendo que o caractere '$' deveria ter sido ignorado. Para isso, deve-se fazer algo como:

char str1[32], str2[32];
scanf("%[^$]$%s", str1, str2);

Ignorar certa entrada

Podemos ignorar determinado dado da entrada a partir do scanf. Por exemplo, a entrada que temos possui 2 números inteiros - porém, só usaremos o primeiro. Podemos ler esse inteiro e nunca mais utilizá-lo, como nesse exemplo:

int numero, lixo;
scanf("%d%d", &numero, &lixo);
// nunca mais uso a variavel lixo

Porém eu posso ignorar essa entrada, sem precisar criar uma variável a mais:

int numero;
scanf("%d%*d", &numero);

Ao colocar o * antes do d, eu estou dizendo ao printf para ignorar um inteiro. Eu também posso fazer isso para outros tipos:

scanf("%*f %*d %*c"); // ignora um float, um int e um char

Isso também será útil em alguns trabalhos, quando quisermos ignorar alguns caracteres - por exemplo, os caracteres de \n e \r.

Tudo bem que, neste caso, poderia simplesmente fazer esse código:

scanf(" ");

Que descarta todos os espaços em branco, até que chegue em algo que não seja um espaço em branco (isso inclui espaço, tab, \n e \r).

Mas podemos ser um pouco mais específicos. Então, também é possível fazer o seguinte:

scanf("%*[\n\r]");

Com isso, eu consigo ignorar os caracteres de \n e \r.

Exemplos práticos

Horário

Digamos que a entrada do seu programa é um horário no formato hh:mm:ss, e você quer salvar a hora, minuto e segundo em 3 inteiros separados. Como podemos fazer isso utilizando apenas o scanf?

int h, m, s; // hora, minuto e segundo
scanf("%d:%d:%d", &h, &m, &s); //le os inteiros ate os `:`

Separados por um #

Vamos para outro exemplo agora: você receberá, no seu programa, duas palavras, separadas por um '#', e quer salvar cada palavra em um array de caracteres diferente. Por exemplo, considere a seguinte entrada:

primeira#segunda

Eu quero salvar "primeira" em um array, e "segunda" em outro. Também é possível fazer isso utilizando scanf, porém temos que tomar alguns cuidados.

Se seguirmos a mesma lógica do exercício anterior, teríamos o seguinte código:

char str1[1024], str2[1024];
scanf("%s#%s", str1, str2);

Porém isso não promove o resultado esperado - com o %s, ocorreria a leitura de todos os caracteres até algum delimitador de espaço ou quebra de linha (' ', '\n', '\r', ...). Portanto, a str1 receberia a entrada inteira (primeira#segunda), enquanto a str2 ficaria vazia, pois o trick de colocar um '#' no meio do scanf foi totalmente inútil. Portanto, teríamos que fazer o seguinte código:

char str1[1024], str2[1024];
scanf("%[^#]#%s", str1, str2);

O que essa máscara faz é o seguinte: ele vai lendo todos os caracteres até chegar no #. Quando chegar no #, ele irá descartar o # (perceba que tem um # jogado no meio do scanf, antes do segundo %s). Depois disso, ele vai ler a segunda string, até chegar em um caractere delimitador de espaço ou final de linha.

Isso pode parecer meio confuso no começo, mas vai ficar mais natural conforme se acostumar. Eu também recomendo testar várias combinações diferentes, e observar na prática como cada coisa funciona.

Mesma coisa, só que com mais coisa

Vamos agora para outro exemplo. Ele será bem parecido com o anterior, mas tem uma diferença: a entrada do seu programa será duas frases separadas por '#', e eu quero salvar cada frase em um array diferente. Ou seja, podem ter espaços entre as palavras da mesma frase, e eu quero considerar tudo isso para salvar nos meus arrays. Por exemplo, suponha a seguinte entrada: primeira string#segunda string. Veja (um exemplo de) o código para isso:

char str1[1024], str2[1024];
scanf("%[^#]#%[^\n\r] ", str1, str2);

A leitura da primeira string não muda nada - ele continuará lendo tudo até chegar em um #. Porém, para a segunda string, perceba a entrada da máscara [^\n\r]. O que isso faz é que serão lidos todos os caracteres até que se chegue em um caractere delimitador de quebra de linha (\n ou \r), ou que a entrada do seu programa tenha acabado (por exemplo, se nao tiver quebra de linha, porem o arquivo chegou no final).

Mostrar um telefone, e espaços se ele não existir

Um outro exemplo que talvez seja útil na sua vida (pelo menos no meu trabalho de arquivos foi heheh). Digamos que você uma string que consiste em um número de telefone, no formato (XX)XXXXX-XXXX. Você precisa imprimir o número inteiro se a string for não nula. Porém, se a string for nula (o primeiro caractere dela é o delimitador '\0'), precisa imprimir 15 espaços vazios (suponha que apenas existem os casos de string nula ou string completa, com todos os 14 caracteres originais). Tipo isso:

char tel1[15] = "(11)11111-1111";
char tel2[15] = "\0";

Para imprimir do jeito desejado, basta fazer o seguinte:

printf("|%-14s|\n", tel1); // esses | estão aí só pra destacar os espaços vazios no output
printf("|%-14s|\n", tel2);

Saída desse programa:

|(11)11111-1111|
|              |

Ler dados indefinidamente, até o final do arquivo

A entrada do programa consiste em vários números inteiros - não se sabe quantos, precisa ler até chegar ao final da entrada.

Para ler dados até o final do arquivo, utiliza-se o EOF - macro utilizada para dizer que o arquivo de leitura (normalmente stdin chegou ao fim). Como visto anteriormente, o scanf retorna esse EOF neste caso, então basta utilizá-lo em algum laço de repetição:

int n;
while (scanf("%d", &n) != EOF) {
	...
}

Referências

https://en.cppreference.com/w/cpp/language/types

https://www.geeksforgeeks.org/format-specifiers-in-c/

https://www.lix.polytechnique.fr/~liberti/public/computing/prog/c/C/FUNCTIONS/format.html

http://a4academics.com/tutorials/77-c-programming/690-c-format-specifiers

http://www.cplusplus.com/reference/cstdio/scanf/

https://wpollock.com/CPlus/PrintfRef.htm

http://www.mobidev.com/shalan/scanf_printf.htm

https://homepages.dcc.ufmg.br/~rodolfo/aedsi-2-10/printf_scanf/printfscanf.html

https://stackoverflow.com/questions/2611764/can-i-use-a-binary-literal-in-c-or-c

https://codeforwin.org/2015/05/list-of-all-format-specifiers-in-c-programming.html

https://stackoverflow.com/questions/1761051/difference-between-n-and-r

About

Aula extra sobre alguns tricks e especificadores de formato na linguagem C

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages