Torna indietro   Hardware Upgrade Forum > Software > Programmazione

ASUS ROG Swift OLED PG34WCDN recensione: il primo QD-OLED RGB da 360 Hz
ASUS ROG Swift OLED PG34WCDN recensione: il primo QD-OLED RGB da 360 Hz
ASUS ROG Swift OLED PG34WCDN è il primo monitor gaming con pannello QD-OLED Gen 5 a layout RGB Stripe Pixel e 360 Hz su 34 pollici: lo abbiamo misurato con sonde colorimetriche e NVIDIA LDAT. Ecco tutti i dati
Recensione Nothing Phone (4a) Pro: finalmente in alluminio, ma dal design sempre unico
Recensione Nothing Phone (4a) Pro: finalmente in alluminio, ma dal design sempre unico
Nothing Phone (4a) Pro cambia pelle: l'alluminio unibody sostituisce la trasparenza integrale, portando una solidità inedita. Sotto il cofano troviamo uno Snapdragon 7 Gen 4 che spinge forte, mentre il display è quasi da top dig amma. Con un teleobiettivo 3.5x e la Glyph Matrix evoluta, è la prova di maturità di Carl Pei. C'è qualche compromesso, ma a 499EUR la sostanza hardware e la sua unicità lo rendono un buon "flagship killer" in salsa 2026
WoW: Midnight, Blizzard mette il primo, storico mattone per l'housing e molto altro
WoW: Midnight, Blizzard mette il primo, storico mattone per l'housing e molto altro
Con Midnight, Blizzard tenta il colpaccio: il player housing sbarca finalmente su Azeroth insieme a una Quel'Thalas ricostruita da zero. Tra il dramma della famiglia Ventolesto e il nuovo Prey System, ecco com'è la nuova espansione di World of Warcraft
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


ASUS ROG Swift OLED PG34WCDN recensione: il primo QD-OLED RGB da 360 Hz ASUS ROG Swift OLED PG34WCDN recensione: il prim...
Recensione Nothing Phone (4a) Pro: finalmente in alluminio, ma dal design sempre unico Recensione Nothing Phone (4a) Pro: finalmente in...
WoW: Midnight, Blizzard mette il primo, storico mattone per l'housing e molto altro WoW: Midnight, Blizzard mette il primo, storico ...
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...
Secondo Elon Musk FSD è più...
Anche Cloudflare fissa il 2029 per la si...
Hacker sfruttano da mesi un bug segreto ...
ASUSTOR Lockerstor 24R Pro Gen2: 24 bay ...
Rigetti supera la soglia dei 100 qubit: ...
eFootball raggiunge il miliardo di downl...
Come provare OpenClaw facilmente grazie ...
Microsoft conferma: questo glitch dell'o...
Toyota bZ7: una berlina da oltre 5 metri...
Artemis II, le prime foto del lato nasco...
Sempre più pubblicità su YouTube: arriva...
Polestar fa +80% in Italia e tocca quota...
Il tuo Mac smette di connettersi a Inter...
La nuova alleanza Intel-Google ridefinis...
Energia troppo cara, regole da rivedere:...
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: 00:45.


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