Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Renault Twingo E-Tech Electric: che prezzo!
Renault Twingo E-Tech Electric: che prezzo!
Renault annuncia la nuova vettura compatta del segmento A, che strizza l'occhio alla tradizione del modello abbinandovi una motorizzazione completamente elettrica e caratteristiche ideali per i tragitti urbani. Renault Twingo E-Tech Electric punta su abitabilità, per una lunghezza di meno di 3,8 metri, abbinata a un prezzo di lancio senza incentivi di 20.000€
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media
Nel Formula 1 Technology and Media Centre di Biggin Hill, la velocità delle monoposto si trasforma in dati, immagini e decisioni in tempo reale grazie all’infrastruttura Lenovo che gestisce centinaia di terabyte ogni weekend di gara e collega 820 milioni di spettatori nel mondo
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica
Il nuovo gimbal mobile DJI evolve il concetto di tracciamento automatico con tre modalità diverse, un modulo multifunzionale con illuminazione integrata e controlli gestuali avanzati. Nel gimbal è anche presente un'asta telescopica da 215 mm con treppiede integrato, per un prodotto completo per content creator di ogni livello
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 28-05-2011, 17: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, 19: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 19:32.
demos88 è offline   Rispondi citando il messaggio o parte di esso
Old 28-05-2011, 21: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, 19: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, 20: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, 21: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 22:15.
Lazy Bit è offline   Rispondi citando il messaggio o parte di esso
Old 29-05-2011, 22: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, 22: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


Renault Twingo E-Tech Electric: che prezzo! Renault Twingo E-Tech Electric: che prezzo!
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media Il cuore digitale di F1 a Biggin Hill: l'infrast...
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica DJI Osmo Mobile 8: lo stabilizzatore per smartph...
Recensione Pura 80 Pro: HUAWEI torna a stupire con foto spettacolari e ricarica superveloce Recensione Pura 80 Pro: HUAWEI torna a stupire c...
Opera Neon: il browser AI agentico di nuova generazione Opera Neon: il browser AI agentico di nuova gene...
La Cina dice addio a NVIDIA? Il governo ...
Microlino, simbolo italiano della mobili...
Apple disattiverà la sincronizzaz...
Google lancia l'allarme: attenzione ai m...
Primo test drive con Leapmotor B10: le c...
'Non può essere un robot': l'uman...
Monopattino elettrico Segway Ninebot Max...
Syberia Remastered è disponibile:...
Sony scopre che tutti i modelli AI hanno...
Amazon nasconde un -15% su 'Seconda Mano...
Due occasioni Apple su Amazon: iPhone 16...
Verso la fine della TV tradizionale? I g...
Cassa JBL a 39€, portatili, smartphone, ...
Cometa interstellare 3I/ATLAS: la sonda ...
Jensen Huang e Bill Dally di NVIDIA prem...
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: 18:36.


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