Portál AbcLinuxu, 8. května 2025 00:15

Dotaz: scanf pro čtení dat z příkazového řádku

4.9.2018 10:36 pat
scanf pro čtení dat z příkazového řádku
Přečteno: 1886×
Odpovědět | Admin
Píši program, kterému se budou vstupní data předávat z příkazového řádku - stdin. Chci se zeptat: Pro čtení několika sekvencí čísel - je dobré používat příkazy jako scanf nebo raději nějaké vyšší funkce co čtou celý řádek a ten se pak naparsuje pomocí split.
Nástroje: Začni sledovat (0) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

4.9.2018 10:56 Aleš Kapica | skóre: 52 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Odpovědět | | Sbalit | Link | Blokovat | Admin
V čem to píšeš?
4.9.2018 11:48 pat
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Záleží v čem to po mně chtějí. Jednou python jindy java, C++, jednou to byl Perl.
4.9.2018 12:30 pat
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Jinak momentálně je to Java, ale zajímá mě to obecně.
4.9.2018 14:58 rastos | skóre: 63 | blog: rastos
Rozbalit Rozbalit vše There is no silver bullet
Kde začať ..

Ak máš riadok a na ňom 10 údajov, ktoré potrebuješ a urobíš scanf(), tak tomu scanf()-u musíš dať správny format string, v ktorom bude 10 konverzí. Ak ale sa na vstup dostane riadok s 25 údajmi, tak split budeš pre 15 údajov robiť zbytočne. Ak tam bude tých údajov len 7, tak to musíš správne ošetriť.

Java nemá scanf. Ale má Scanner. Scanner-u je ale v princípe jedno, či si ešte na tom istom riadku alebo už nie. Vlastne scanf-u tiež.

Ak potrebuješ parsovať údaje v do iných typov, než java ponúka ... napr. bezznamienkové typy, ...

Ak potrebuješ parsovať napr. čísla v číselnej sústave inej ako 16/10/8, tak Scanner.useRadix() príde vhod.

Ak máš projekt v jazyku X tak ho nebudeš predsa znásilňovať a robiť samostaný modul pre parsovanie vztupu v jazyku Y.

Osobne si myslím, že v C to bude malinko rýchlejšie, ale v Jave to bude robustnejšie.
4.9.2018 15:40 pat
Rozbalit Rozbalit vše Re: There is no silver bullet
Scanner-u je ale v princípe jedno, či si ešte na tom istom riadku alebo už nie.

Tohle by mohlo vadit.
4.9.2018 12:39 x14
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Odpovědět | | Sbalit | Link | Blokovat | Admin
Pokud máš možnost čehokoliv vyššího než scanf, tak jdi do toho. Funkce scanf je o hubu. Stačí předhodit špatný typ a už se to veze.
4.9.2018 14:25 pat
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Funkce scanf je o hubu. Stačí předhodit špatný typ a už se to veze.

To se dá ošetřit výjimkou nebo kontrolou návratové hodnoty, tam bych problém neviděl.
4.9.2018 16:13 x14
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
V C++ výjimkou? Kontrolou návratové hodnoty? Tak určitě :-) :-) :-)
4.9.2018 20:01 Radovan
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Jako že to nejde nebo že to neumíš?
5.9.2018 01:16 Salam
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Viz dokumentace - scanf neháže výjimky. A když ho budeš nutit zapsat 32 bitů do bajtu, tak to tam klidně zapíše - nemá jak zjistit, co je to za ukazatel (a jestli to vůbec je ukazatel). Stačí omylem prohodit %d s %s a je všechno v háji.

No a kontrola, kolik parametrů načetl (když už je narušená okolní paměť) je jen k smíchu.
5.9.2018 06:27 Radovan
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
To je ale problém toho debila, co se považuje za programátora, a používá scanf k načítání řetězců ;-)
5.9.2018 09:14 rastos | skóre: 63 | blog: rastos
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
A když ho budeš nutit zapsat 32 bitů do bajtu, tak to tam klidně zapíše

??
$ gcc -Wall -c scanftest.c 
scanftest.c: In function ‘main’:
scanftest.c:7:10: warning: format ‘%d’ expects argument of type ‘int *’, but argument 2 has type ‘unsigned char *’ [-Wformat=]
  scanf("%d",&c);
         ~^  ~~
         %hhd
No keď budeš ignorovať warningy, tak sa nečuduj.

Inými slovami: vieš ukázať správne napísané volanie scanf(), pri ktorom sa nedozvieš, že došlo k chybe, alebo že sa zapisuje na nesprávne miesto v pamäti?
5.9.2018 12:52 x14
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Proč ty otazníky? Zapíše to tam, co by nezapsal? Nemá vůbec žádnou možnost, jak zjistit problém. Kdo někdy implementoval "variable argument" funkci, ví, o co jde.
To že některé dnešní překladače ad hoc kontrolují i správnost volání funkcí typu scanf nic neznamená. Dříve to nedělaly! (Nápověda: nejde o kontrolu syntaxe jazyka a je zapotřebí si uvědomit, že všechno zkontrolovat nikdy nemůže!).
A ta řečnická otázka je zvláštní.
Výše píšu, že použití funkce scanf může být problém, když se přehodí špatný typ.
To potřebuje hodně pomazanou hlavu napsat, že když se typ nesplete, že to bude fungovat správně :-) :-) :-)
No a o tom, že se zapisuje na špatné místo v paměti, se člověk dozvědět může a nemusí...
Ale to programátor s praxí v C/C++ dobře ví :-)
4.9.2018 14:49 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Odpovědět | | Sbalit | Link | Blokovat | Admin
Obvykle je praktičtější načíst celý řádek a ten následně rozparsovat dle potřeby.
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
5.9.2018 13:31 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Odpovědět | | Sbalit | Link | Blokovat | Admin
kterému se budou vstupní data předávat z příkazového řádku - stdin

Jen pro pořádek: příkazový řádek a stdin jsou dvě různé věci.

5.9.2018 18:56 sad
Rozbalit Rozbalit vše Re: scanf pro čtení dat z příkazového řádku
Odpovědět | | Sbalit | Link | Blokovat | Admin
Mohl bys například načítat jednotlivé řádky pomocí funkce fgets, řádek převedeš na číslo a číslo uložíš do dynamicky alokovaného pole. Načítání řádků ukončíš CTRL+D.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <math.h>

typedef struct Numbers Numbers;

struct Numbers {
	size_t n;
	size_t cap;
	double *nums;
};

enum {
	MAXLINE = 100,
	INIT = 1,
	GROW = 2
};

void addnumber(double num, Numbers *p);
int estrtod(char *line, double *num);

int main()
{	
	char line[MAXLINE];
	double number;
	Numbers ns;
	int i;
		
	ns.nums = NULL;	
	
	while (fgets(line, MAXLINE, stdin) != NULL) {
		line[strlen(line)-1] = '\0';
		if (estrtod(line, &number))
			addnumber(number, &ns);
	}	
	
	printf("\n");
	
	for (i = 0; i < ns.n; i++)
		printf("%0.2f\n", ns.nums[i]);	
	
	free(ns.nums);
	
	return 0;
}

void addnumber(double num, Numbers *p)
{
	double *t;
	
	if (p->nums == NULL) {
		p->nums = malloc(INIT * sizeof(double));		
		p->n = 0;
		p->cap = INIT; 
	} else if (p->n == p->cap) {
		t = realloc(p->nums, GROW * p->cap * sizeof(double));
		p->cap *= GROW; 
		p->nums = t;
	}
	p->nums[p->n] = num;
	p->n++;		
}

int estrtod(char *line, double *num)
{
	char *endptr;
	
	if (*line == '\0')
		return 0;
	
	errno = 0;
	
	*num = strtod(line, &endptr);
	
	if (errno == ERANGE || *endptr != '\0') {
		printf("error number: %s\n", line);
		return 0;
	}
		
	return 1;
}
58   
45.236
abc666
error number: abc666
22xyz
error number: 22xyz
22222222222222222222222222222
400

58.00
45.24
22222222222222223739180810240.00
400.00
Bohužel mi strtod nenastavuje errno na ERANGE při zadání příliš vysokého čísla, přičemž strtol mi funguje normálně. Možná chyba v linuxu, možná mezi počítačem a židlí, nevím.

Tohle je jen takový nástřel, ještě by bylo vhodné ošetřit chyby u malloc a realloc, mít na výpis čísel vlastní funkci, rozdělit vše do souborů atd.

Psát tohle v C je docela nepraktické.

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.