在ncurses中访问WINDOW结构数组时出现分段错误

问题描述

我正在使用ncurses制作一个小型益智游戏,该游戏允许用户敲击任何箭头键来移动一个包含每个数字的框,这些框都在一个大框中,并按升序排列。这是一个4x4的框,一个子框为空白,以便其他子框可以移动。

现在,我在访问函数中创建的WINDOW(不是指针)数组时遇到问题,这样我就可以销毁所有sub_Box,更改数字数组并根据该数组重新创建子框。

#include <stdio.h>
#include <ncurses.h>
#include <math.h>
#include <string.h>


#define HEIGHT 16
#define WIDTH 32
#define STARTY ((LInes - HEIGHT) / 2)
#define STARTX ((COLS - WIDTH) / 2)

#define QEXIT 81
#define qEXIT 113



WINDOW *create_win(int height,int width,int starty,int startx){
        WINDOW *local_win;

        local_win = newwin(height,width,starty,startx);
        Box(local_win,0);
        wrefresh(local_win);

        return local_win;
};

void destroy_win(WINDOW *local_win){
        wborder(local_win,' ',' ');
        wrefresh(local_win);
        delwin(local_win);
};


void draw_list(WINDOW *local_win,WINDOW *num_wins,int arr[][4],int length){

        for(int i=0; i < length; i++){
                for(int j=0; j < length; j++){
                        refresh();
                        WINDOW *sub_win;
                        int height = HEIGHT / length,width = WIDTH / length,starty = STARTY + i*height,startx = STARTX + j*width;
                        sub_win = create_win(height,startx);
                        //wprintw(stdscr,"%d |",sizeof(*sub_win));

                        *(num_wins + i*4+j) = *sub_win;
                        //wprintw(stdscr,"%d - ",i*4+j);
                        //overwrite(sub_win,(num_wins + i*4+j));

                        //memcpy(num_wins + i*4+j,sub_win,sizeof(*sub_win));

                        int digity = height / 2,digitx = width / 2 - 1;
                        if(arr[i][j])
                                mvwprintw(sub_win,digity,digitx,"%d ",arr[i][j]);
                        else
                                mvwprintw(sub_win,"X ");

                        wrefresh(sub_win);
                };
        };
};

void destroy_list(WINDOW *num_wins){
        int length = 16;
        for(int i=0; i < length; i++){
                wprintw(stdscr,i);
                wborder((num_wins + i),' ');
                wrefresh((num_wins + i));
                delwin((num_wins + i));
        };
};

void play(){
        int arr[4][4] = {1,4,15,7,8,10,2,11,14,3,6,13,12,9,5,0};
        int length = 4;

        WINDOW *win;
        int starty,startx;

        //Placement
        starty = STARTY;
        startx = STARTX;

        printw("Press Q or q to exit\n");
        //printw("%d - %d\n%d - %d\n",LInes,COLS,HEIGHT,WIDTH);
        printw("\n");

        refresh();

        // Create original window               (Maybe it is useless)
        win = create_win(HEIGHT,WIDTH,startx);

        // Create sub windows for numbers
        WINDOW sub_wins[16];
        draw_list(win,sub_wins,arr,length);

        refresh();

        getch();

        destroy_win(&sub_wins[0]);
        refresh();
        //for(int i=0; i < 16; i++){
        //      wprintw(stdscr,"%d -",sizeof(sub_wins[i]));
                //destroy_win(sub_wins + i);
        //};
        wrefresh(win);

        int ch = getch();
        while(ch != QEXIT && ch != qEXIT){
                switch(ch){
                        case KEY_UP:
                                destroy_win(win);
                                win = create_win(HEIGHT,--starty,startx);
                                //destroy_list(sub_wins);
                                break;
                        case KEY_RIGHT:
                                destroy_win(win);
                                win = create_win(HEIGHT,++startx);
                                break;
                        case KEY_DOWN:
                                destroy_win(win);
                                win = create_win(HEIGHT,++starty,startx);
                                break;
                        case KEY_LEFT:
                                destroy_win(win);
                                win = create_win(HEIGHT,--startx);
                                break;

                };
                ch = getch();
        };
};

int main(void){
        //printList(arr,length);

        printf("\n");

        // Init ncurses
        initscr();
        cbreak();
        noecho();
        keypad(stdscr,TRUE);

        // Game mechanics
        play();

        endwin();

        return 0;
};

我在代码中抛出了多个问题,而某些部分与我的主要问题并不完全相关。抱歉。但我想主要解决有关破坏子窗口的问题。

谢谢。

解决方法

您不能将一个窗口分为两个窗口。如果您复制窗口结构,则只需使用一个实例即可引用该窗口。

                    *(num_wins + i*4+j) = *sub_win;

在这里复制窗口。窗口现在位于两个位置,sub_winnum_wins数组中。您没有两个窗口,只有一个窗口。因此,您需要选择其中之一。

                    if(arr[i][j])
                            mvwprintw(sub_win,digity,digitx,"%d ",arr[i][j]);
                    else
                            mvwprintw(sub_win,"X ");

                    wrefresh(sub_win);

好的,所以您修改sub_win。这意味着您制作的副本不是窗口。请记住,只有一个窗口。由于您修改了sub_win,因此它是窗口。因此,在修改窗口之前复制窗口的目的是什么,不再是窗口了。

void destroy_list(WINDOW *num_wins){
        int length = 16;
        for(int i=0; i < length; i++){
                wprintw(stdscr,i);
                wborder((num_wins + i),' ',' ');
                wrefresh((num_wins + i));
                delwin((num_wins + i));
        };
};

您在这里做什么?您正在修改窗口之前对其进行操作的副本。但是您对其进行了修改-因此该窗口不再存在。这段代码没有意义。

一个窗口是一个东西,只有一个东西。如果制作WINDOW结构的副本,则可以使用任何一个副本来表示窗口。但是您不能同时使用两者-只有一个窗口。您的代码同时使用。那是无效的。

摘要:复制该窗口,然后修改该窗口(不复制)。这意味着该副本是您修改窗口之前的副本,没有 更长 存在,因为您修改了复制的对象。因此,尝试访问该副本是一个错误,因为它是不再存在的副本。

有点像这样做:

  1. 您拿一张白纸。
  2. 您要复制它。
  3. 您在原始纸上写“ 1”。
  4. 您尝试删除副本上的“ 1”。

复制后,您可以将任一面都视为原件,因为它们在 点相同。但是您也不能与对方混淆,否则您将感到非常困惑。