Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Ecovacs Goat O1200 LiDAR Pro: la prova del robot tagliaerba con tagliabordi integrato
Ecovacs Goat O1200 LiDAR Pro: la prova del robot tagliaerba con tagliabordi integrato
Nuova frontiera per i robot tagliaerba, con Ecovacs GOAT O1200 LiDAR Pro che riconosce l'ambiente in maniera perfetta, grazie a due sensori LiDAR, e dopo la falciatura può anche rifinire il bordo con il tagliabordi a filo integrato
Recensione Samsung Galaxy S26+: sfida l'Ultra, ma ha senso di esistere?
Recensione Samsung Galaxy S26+: sfida l'Ultra, ma ha senso di esistere?
Equilibrio e potenza definiscono il Samsung Galaxy S26+, un flagship che sfida la variante Ultra e la fascia alta del mercato con il primo processore mobile a 2nm. Pur mantenendo l'hardware fotografico precedente, lo smartphone brilla per un display QHD+ da 6,7 pollici d'eccellenza, privo però del trattamento antiriflesso dell'Ultra, e per prestazioni molto elevate. Completano il quadro la ricarica wireless a 20W e, soprattutto, un supporto software settennale
Zeekr X e 7X provate: prezzi, autonomia fino a 615 km e ricarica in 13 minuti
Zeekr X e 7X provate: prezzi, autonomia fino a 615 km e ricarica in 13 minuti
Zeekr sbarca ufficialmente in Italia con tre modelli elettrici premium, X, 7X e 001, distribuiti da Jameel Motors su una rete di 52 punti vendita già attivi. La Zeekr X parte da 39.900 euro, la 7X da 54.100: piattaforma a 800V, chip Snapdragon di ultima generazione, ricarica ultraveloce e un'autonomia dichiarata fino a 615 km WLTP. Le prime consegne sono previste a metà aprile
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 28-05-2011, 16:59   #1
devil_prince
Junior Member
 
Iscritto dal: May 2011
Messaggi: 17
[C]Prodotto di matrici allocate dinamicamente

Ragazzi devo fare il prodotto di matrici, però allocandole (sia le matrici di partenza che quella risultate) dinamicamente. Questo è quello che ho fatto fin ora:
Codice:
#include <stdio.h>
#include <stdlib.h>

void main ()
{
    int i, j, k, m, n, p, *A, *B, *C, D[100][100];

    puts("-------------------------------");
    puts("PRODOTTO DI MATRICI\n");
    puts("-------------------------------");
    /*input dei parametri m p ed n*/
    printf("Inserire le righe per la prima matrice (parametro -m-): ");
    scanf("%d",&m);
    printf("Inserire le colonne per la prima matrice e righe per la seconda matrice (parametro -p-): ");
    scanf("%d",&p);
    printf("Inserire le colonne per la seconda matrice (parametro -n-): ");
    scanf("%d",&n);
    /*allocazione delle matrici*/
    A=malloc(m*p*sizeof(int));
    B=malloc(p*n*sizeof(int));
    C=malloc(m*n*sizeof(int));


    /*Azzeramento della matrice C*/
    for(i=0;i<m;i++){
        for(j=0;j<n;j++){
        *(C+i*m*j)=0;}
        }


    /*inserimento degli elementi per le matrici A e B*/
    for(i=0;i<m;i++){
        for(j=0;j<p;j++){
        printf("Inserire l'elemento dell'array A di posto (%d,%d): ",i,j);
        scanf("%d",&D[i][j]);}
        }
    printf("Ecco la matrice A allocata dinamicamente\n");
     for(i=0;i<m;i++){             //assegnati i valori alla matrice di appoggio li copiamo in A
        for(j=0;j<p;j++)
        {   *(A+i*p*j)=D[i][j];
            printf("%d ",*(A+i*p*j));}
        puts("\n");}



    for(i=0;i<m;i++){               //azzeriamo la matrice di appoggi
        for(j=0;j<n;j++){
        D[i][j]=0;}
        }
    for(i=0;i<p;i++){                  //assegnati i valori alla matrice di appoggio li copiamo in B
        for(j=0;j<n;j++){
        printf("Inserire l'elemento dell'array B di posto (%d,%d): ",i,j);
        scanf("%d",&D[i][j]);}
        }
    printf("Ecco la matrice B allocata dinamicamente\n");
     for(i=0;i<p;i++){
        for(j=0;j<n;j++)
        {   *(B+i*n*j)=D[i][j];
            printf("%d ",*(B+i*n*j));}
        puts("\n");}



    /*Prodotto di AxB*/
    for(i=0;i<m;i++){
        for(j=0;j<n;j++){
            for(k=0;k<p;k++){
              *(C+i*m*j) = *(A+i*p*k)**(B+k*p*j)+*(C+i*m*j);}
              }
        }
     printf("Ecco la matrice C allocata dinamicamente\n");
     for(i=0;i<n;i++){
        for(j=0;j<m;j++)
        {printf("%d ",*(C+i*m*j));}
        puts("\n");}
}
le matrici di partenza si allocano bene, ma la matrice C risultante ha valori tutti sballati...sbaglio qualcosa nella formula??
devil_prince è offline   Rispondi citando il messaggio o parte di esso
Old 28-05-2011, 18:22   #2
demos88
Senior Member
 
Iscritto dal: Nov 2004
Città: Padova
Messaggi: 2342
è da un po' che non uso l'allocazione dinamica ma a occhio pare che ci sia qualche errore in tutti gli accessi alla memoria.
Prendendo in esempio l'azzeramento della matrice C
Codice:
/*Azzeramento della matrice C*/
    for(i=0;i<m;i++){
        for(j=0;j<n;j++){
           *(C+i*m*j)=0;
        }
    }
Se ho ben inteso, i che va da 0 a m rappresenta la riga, mentre j da 0 a n la colonna. Tu quindi scrivi da sinistra verso destra una riga alla volta.
Inteso questo, mi vien qualche dubbio su come siano allocati in memoria i dati.
L'indirizzo inferiore si ottiene con i=j=0 e quindi è l'indirizzo di C ed è giusto. Il limite maggiore invece si ha per i=m-1 e j=n-1. Posti per esempio m=n=10, tu allochi 4*10*10=400byte ma il limite maggiore durante l'inserimento arriva ad essere un offset rispetto a C di 9*10*9=810. Sei fuori dall'allocazione che avevi previsto per C... infatti l'indirizzamento corretto sarebbe *(C + i*m + j), ma questo sarebbe giusto se scrivessi char da 1 byte... tu stai scrivendo int da 4 byte (do per scontato che sizeof(int)=4, controlla per sicurezza), quindi l'indirizzamento giusto per te è *(C + 4*(i*m + j)). Facciamo la verifica del limite maggiore: in tal caso si ha un offset massimo di 4*(9*10 + 9)=396, ed è giusto
Innanzi tutto prova a cambiare questo su tutte le matrici, probabilmente scrivendo una matrice andavi a sovrascrivere le altre matrici ma addirittura scrivevi a indirizzi forse occupati da altri dati, quindi mescolavi parecchio i valori. Il motivo per cui ti sembrasse giusta l'allocazione delle prime 2 matrici è banale: tu andavi a scrivere in una locazione di memoria sbagliata e la leggevi subito dopo, è ovvio che leggi il valore che hai appena inserito.

ps: non sono sicuro che ti servano quegli azzeramenti, di fatto tu azzeri la matrice per scriverci poi sopra, tanto vale scriverci sopra direttamente, no?
__________________
CPU Ryzen 2600 @ 3,95Ghz + Bequiet Dark Rock TF / MB Asus X470-F Gaming / RAM 2x8GB DDR4 G.Skill FlareX 3200 CL14 / VGA Sapphire RX 7900 XT Nitro+ @ 3200Mhz / SSD Samsung 970 Pro 512GB + Sandisk 240GB Plus + Sandisk 960GB Ultra II PSU Seasonic Platinum P-660 / Headset Kingston HyperX Flight

Ultima modifica di demos88 : 28-05-2011 alle 18:32.
demos88 è offline   Rispondi citando il messaggio o parte di esso
Old 28-05-2011, 20:35   #3
BlackShark92
Member
 
Iscritto dal: May 2011
Messaggi: 39
allora, se usi la calloc non hai alcun bisogno di azzerare la memoria, lo fa da sè.
La sintassi è calloc(numero_di_elementi, dimensione_dell'elemento)
BlackShark92 è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 18:41   #4
Lazy Bit
Member
 
Iscritto dal: May 2011
Messaggi: 47
Non vorrei sbagliarmi ma dovresti usare i puntatori di puntatori: int** matrix, altrimenti la matrice ti viene riconosciuta come unico blocco e non puoi usare la notazione matrix[i][j]. Ricorda di liberare la memoria con free(matrix).
Lazy Bit è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 19:28   #5
BlackShark92
Member
 
Iscritto dal: May 2011
Messaggi: 39
Quote:
Originariamente inviato da Lazy Bit Guarda i messaggi
Non vorrei sbagliarmi ma dovresti usare i puntatori di puntatori: int** matrix, altrimenti la matrice ti viene riconosciuta come unico blocco e non puoi usare la notazione matrix[i][j]. Ricorda di liberare la memoria con free(matrix).
Ah certo, quello è ovvio, meno male che lo hai fatto notare. Così stai allocando un vettore, invece se vuoi una matrice la devi dichiarare come puntatore a puntatori, ovvero int **A (con 2 star)
BlackShark92 è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 20:58   #6
Lazy Bit
Member
 
Iscritto dal: May 2011
Messaggi: 47
In teoria si potrebbe allocare un unico blocco, utilizzando un normale puntatore, come ha fatto Devil Prince, ma nel caso le matrici contenessero parecchi elementi, sarebbe difficile trovare un intero blocco di grandi dimensioni. Invece, utilizzando i puntatori a puntatori, i blocchi sarebbero di dimensioni ridotte e meglio gestibili (utilizzando la notazione matrix[i][j]).

Comunque, anche se ciò non risolverà il problema, la formula per il calcolo del prodotto di matrici è scorretta, perché la matrice prodotto è C[M, N], e si devono moltiplicare gli elementi riga per colonna. Quindi la formula corretta sarà la seguente:

Codice:
for(i = 0; i < m; i++)
{
     for(j = 0; j < n; j++)
     {
         for(k = 0; k < m; k++)
         {
             c[i][j] += a[i][k] * b[k][j];
         }
     }
}
L'inizializzazione della matrice a zero non è necessaria. Appena avrò un po' di tempo, inserirò l'algoritmo completo.

PS: Per verificare la correttezza della formula, prova a creare una matrice in modo statico, con dimensioni definite a priori. Una volta avuta la conferma, potrai cercare di allocare la matrice dinamicamente.

Ultima modifica di Lazy Bit : 29-05-2011 alle 21:15.
Lazy Bit è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 21:04   #7
BlackShark92
Member
 
Iscritto dal: May 2011
Messaggi: 39
Quote:
Originariamente inviato da Lazy Bit Guarda i messaggi
In teoria si potrebbe allocare un unico blocco, utilizzando un normale puntatore, come ha fatto Devil Prince, ma nel caso le matrici contenessero parecchi elementi, sarebbe difficile trovare un intero blocco di grandi dimensioni. Invece, utilizzando i puntatori a puntatori, i blocchi sarebbero di dimensioni ridotte e meglio gestibili (utilizzando la notazione matrix[i][j]).

Comunque, anche se ciò non risolverà il problema, la formula per il calcolo del prodotto di matrici è scorretta, perché la matrice prodotto è C[M, N], non C[M, P]. Quindi la formula corretta sarà la seguente:

Codice:
for(i = 0; i < m; i++)
    {
        for(j = 0; j < n; j++)
        {
            for(k = 0; k < m; k++)
            {
                c[i][j] = a[i][k] * b[k][j];
            }
        }
    }
L'inizializzazione della matrice a zero non è necessaria. Appena avrò un po' di tempo, inserirò l'algoritmo completo.
Ti manca un più lì...l'elemento c[i][j]esimo è dato dalla somma di quei prodotti lì...
Ecco qua
Codice:
int matXmatt(double **matrice1, int nrighe1, int ncolon1, double **matrice2, int nrighe2, int ncolon2, double **res) {
    int i, j, k;
    if (ncolon1!=nrighe2) return -1;
    else {for (i=0; i<nrighe1; i++) {
            for (j=0; j<ncolon2; j++) { res[i][j]= 0.0;
                                        for (k=0; k<ncolon1; k++)
                                        res[i][j]+= matrice1[i][k]*matrice2[k][j];
            }
                                    }
    return 0;
    }
}
Ho messo anche il controllo per gli errori
BlackShark92 è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 21:15   #8
Lazy Bit
Member
 
Iscritto dal: May 2011
Messaggi: 47
Quote:
Originariamente inviato da BlackShark92 Guarda i messaggi
Ti manca un più lì...l'elemento c[i][j]esimo è dato dalla somma di quei prodotti lì...
Giusto!
Lazy Bit è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Ecovacs Goat O1200 LiDAR Pro: la prova del robot tagliaerba con tagliabordi integrato Ecovacs Goat O1200 LiDAR Pro: la prova del robot...
Recensione Samsung Galaxy S26+: sfida l'Ultra, ma ha senso di esistere? Recensione Samsung Galaxy S26+: sfida l'Ultra, m...
Zeekr X e 7X provate: prezzi, autonomia fino a 615 km e ricarica in 13 minuti Zeekr X e 7X provate: prezzi, autonomia fino a 6...
Marathon: arriva il Fortnite hardcore Marathon: arriva il Fortnite hardcore
HP Imagine 2026: abbiamo visto HP IQ all’opera, ecco cosa può (e non può) fare HP Imagine 2026: abbiamo visto HP IQ all’opera, ...
Amazon Haul: -25% sul primo ordine con i...
Windows 11: Microsoft promette (ancora u...
NVIDIA non può più usare il suo stesso t...
Anthropic, con Claude triplicati i ricav...
Questa sedia ergonomica da ufficio a 169...
I 2 migliori portatili su Amazon: gran C...
Speciale monitor MSI: a partire da meno ...
La fine di Fortnite si sta avvicinando? ...
The Last of Us Online: a che punto era e...
MacBook con M5: il raffreddamento attivo...
Samsung contrasta la crisi delle memorie...
Google Meet arriva su CarPlay: le riunio...
Le 10 migliori offerte Amazon di Pasqua:...
Nuove fotografie dagli astronauti di Art...
La toilette della capsula Orion Integrit...
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:25.


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