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

标签 c arrays pointers struct

当 crew 挖出值“s”的整个空间时,输出应该像这样打印出来,它变为值“T”,如果你能帮助创建一个函数来打印 map 结构,而不是每次都使用 for 循环时间我们也将不胜感激!:

Crew Dig Carry<br/> 1 3 1<br/> 2 2 5<br/> 3 4 1<br/> 4 1 3

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

1 3

您已移除此部分的所有沙子!

您想将船员 2 送到哪里?
4s 1s 1T<br/> 1s 2s 3s<br/> 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<br/> 1s 2s 3s<br/> 1s 5s 1s
1 3
你想把船员 2 送到哪里?
4s 1s 0s 1T<br/> 1s 2s 3s<br/> 1s 5s 1s
2 2
你想把船员 3 送到哪里?
4s 1s 0s 1T<br/> 1s 0s 1T 3s<br/> 1s 5s 1s

至于打印 struct map 函数,我不断收到以下错误:
在函数“打印”中:|
错误:下标值既不是数组也不是指针也不是 vector |

最佳答案

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

int i;`

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

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");
}

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

注意 scanf() 和 family 返回成功赋值的次数;应检查此值以确保输入符合预期。在发布的代码中,如果用户狡猾地从键盘发出文件结束信号,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;

关于c - 我如何将二维数组结构中的值替换为 c 中二维数组结构的不同值和打印函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47615518/

相关文章:

在C中将uint8_t数组连接到uint32_t数组

c++ - “_snprintf”未在此范围内声明

javascript - findIndex() 总是在包含对象的数组上返回 -1

c - void* 将具有与指向 char 的指针相同的表示和内存对齐方式

c++ - 关于 C++ 中的 & 运算符的问题

C - 构建动态分配的指针数组,指向由文件输入填充的结构

c - main 函数和之后的变量

c - 99瓶啤酒递归好像不行

php - 同一数组的数组索引合并

python - 如何在这两个 numpy 数组过程中保持相同的精度?