[英]How to return pointer to array of struct in C
我使用malloc
在 function 中创建了一个student
结构,并用数据填充了它。 我想将结构数组的地址返回给main
并逐行打印,但是在我的实现中,它没有打印。 我调试了我的代码,确实,它能够将我的数组的地址返回给main
。 我不知道为什么它不打印。 有什么想法吗?
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct student Student;
struct student
{
char name[100];
int age;
char sex;
};
Student **getstudents(int n)
{
Student **t = malloc(sizeof *t * n); // memory for the array of pointers
for (int i = 0; i < n; i++) // memory for each individual pointer
{
t[i] = malloc(sizeof **t);
}
/* Data is inputted by user with form <student> <sex> <age>, then get mapped to the struct *t */
return t; /* Pointer to an array of pointers */
}
int main()
{
Student **students;
int n;
scanf("%d\n", &n);
students = getstudents(n);
for (int i = 0; i < n; i++)
{
printf("Name: %s, Sex: %s, Age: %d\n", students[i]->name, students[i]->sex, students[i]->age);
}
for (int i = 0; i < n; i++)
{
free(students[i]);
}
free(students);
return 0;
}
I am only allowed to modify the code in `Student **getstudents(int n)`.
在行中:
Student *t = (char*)malloc(sizeof(Student)*n);
您正在为一个指针分配 memory ,如果要返回指向指针的指针,则需要相应地分配 memory :
Student **t = malloc(sizeof *t * n); // memory for the array of pointers
for(int i = 0; i < n; i++){ // memory for each individual pointer
t[i] = malloc(sizeof **t);
}
要稍后释放指针,您还需要释放先前分配的每个单独的指针:
for(int i = 0; i < n; i++){ // memory for each individual pointer
free(students[i]);
}
free(students);
请注意,单个字符的说明符是%c
,需要更正 printf :
printf("Name: %s, Sex: %c, Age: %d\n", students[i]->name, students[i]->sex, students[i]->age);
// ^^
我要更改的另一件事是在strncpy
而不是 null 稍后终止字符串,我会让 function 这样做:
// one more byte and strncpy terminates the string for you
strncpy(t[i]->name, data[0], strlen(data[0]) + 1);
// ^^^^
// already corrected for the new pointer
更正了此处的问题是一种可能的替代方法,您可以使用sscanf
从一个 go 中的entry
解析结构中的所有元素,如果您愿意:
Student **getstudents(int n)
{
Student **t = malloc(sizeof *t * n); // memory for the array of pointers
if (t == NULL)
{
perror("malloc");
exit(EXIT_FAILURE);
}
for (int i = 0; i < n; i++) // memory for each individual pointer
{
t[i] = malloc(sizeof **t);
if (t[i] == NULL)
{
perror("malloc");
exit(EXIT_FAILURE);
}
}
for (int i = 0; i < n; i++)
{
char entry[100];
if (fgets(entry, sizeof entry, stdin))
{
if (sscanf(entry, "%25s %c %d", t[i]->name, &t[i]->sex, &t[i]->age) != 3)
{
// deal with bad input
}
}
}
return t;
}
anastaciu 的回答指出了很多麻烦,但还有其他问题:
您的代码中的另一个问题是您使用 '%s' 作为性别,因为性别是唯一的字符。 您应该使用 %c 否则 printf function 将尝试解析字符串并获得 SEGFAULT。
我也敦促您严格检查每个 memory 分配。 总是。
我的pov修改后的代码:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct student Student;
struct student{
char name[100];
int age;
char sex;
};
Student **getstudents(int);
void free_students(Student**, int);
int main(){
Student **students;
int n = 4;
students = getstudents(n);
for(int i = 0; i < n; i++){
if (students[i] != NULL) {
printf("Name: %s, Sex: %c, Age: %d\n", students[i]->name, students[i]->sex, students[i]->age);
}
}
free_students(students, n);
return 0;
}
Student **getstudents(int n){
Student **t = (Student **)malloc(sizeof(Student *)*n);
if (t==NULL) {
perror("Memory: can't allocate.");
return(NULL);
}
/* Input: <name> <sex> <age> */
char entry[100];
for(int i = 0; i < n; i++){
t[i]=NULL;
if (fgets(entry,100,stdin) != NULL) {
int readBytes = strlen(entry);
char newString[3][25];
int k,j,ctr;
j=0; ctr=0;
for(k=0;k<=readBytes;k++)
{
if(entry[k]==' '||entry[k]=='\0'||entry[k]=='\n')
{
newString[ctr][j]='\0';
ctr++;
j=0;
}
else
{
newString[ctr][j]=entry[k];
j++;
}
}
t[i] = (Student *)malloc(sizeof(Student));
if (t[i] == NULL) {
perror("Memory: can't allocate.");
return(NULL);
}
strncpy(t[i]->name, newString[0], strlen(newString[0]));
t[i]->name[strlen(newString[0])] = '\0';
t[i]->sex = *newString[1];
t[i]->age = atoi(newString[2]);
}
}
return t;
}
void free_students(Student **students, int n){
for(int i=0; i<n; i++){
free(students[i]);
}
free(students);
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.