Torna indietro   Hardware Upgrade Forum > Software > Programmazione

iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile
iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile
C'è tanta sostanza nel nuovo smartphone della Mela dedicato ai creator digitali. Nuovo telaio in alluminio, sistema di raffreddamento vapor chamber e tre fotocamere da 48 megapixel: non è un semplice smartphone, ma uno studio di produzione digitale on-the-go
Intel Panther Lake: i processori per i notebook del 2026
Intel Panther Lake: i processori per i notebook del 2026
Panther Lake è il nome in codice della prossima generazione di processori Intel Core Ultra, che vedremo al debutto da inizio 2026 nei notebook e nei sistemi desktop più compatti. Nuovi core, nuove GPU e soprattutto una struttura a tile che vede per la prima volta l'utilizzo della tecnologia produttiva Intel 18A: tanta potenza in più, ma senza perdere in efficienza
Intel Xeon 6+: è tempo di Clearwater Forest
Intel Xeon 6+: è tempo di Clearwater Forest
Intel ha annunciato la prossima generazione di processori Xeon dotati di E-Core, quelli per la massima efficienza energetica e densità di elaborazione. Grazie al processo produttivo Intel 18A, i core passano a un massimo di 288 per ogni socket, con aumento della potenza di calcolo e dell'efficienza complessiva.
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 24-05-2007, 15:31   #1
k_mishima
Member
 
Iscritto dal: Jan 2007
Messaggi: 173
[C] Lettura caratteri dinamica

Purtroppo non riesco a capire cosa c'è che non va in questo prog, potreste correggermelo? Grazie.

Il problema lo da qui: memcpy(*a+n,b,sizeof(char))
WARNING: passing arg 2 of `memcpy' makes pointer from integer without a cast


Codice:
Scrivere una function c che legge i singoli caratteri e crea la stringa
che li contiene usando allocazione dinamica e le funzioni memcpy(...) e memmove (...)
Codice:
#include <stdio.h>
#include <stdlib.h>
#define N 50
void lettura_alloc_dinamica (char **a);
main()
{
      char *a;
      a=(char *)malloc(N*sizeof(char));
      lettura_alloc_dinamica(&a);
      printf("%s\n",a);
      free(a);
      system("pause");
      }
      
void lettura_alloc_dinamica (char **a)  
{    
     char b;
     short n=0;
     do
      {
      b=getchar();   
      memcpy(*a+n,b,sizeof(char));
      fflush(stdin);
      n+=1;
      }
      while (n!=N && b!=EOF);
      (*a)[n]=(char)NULL;
}
k_mishima è offline   Rispondi citando il messaggio o parte di esso
Old 25-05-2007, 09:04   #2
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Hai sbagliato l'approccio...quello che ti si chiedeva era evidentemente eliminare l'allocazione preventiva di spazio...quindi devi ogni volta che c'è un nuovo carattere devi usare realloc per allargare lo spazio contenuto nella stringa di un carattere...e poi inserisci il carattere letto...
Non usare fflush(stdin) perché non ha un comportamento definito dallo standard...ed inoltre non serve...

Non è detto che, nonostante tu sia leggendo carattere per carattere, tu debba leggere soltanto il primo carattere per ogni volta che viene premuto l'invio in input (questo stai cercando di fare, visto l'uso di fflush)...invece puoi permettere in questo modo di tutti i caratteri immessi fino all'invio, facnedo appunto terminare la lettura di caratteri quando trovi '\n' e non EOF...

Tra l'altro per come hai scritto te il codice, è errato anche l'uso che fai del doppio puntatore...visto che il puntatore ad a non viene modificato dopo che viene passato alla funzione...

Ultima modifica di cionci : 25-05-2007 alle 09:06.
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2007, 20:22   #3
k_mishima
Member
 
Iscritto dal: Jan 2007
Messaggi: 173
Quote:
Originariamente inviato da cionci
Hai sbagliato l'approccio...quello che ti si chiedeva era evidentemente eliminare l'allocazione preventiva di spazio...quindi devi ogni volta che c'è un nuovo carattere devi usare realloc per allargare lo spazio contenuto nella stringa di un carattere...e poi inserisci il carattere letto...
Non usare fflush(stdin) perché non ha un comportamento definito dallo standard...ed inoltre non serve...
Ho usato realloc, ma è sbagliato mi sa, cosi' n vale 1 al primo realloc e dovrebbe valere 2, ma se inizializzo n a 1 all'inizio allora non va piu' bene l'istruzione a[n]=getchar(); perchè farebbe a[1]=getchar();

Codice:
#include <stdio.h>
#include <stdlib.h>
void lettura_alloc_dinamica (char *a);
main()
{
      char *a;
      a=(char *)malloc(2*sizeof(char));
      puts("Digita i caratteri che compongono una parola");
      lettura_alloc_dinamica(a);
      puts("Parola intera");
      printf("%s\n",a);
      free(a);
      system("pause");
      }
      
void lettura_alloc_dinamica (char *a)  
{    
     short n=0;

     do
      {
      a[n]=getchar();
      n+=1;   
      a=(char *)realloc(a,n*sizeof(char));
      }
      while (a[n-1]!=EOF);
      a[n]='\0';
      }
Quote:
Non è detto che, nonostante tu sia leggendo carattere per carattere, tu debba leggere soltanto il primo carattere per ogni volta che viene premuto l'invio in input (questo stai cercando di fare, visto l'uso di fflush)...invece puoi permettere in questo modo di tutti i caratteri immessi fino all'invio, facnedo appunto terminare la lettura di caratteri quando trovi '\n' e non EOF...
Mah, io sto litigando con ogni traccia di questi maledetti esercizi che devo fare per l'esame, sono tutte poco chiare, nel dubbio cmq lascerei 1 carattere per volta come ho fatto ora.

Quote:
Originariamente inviato da cionci
Tra l'altro per come hai scritto te il codice, è errato anche l'uso che fai del doppio puntatore...visto che il puntatore ad a non viene modificato dopo che viene passato alla funzione...
Il passaggio di queste variabili dinamiche alla function non l'ho proprio capito perchè gli esercizi di esempio che ho sono tutti main, ma poi mi si richiedono le funcion senza averne mai vista 1 -.-
Ho copiazzato un po' il passaggio fatto in quell'altro esercizio sulla strcat.



Comunque l'output attuale è la stringa scritta tutta in colonna, cosi'

a
r
r
a
y


Non ho capito proprio perchè, sai come risolvere? grazie 1000 come sempre
k_mishima è offline   Rispondi citando il messaggio o parte di esso
Old 27-05-2007, 06:46   #4
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Te ne trovi una su ogni riga perchè getchar legge anche \n...
Ritornerei al passaggio con il doppio puntatore perchè realloc potrebbe ritornare anche una locazione diversa da quella inziale.
Con la prima malloc alloca spazio per un solo carattere...

Attento che la realloc deve allocare n+1 caratteri: immaginati di essere alla prima iterazione con n = 0, in tal caso realloc allocherebbe un solo carattere (n = 1)...quindi all'iterazione stai accedendo sempre in overflow alla stringa (cioè nella cella successiva all'ultima allocata).

Inoltre se il ciclo finisce quando a[n-1]==EOF, EOF ti conviene toglierlo dal buffer, basta sovrascriverlo con \0
Quindi a[n-1]='\0' e non a[n]='\0'
A questo punto però ti resta un carattere in più allocato rispetto al numero di caratteri che hai, quindi evita di fare realloc se l'ultimo carattere letto è uguale a EOF.
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 27-05-2007, 12:59   #5
k_mishima
Member
 
Iscritto dal: Jan 2007
Messaggi: 173
E' venuto

Codice:
#include <stdio.h>
#include <stdlib.h>
void lettura_alloc_dinamica (char *a);
main()
{
      char *a;
      a=(char *)malloc(2*sizeof(char));
      puts("Digita i caratteri che compongono una parola");
      lettura_alloc_dinamica(a);
      puts("Parola intera");
      printf("%s\n",a);
      free(a);
      system("pause");
      }
      
void lettura_alloc_dinamica (char *a)  
{    
     short n=0;

     do
      {
      a[n]=getchar();
      fflush(stdin);
      n+=1;   
      a=(char *)realloc(a,sizeof(char));
      }
      while (a[n-1]!=EOF);
      a[n-1]='\0';
      }
Col primo malloc avevo allocato 2 spazi, in modo che mi trovero' sempre 1 spazio in piu' da usare alla fine per lo \0

a=(char *)realloc(a,n*sizeof(char)); ho tolto la n perchè allocavo tantissimo spazio in piu' inutilmente vero? Ne basta solo 1 in piu' a ciclo.

Per togliere lo \n ho rimesso fflush(stdin), non so tu dici sempre che non è standard ma è l'unico modo che conosco per pulire il buffer.
Quindi a[n-1]='\0' e non a[n]='\0' corretto

Infine ho lasciato con 1 puntatore normale, mi spieghi perchè a volte se ne usano 2 (l'altro esercizio) e qui 1 solo?
k_mishima è offline   Rispondi citando il messaggio o parte di esso
Old 27-05-2007, 13:21   #6
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
a=(char *)realloc(a,sizeof(char));
E' errato...con realloc non devi tenere conto di quello che hai già allocato precedentemente...quindi serve n (anzi n+1 in questo caso)

Per il doppio puntatore:

Codice:
void f(int **a)
{
    *a = (int *)malloc(sizeof(int));
}

...
int *a = NULL:
f(&a);
<-----qui a non punta più a NULL, ma punta allo spazio allocato con malloc

Codice:
void f2(int *a)
{
    a = (int *)malloc(sizeof(int));
}

...
int *a = NULL:
f(a);
<-----qui a punta sempre a NULL
In sostanza si usa il doppio puntatore quando si prevede che l'indirizzo a cui punta il puntatore venga cambiato dalla funzione chiamata

In questo caso realloc potrebbe cambiare l'indirizzo puntato...

Per pulire il buffer puoi tranquillamente fare così:
Codice:
int c;
do
{
   c = getchar();
}
while( c != '\n' && c != EOF);
fflush su stdin non si può fare perché lo standard prevede che si possa fare solo sui buffer di output...mentre il comportamento sui buffer di input non è specificato.

Per la prima allocazione non devi tenere conto di \0 in questo caso perché se anche leggessi un solo carattere seguito da EOF faresti comunque almeno una realloc che porterebbe la stringa a 2 caratteri

Ultima modifica di cionci : 29-05-2007 alle 08:04.
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2007, 02:18   #7
k_mishima
Member
 
Iscritto dal: Jan 2007
Messaggi: 173
ok, allora ho sostituito con il doppio puntatore, ora l'esercizio è completo

grazie ancora, un altro esercizio è completo
k_mishima è offline   Rispondi citando il messaggio o parte di esso
Old 26-11-2014, 19:16   #8
caramelleamare
Member
 
Iscritto dal: Jan 2010
Messaggi: 166
Salve, riapro questo vecchio post perché voglio fare qualcosa di simile. Mettiamo che abbia una stringa in input formata da N parole di M lettere. Vorrei creare un array dinamico che riallochi memoria per ogni diversa parola ogni volta che incontro uno spazio nella stringa, e allo stesso tempo voglio riallocare spazio dinamicamente per ogni singolo char che leggo.
Quindi alloco char per char, e quando scatta la parola alloco per quella parola, cioè zero spreco di spazio e nessun limite di inserimento.
Non mi è richiesto ma voglio capire bene come funzionano i puntatori a puntatori, malloc e la realloc.

Io ho immaginato un punt di punt con un doppio ciclo for: quello interno che rialloca i char e quello esterno le stringhe(parola), ma anche fosse giusta l'idea non riesco a scriverlo. Una guida?
caramelleamare è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile iPhone 17 Pro: più di uno smartphone. &Eg...
Intel Panther Lake: i processori per i notebook del 2026 Intel Panther Lake: i processori per i notebook ...
Intel Xeon 6+: è tempo di Clearwater Forest Intel Xeon 6+: è tempo di Clearwater Fore...
4K a 160Hz o Full HD a 320Hz? Titan Army P2712V, a un prezzo molto basso 4K a 160Hz o Full HD a 320Hz? Titan Army P2712V,...
Recensione Google Pixel Watch 4: basta sollevarlo e si ha Gemini sempre al polso Recensione Google Pixel Watch 4: basta sollevarl...
Il nuovo Snapdragon 8 Elite Gen 5 'for G...
2 Smart TV 4K in super offerta su Amazon...
AMD e Intel celebrano un anno di collabo...
NVIDIA consegna il suo mini supercompute...
La concorrenza cinese si fa sentire: il ...
4 accessori auto da non perdere su Amazo...
Era finito, eccolo ancora: torna il supe...
Game Pass: tutte le novità di nov...
Xiaomi 15T e 15T Pro con fotocamere by L...
3 portatili un super sconto: 355€ con 16...
Il nuovo iPhone pieghevole costerà...
MediaTek Dimensity 9500 sfida lo Snapdra...
Nuovo storico lancio per il razzo spazia...
Elgato Embrace: una sedia ergonomica pro...
Brad Pitt torna in pista: F1 – Il Film a...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 09:10.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
Served by www3v