Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Lenovo LOQ 15i Gen 10 (15IRX10) alla prova: il notebook gaming 'budget' che non ti aspetti
Lenovo LOQ 15i Gen 10 (15IRX10) alla prova: il notebook gaming 'budget' che non ti aspetti
Il Lenovo LOQ 15i Gen 10 (15IRX10) offre prestazioni convincenti grazie al Core i7-13650HX e alla RTX 5060 Laptop a 100W, mantenendo un prezzo competitivo tra 1100 e 1300 euro. Costruzione solida, buon display e ampia espandibilità lo rendono una scelta equilibrata per chi cerca un notebook gaming accessibile ma moderno.
Due mesi di Battlefield 6: dalla campagna al battle royale, è l'FPS che stavamo aspettando
Due mesi di Battlefield 6: dalla campagna al battle royale, è l'FPS che stavamo aspettando
Abbiamo giocato a lungo a Battlefield 6, abbiamo provato tutte le modalità multiplayer, Redsec, e le numerose personalizzazioni. In sintesi, ci siamo concentrati su ogni aspetto del titolo per comprendere al meglio uno degli FPS più ambiziosi della storia dei videogiochi e, dopo quasi due mesi, abbiamo tirato le somme. In questo articolo, condividiamo con voi tutto ciò che è Battlefield 6, un gioco che, a nostro avviso, rappresenta esattamente ciò che questo genere attendeva da tempo
Antigravity A1: drone futuristico per riprese a 360° in 8K con qualche lacuna da colmare
Antigravity A1: drone futuristico per riprese a 360° in 8K con qualche lacuna da colmare
Abbiamo messo alla prova il drone Antigravity A1 capace di riprese in 8K a 360° che permette un reframe in post-produzione ad eliche ferme. Il concetto è molto valido, permette al pilota di concentrarsi sul volo e le manovre in tutta sicurezza e decidere con tutta tranquillità come gestire le riprese. La qualità dei video, tuttavia, ha bisogno di uno step in più per essere competitiva
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 24-05-2009, 16:52   #1
Ntropy
Senior Member
 
L'Avatar di Ntropy
 
Iscritto dal: Oct 2003
Messaggi: 2002
[C/C++] Problemi nell'allocazione di un'array si strutture

Salve a tutti, vorrei un aiuto per un piccolo problema.

Ho scritto un programma e vi riporto il codice:
Codice:
#include <cstdlib>
#include <iostream>
#define SLEN 256
#define ARRAYSIZE 5
#define NUMNODI 3
  
  struct clienti{
         bool flag;
         char filename[SLEN];
         double Xcoords[ARRAYSIZE];
         double Ycoords[ARRAYSIZE];        
  };

  
  
struct clienti* funzione(void);

int main(int argc, char *argv[])
{
  int i,j=0;
  struct clienti *ptr=NULL;

  ptr=funzione();
  
  if(ptr==NULL){
           printf("ptr null!\n");
           exit(-1);
  }
  
  for(j=0;j<NUMNODI;j++){
        
  printf("%s\n",ptr[j].filename);
    
  for( i=0;i<ARRAYSIZE;i++)
       printf("1 - %.2f\n",ptr[j].Xcoords[i]);
        
  for( i=0;i<ARRAYSIZE;i++)
       printf("2 - %.2f\n",ptr[j].Ycoords[i]);   
    
       }

  free(ptr);	
  return 0;
}

struct clienti* funzione(void){
   
   char buffer[SLEN];
   int i,j;
   int randm;
   struct clienti *nodo; 
   
   if (!(nodo=(struct clienti*)malloc(NUMNODI*sizeof(struct clienti)))){
     printf("Memoria non allocabile!\n");
     return NULL;
     }

     for(j=0;j<NUMNODI;j++){
                          
         printf("Inserisci il nome:");
         scanf("%s",buffer); 
         strcpy(nodo[j].filename,buffer);
         
         for( i=0;i<ARRAYSIZE;i++)
             nodo[j].Xcoords[i]=1+rand()%100;
   
         for( i=0;i<ARRAYSIZE;i++)
             nodo[j].Ycoords[i]=1+rand()%100;
                      
   }                
    return nodo;         

}
In pratica il main chiama "funzione" passandogli void e aspetta come valore di ritorno un puntatore alla struttura clienti.
Funzione dal canto suo crea un'array di stutture clienti, fa alcune operazioni e restituisce il puntatore all'array di strutture.

Il programma è compilato e funziona bene ma.........non dovrei usare l'operatore -> per riferirmi ai campi delle singole strutture piuttosto che il punto .
Quando ho provato con -> il compilatore mi sputa fuori che: base operand of `->' has non-pointer type `clienti' .

Qualche suggerimento?
__________________
Trattative sul mercatino del forum:tante e mai un problema! Controlla pure i miei vecchi post
Ntropy è offline   Rispondi citando il messaggio o parte di esso
Old 25-05-2009, 23:49   #2
Ntropy
Senior Member
 
L'Avatar di Ntropy
 
Iscritto dal: Oct 2003
Messaggi: 2002
nessun aiutino ?
__________________
Trattative sul mercatino del forum:tante e mai un problema! Controlla pure i miei vecchi post
Ntropy è offline   Rispondi citando il messaggio o parte di esso
Old 25-05-2009, 23:57   #3
BrutPitt
Senior Member
 
L'Avatar di BrutPitt
 
Iscritto dal: Mar 2009
Città: Bologna
Messaggi: 1174
Be' la risposta te l'ha data il compilatore.

L'indicizzazione nodo[j] fa si' che tu ti riferisca al j-iesimo elemento dell'array di strutture... mentre l'operatore "freccia" si usa col puntatore.

Insomma o usi:
(nodo+j)->Xcoords[i]

o usi:
nodo[j].Xcoords[i]

... o ancora:
(*(nodo+j)).Xcoords[i]

Cioe' si usa "." quando si accede direttamente a variabili di tipo struttura... mentre l'operatore "->" si usa quando si accede tramite l'indirizzo, il puntatore alla struttura.

Ultima modifica di BrutPitt : 26-05-2009 alle 01:59.
BrutPitt è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2009, 09:17   #4
Ntropy
Senior Member
 
L'Avatar di Ntropy
 
Iscritto dal: Oct 2003
Messaggi: 2002
Quote:
Originariamente inviato da BrutPitt Guarda i messaggi
Be' la risposta te l'ha data il compilatore.

L'indicizzazione nodo[j] fa si' che tu ti riferisca al j-iesimo elemento dell'array di strutture... mentre l'operatore "freccia" si usa col puntatore.

Insomma o usi:
(nodo+j)->Xcoords[i]

o usi:
nodo[j].Xcoords[i]

... o ancora:
(*(nodo+j)).Xcoords[i]

Cioe' si usa "." quando si accede direttamente a variabili di tipo struttura... mentre l'operatore "->" si usa quando si accede tramite l'indirizzo, il puntatore alla struttura.
Si credo di aver capito, in pratica accedevo alla variabile senza usare il puntatore ma "toccando" direttamente l'array diciamo così. In pratica per fare la stessa cosa con il main devo usare: (ptr+j)->

grazie di tutto.
__________________
Trattative sul mercatino del forum:tante e mai un problema! Controlla pure i miei vecchi post
Ntropy è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2009, 18:37   #5
BrutPitt
Senior Member
 
L'Avatar di BrutPitt
 
Iscritto dal: Mar 2009
Città: Bologna
Messaggi: 1174
Quote:
Originariamente inviato da Ntropy Guarda i messaggi
Si credo di aver capito, in pratica accedevo alla variabile senza usare il puntatore ma "toccando" direttamente l'array diciamo così. In pratica per fare la stessa cosa con il main devo usare: (ptr+j)->

grazie di tutto.
Si, anche se i metodi ptr[j]. o (ptr+j)-> sono equivalenti. Forse il primo, quello usato da te, gode di maggiore leggibilita' a livello di codice... ma queste sono solo considerazione di forma e assolutamente soggettive.

La potenza dei "puntatori" risiede in altro... ti modifico la tua funzione:

Codice:
struct clienti* funzione(void){
   
   char buffer[SLEN];
   int i,j;
   int randm;
   struct clienti *nodo,*ptrNodo; //copia di nodo.
   
   if (!(nodo=(struct clienti*)malloc(NUMNODI*sizeof(struct clienti)))){
     printf("Memoria non allocabile!\n");
     return NULL;
     }

    ptrNodo=nodo; //copio l'indirizzo di memoria appena allocato
     for(j=0;j<NUMNODI;j++){
                          
         printf("Inserisci il nome:");
         scanf("%s",buffer); 
         strcpy(ptrNodo->filename,buffer); //accedo tramite puntatore
         
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Xcoords[i]=1+rand()%100; //accedo tramite puntatore
   
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Ycoords[i]=1+rand()%100; //accedo tramite puntatore

        ptrNodo++; //incremento la copia del puntatore
                      
   }                
    return nodo;         

}
A parte la leggibilita', soggettiva, vene ottimizzata anche l'esecuzione:
si evita che ad ogni accesso alla struttura il programma calcoli indirizzo+spiazzamento... (nodo+j) o nodo[j]... incrementando invece direttamente, una sola volta a ciclo, l'indirizzo contenuto in ptrNodo.

Due cosniderazioni:
1) E' necessario fare una copia dell'indirizzo e non agire direttamente su nodo, perche' alla fine del ciclo ptrNodo puntera' all'ultimo elemento, mentre bisogna far "ritornare" l'indirizzo al primo elemento dell'array di strutture.
2) Il codice sarebbe ulteriormente ottimizzabile, ma lasciandolo piu' vicino possibile all'originale si evidenziano meglio le differenze.

Ultima modifica di BrutPitt : 26-05-2009 alle 18:41.
BrutPitt è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2009, 18:54   #6
Ntropy
Senior Member
 
L'Avatar di Ntropy
 
Iscritto dal: Oct 2003
Messaggi: 2002
Quote:
Originariamente inviato da BrutPitt Guarda i messaggi
Si, anche se i metodi ptr[j]. o (ptr+j)-> sono equivalenti. Forse il primo, quello usato da te, gode di maggiore leggibilita' a livello di codice... ma queste sono solo considerazione di forma e assolutamente soggettive.

La potenza dei "puntatori" risiede in altro... ti modifico la tua funzione:

Codice:
struct clienti* funzione(void){
   
   char buffer[SLEN];
   int i,j;
   int randm;
   struct clienti *nodo,*ptrNodo; //copia di nodo.
   
   if (!(nodo=(struct clienti*)malloc(NUMNODI*sizeof(struct clienti)))){
     printf("Memoria non allocabile!\n");
     return NULL;
     }

    ptrNodo=nodo; //copio l'indirizzo di memoria appena allocato
     for(j=0;j<NUMNODI;j++){
                          
         printf("Inserisci il nome:");
         scanf("%s",buffer); 
         strcpy(ptrNodo->filename,buffer); //accedo tramite puntatore
         
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Xcoords[i]=1+rand()%100; //accedo tramite puntatore
   
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Ycoords[i]=1+rand()%100; //accedo tramite puntatore

        ptrNodo++; //incremento la copia del puntatore
                      
   }                
    return nodo;         

}
A parte la leggibilita', soggettiva, vene ottimizzata anche l'esecuzione:
si evita che ad ogni accesso alla struttura il programma calcoli indirizzo+spiazzamento... (nodo+j) o nodo[j]... incrementando invece direttamente, una sola volta a ciclo, l'indirizzo contenuto in ptrNodo.

Due cosniderazioni:
1) E' necessario fare una copia dell'indirizzo e non agire direttamente su nodo, perche' alla fine del ciclo ptrNodo puntera' all'ultimo elemento, mentre bisogna far "ritornare" l'indirizzo al primo elemento dell'array di strutture.
2) Il codice sarebbe ulteriormente ottimizzabile, ma lasciandolo piu' vicino possibile all'originale si evidenziano meglio le differenze.
Perfetto è tutto chiarissimo adesso, ti ringrazio vivamente

Se hai qualche ottimizzazione da suggerirmi fallo pure, ogni consiglio è sempre molto gradito.
__________________
Trattative sul mercatino del forum:tante e mai un problema! Controlla pure i miei vecchi post
Ntropy è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2009, 19:38   #7
BrutPitt
Senior Member
 
L'Avatar di BrutPitt
 
Iscritto dal: Mar 2009
Città: Bologna
Messaggi: 1174
Quote:
Originariamente inviato da Ntropy Guarda i messaggi
Se hai qualche ottimizzazione da suggerirmi fallo pure, ogni consiglio è sempre molto gradito.
Nulla di che... eliminavo anche la variabile j che ora funge solo da contatore di ciclo, e non piu' da indice di spiazzamento... e gestivo il ciclo direttamente con un confronto tra indirizzi:

Codice:
struct clienti* funzione(void){
   
   char buffer[SLEN];
   int i; 
   int randm;
   struct clienti *nodo, *ptrNodo, *ptrEnd; //copia di nodo e indirizzo fineciclo
   
   if (!(nodo=(struct clienti*)malloc(NUMNODI*sizeof(struct clienti)))){
     printf("Memoria non allocabile!\n");
     return NULL;
     }

     ptrEnd = nodo+NUMNODI; //identifico "indirizzo" di fine array...

     //cosi' gestsisco il ciclo con gli indirizzi 

     for(ptrNodo=nodo; ptrNodo<ptrEnd; ptrNodo++){ 
                          
         printf("Inserisci il nome:");
         scanf("%s",buffer); 
         strcpy(ptrNodo->filename,buffer); //accedo tramite puntatore
         
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Xcoords[i]=1+rand()%100; //accedo tramite puntatore
   
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Ycoords[i]=1+rand()%100; //accedo tramite puntatore

   }                

    return nodo;         

}
BrutPitt è offline   Rispondi citando il messaggio o parte di esso
Old 26-05-2009, 22:17   #8
Ntropy
Senior Member
 
L'Avatar di Ntropy
 
Iscritto dal: Oct 2003
Messaggi: 2002
Quote:
Originariamente inviato da BrutPitt Guarda i messaggi
Nulla di che... eliminavo anche la variabile j che ora funge solo da contatore di ciclo, e non piu' da indice di spiazzamento... e gestivo il ciclo direttamente con un confronto tra indirizzi:

Codice:
struct clienti* funzione(void){
   
   char buffer[SLEN];
   int i; 
   int randm;
   struct clienti *nodo, *ptrNodo, *ptrEnd; //copia di nodo e indirizzo fineciclo
   
   if (!(nodo=(struct clienti*)malloc(NUMNODI*sizeof(struct clienti)))){
     printf("Memoria non allocabile!\n");
     return NULL;
     }

     ptrEnd = nodo+NUMNODI; //identifico "indirizzo" di fine array...

     //cosi' gestsisco il ciclo con gli indirizzi 

     for(ptrNodo=nodo; ptrNodo<ptrEnd; ptrNodo++){ 
                          
         printf("Inserisci il nome:");
         scanf("%s",buffer); 
         strcpy(ptrNodo->filename,buffer); //accedo tramite puntatore
         
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Xcoords[i]=1+rand()%100; //accedo tramite puntatore
   
         for( i=0;i<ARRAYSIZE;i++)
             ptrNodo->Ycoords[i]=1+rand()%100; //accedo tramite puntatore

   }                

    return nodo;         

}
Eh no ottimo consiglio invece. Ti ringrazio perchè sei stato utilissimo.
__________________
Trattative sul mercatino del forum:tante e mai un problema! Controlla pure i miei vecchi post
Ntropy è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Lenovo LOQ 15i Gen 10 (15IRX10) alla prova: il notebook gaming 'budget' che non ti aspetti Lenovo LOQ 15i Gen 10 (15IRX10) alla prova: il n...
Due mesi di Battlefield 6: dalla campagna al battle royale, è l'FPS che stavamo aspettando Due mesi di Battlefield 6: dalla campagna al bat...
Antigravity A1: drone futuristico per riprese a 360° in 8K con qualche lacuna da colmare Antigravity A1: drone futuristico per riprese a ...
Sony Alpha 7 V, anteprima e novità della nuova 30fps, che tende la mano anche ai creator Sony Alpha 7 V, anteprima e novità della ...
realme GT 8 Pro Dream Edition: prestazioni da flagship e anima racing da F1 realme GT 8 Pro Dream Edition: prestazioni da fl...
La NASA prova i droni marziani nella Dea...
IBM acquisisce Confluent, leader nella g...
La Commissione Europea indaga AWS e Azur...
Sonda spaziale ESA JUICE: rilasciata una...
PS5 PRO a 698€? L'offerta che nessuno si...
L'equipaggio di Shenzhou-21 ha completat...
Dal milione di download alla rimozione f...
Legno, pelle naturale e socket AM5: la G...
Attenti a X-GPU: la piattaforma sospetta...
La Lotus Eletre Plug-in esiste davvero, ...
Garanzia estesa su Pixel 9 Pro e 9 Pro X...
Addio intelligenza artificiale su Window...
FRAME 4000D LCD RS ARGB, il nuovo case d...
Netflix: AV1 ora gestisce il 30% dello s...
Tesla Optimus inciampa in pubblico e div...
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: 20:56.


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