如何在二维数组结构中替换不同的值和c中的二维数组结构的打印函数?

时间:2017-12-03 05:20:49

标签: c arrays pointers struct

当工作人员挖掘整个值的空间时,输出应该打印出来“s”它会变为值“T”并且如果你可以帮助创建一个打印地图结构的功能而不是每个都使用for循环时间我们也会非常感激!:

Crew Dig Carry
1 3 1
2 2 5
3 4 1
4 1 3

您想将船员1送到哪里? 4s 1s 3s
1s 2s 3s
1s 5s 1s

1 3

您已从此部分删除了所有沙子!

您想在哪里派遣2名船员? 4s 1s 1T
1s 2s 3s
1s 5s 1s

2 2

到目前为止我所拥有的:
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>

#define NUMCREW 4    
#define MAXHOUR 8    
#define MINHOUR 1    
#define ROW 3    
#define COL 3    


struct pirate {
    int dig;
    int carry;
};  

struct map {
    int sand;
    int treasure;
};

void printcrew(struct pirate * pirate_data);  
void print(struct map * map_data);    


int main() {
char filename[20];
int hour=8;
int crew=0;
int i, j, x, y;
int space;
struct map map_data[ROW][COL];
struct pirate pirate_data[NUMCREW];
FILE * ifp;

printf("You have arrived at Treasure Island!\n");
printf("What is the name of your map?\n");
scanf("%s", filename);

ifp = fopen(filename, "r");



for (i=0; i<ROW; i++){
    for (j=0; j<COL; j++) {
        fscanf(ifp, "%d %d", &map_data[i][j].sand, &map_data[i][j].treasure);
    }
}


for (i=0; i<NUMCREW; i++) {
    fscanf(ifp, "%d", &pirate_data[i].dig);
    fscanf(ifp, "%d", &pirate_data[i].carry);
}

fclose(ifp);
for (hour=8; hour >= MINHOUR; hour--) {
    printf("\nYou have %d hours left to dig up the treasure.\n", hour);
    printcrew(pirate_data);
    printf("\n");

    for (crew=0; crew<NUMCREW; crew++){
        printf("Where would you like to send crew member %d?\n", crew+1);

        for(i=0; i<ROW; i++) {
            for (j=0; j<COL; j++) {
            printf("%ds\t", map_data[i][j].sand);
                if (map_data[i][j].sand == 0) {
                    printf("%dT\t", map_data[i][j].treasure);
            }
        }
            printf("\n");
            }
            scanf("%d %d", &x, &y);
            map_data[x-1][y-1].sand = map_data[x-1][y-1].sand - pirate_data[crew].dig;
        }
    }

return 0;
}

void printcrew(struct pirate * pirate_data) {
int i;`

printf("Crew \t Dig \t Carry \n");
for (i=0; i<NUMCREW; i++)
    printf("%d \t %d \t %d \n", i+1, pirate_data[i].dig, pirate_data[i].carry);

return;

}

void print(struct map * map_data) {
int i, j;

for(i=0; i<ROW; i++) {
    for (j=0; j<COL; j++) {
        printf("%ds\t", map_data[i][j].sand);
    }
    printf("\n");
}

return;

}

我的输出不断出现如下:
你想在哪里派遣机组人员1?     4s 1s 3s
1s 2s 3s
1s 5s 1s

1 3
你想在哪里派遣2名船员?    4s 1s 0s 1T
1s 2s 3s
1s 5s 1s

2 2
你想在哪里派遣3名船员?     4s 1s 0s 1T
1s 0s 1T 3s
1s 5s 1s

至于打印结构图功能,我不断收到以下错误:
在功能'打印'中:|
错误:下标值既不是数组也不是指针,也不是向量|

1 个答案:

答案 0 :(得分:0)

发布的代码中似乎有拼写错误; printcrew()函数中有一个迷路`字符。应该看到编译器警告,并且应修复所有编译器警告。

int i;`

关于print()函数的问题,函数原型是错误的。在大多数表达式中,包括函数调用,数组衰减到指向其第一个元素的指针。现在,像struct map map_data[ROW][COL]这样的二维数组实际上是ROWCOL元素数组的数组。也就是说,此数组的每一行都具有类型struct map map_data[COL]。当指针衰减发生时,结果是指向第一个元素的指针,该指针是指向数组的指针(不是指向map struct的指针)。表达这种指向数组的指针的语法是:struct map (*ptr)[COL]。所以,函数原型应该是:

void print(struct map (*map_data)[COL]);

要解决更新显示屏以显示找到的宝藏的问题,请检查是否在打印沙子之前找到了的宝藏,并且如果没有找到宝藏则只打印沙子:

for(i=0; i<ROW; i++) {
    for (j=0; j<COL; j++) {
        int sand = map_data[i][j].sand;
        if (sand > 0) {
            printf("%ds\t", sand);
        } else {
            printf("%dT\t", map_data[i][j].treasure);
        }
    }
    printf("\n");
}

有关输入验证的一些评论是有序的。使用带有%s族函数的%[]scanf()指令读取字符串输入时,请始终指定最大宽度;如果不这样做会导致缓冲区溢出和未定义的行为。

请注意scanf()和家人会返回成功作业的数量;应检查此值以确保输入符合预期。在发布的代码中,如果用户通过键盘狡猾地发出文件结束信号,scanf()将返回而不进行分配。由于filename[]未初始化,因此在使用filename时会导致未定义的行为。相反,尝试类似:

if (scanf("%19s", filename) < 1) {
    fprintf(stderr, "Input error\n");
    exit(EXIT_FAILURE);
}

验证打开文件的尝试是否成功也是至关重要的。当代码继续打开时文件无法打开,就像它们成功打开一样,接下来是未定义的行为。 fopen()在失败时返回空指针:

ifp = fopen(filename, "r");
if (ifp == NULL) {
    fprintf(stderr, "Unable to open file %s\n", filename);
    exit(EXIT_FAILURE);
}

这个故事的寓意是你应该检查从任何返回有意义值的函数返回的值。

由于未能在此验证输入,还有可能出现问题:

scanf("%d %d", &x, &y);
map_data[x-1][y-1].sand = map_data[x-1][y-1].sand - pirate_data[crew].dig;

如果用户输入非数字输入,则相关变量中不会存储任何值。由于使用先前存储的值,这最多会导致令人惊讶的行为。在最坏情况下,当相关变量的值不确定时(例如,未初始化),这会导致未定义的行为。此外,用户可以在此输入非正数,导致负数组索引和未定义的行为。或者用户可能输入太大的值,导致尝试越界数组访问和未定义的行为。此外,sand成员即使在否定时也会递减;这可能是也可能不是可取的,但可能导致带有签名整数溢出的未定义行为,因为不太可能是勤劳的盗版者和恶意用户的组合。

验证所有输入,尤其是永远不会被信任且始终被视为恶意的用户输入。检查scanf()返回的值,以确保输入至少与预期一致,并检查输入值是否在可接受的范围内。处理输入的错误取决于程序员;一个选项是再次提示正确的输入:

while (scanf("%d %d", &x, &y) < 2
       || x < 1
       || x > COL
       || y < 1
       || y > ROW) {
    printf("Please enter numbers in the ranges [1, %d] [1, %d]\n",
           ROW, COL);
}
map_data[x-1][y-1].sand = map_data[x-1][y-1].sand - pirate_data[crew].dig;