Passando alocada dinamicamente inteiro matrizes em C
-
23-08-2019 - |
Pergunta
Eu li o exemplo em " Passando matrizes multidimensionais em C "neste site.
É um grande exemplo utilizando matrizes de caracteres, e eu aprendi muito com ele. Eu gostaria de fazer a mesma coisa, criando uma função para lidar com uma matriz de inteiros unidimensional alocada dinamicamente, e depois disso, criar uma outra função para lidar com uma matriz de inteiros multi-dimensional. Eu sei como fazê-lo como um valor de retorno para uma função. Mas nesta aplicação eu preciso fazê-lo na lista de argumentos para a função.
Tal como no exemplo mencionado acima, gostaria de passar um ponteiro para uma matriz de números inteiros para uma função, juntamente com o número de elementos de "num" (ou "linha" e "col" para uma função de matriz 2D , etc). Eu tenho uma versão retrabalhada do outro exemplo aqui, mas eu não posso chegar a este trabalho, por mais que tentasse (linhas de código que é novo ou modificado, a partir desse exemplo, estão marcados). Alguém sabe como resolver isso?
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define ELEMENTS 5
void make(char **array, int **arrayInt, int *array_size) {
int i;
char *t = "Hello, World!";
int s = 10; // new
array = malloc(ELEMENTS * sizeof(char *));
*arrayInt = malloc(ELEMENTS * sizeof(int *)); // new
for (i = 0; i < ELEMENTS; ++i) {
array[i] = malloc(strlen(t) + 1 * sizeof(char));
array[i] = StrDup(t);
arrayInt[i] = malloc( sizeof(int)); // new
*arrayInt[i] = i * s; // new
}
}
int main(int argc, char **argv) {
char **array;
int *arrayInt1D; // new
int size;
int i;
make(array, &arrayInt1D, &size); // mod
for (i = 0; i < size; ++i) {
printf("%s and %d\n", array[i], arrayInt1D[i]); // mod
}
return 0;
}
Solução
Há um monte de problemas em que o código. Ter um olhar para o seguinte:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define ELEMENTS 5
/*
* A string is an array of characters, say char c[]. Since we will be creating
* an array of those, that becomes char *(c[]). And since we want to store the
* memory we allocate somewhere, we must be given a pointer. Hence char
* **(c[]).
*
* An int doesn't require a complete array, just int i. An array of those is
* int i[]. A pointer to those is then int *(i[]).
*/
void
make(char **(chars[]), int *(ints[]), size_t len)
{
static char hw[] = "Hello, World!";
size_t i = 0;
/*
* Allocate the memory required to store the addresses of len char arrays.
* And allocate the memory required to store len ints.
*/
*chars = malloc(len * sizeof(char *));
*ints = malloc(len * sizeof(int));
/* Fill each element in the array... */
for (i = 0; i < ELEMENTS; i++) {
/* ... with a *new copy* of "Hello world". strdup calls malloc under
* the hood! */
(*chars)[i] = strdup(hw);
/* ...with a multiple of 10. */
(*ints)[i] = i * 10;
}
}
int
main(void)
{
/* A string c is a character array, hence char c[] or equivalently char *c.
* We want an array of those, hence char **c. */
char **chars = NULL;
/* An array of ints. */
int *ints = NULL;
size_t i = 0;
/* Pass *the addresses* of the chars and ints arrays, so that they can be
* initialized. */
make(&chars, &ints, ELEMENTS);
for (i = 0; i < ELEMENTS; ++i) {
printf("%s and %d\n", chars[i], ints[i]);
/* Don't forget to free the memory allocated by strdup. */
free(chars[i]);
}
/* Free the arrays themselves. */
free(ints);
free(chars);
return EXIT_SUCCESS;
}
Outras dicas
Está faltando tamanho de uma linha aqui:
arrayInt[i] = malloc( sizeof(int)); // new
deve ser algo como:
arrayInt[i] = malloc( row_len * sizeof(int)); // new
onde antes você estava usando o comprimento da corda dada como o tamanho da linha (também strlen(t)+1
deve ser entre parênteses, embora o efeito é o mesmo, porque sizeof(char)
é 1)