Update console without flickering - c++

前端 未结 3 1369
庸人自扰
庸人自扰 2020-12-04 13:33

I\'m attempting to make a console side scrolling shooter, I know this isn\'t the ideal medium for it but I set myself a bit of a challenge.

The problem is that whene

相关标签:
3条回答
  • 2020-12-04 14:21

    One method is to write the formatted data to a string (or buffer) then block write the buffer to the console.

    Every call to a function has an overhead. Try go get more done in a function. In your Output, this could mean a lot of text per output request.

    For example:

    static char buffer[2048];
    char * p_next_write = &buffer[0];
    for (int y = 0; y < MAX_Y; y++)
    {
        for (int x = 0; x < MAX_X; x++)
        {
            *p_next_write++ = battleField[x][y];
        }
        *p_next_write++ = '\n';
    }
    *p_next_write = '\0'; // "Insurance" for C-Style strings.
    cout.write(&buffer[0], std::distance(p_buffer - &buffer[0]));
    

    I/O operations are expensive (execution-wise), so the best use is to maximize the data per output request.

    0 讨论(0)
  • 2020-12-04 14:25

    Ah, this brings back the good old days. I did similar things in high school :-)

    You're going to run into performance problems. Console I/O, especially on Windows, is slow. Very, very slow (sometimes slower than writing to disk, even). In fact, you'll quickly become amazed how much other work you can do without it affecting the latency of your game loop, since the I/O will tend to dominate everything else. So the golden rule is simply to minimize the amount of I/O you do, above all else.

    First, I suggest getting rid of the system("cls") and replace it with calls to the actual Win32 console subsystem functions that cls wraps (docs):

    #define NOMINMAX
    #define WIN32_LEAN_AND_MEAN
    #include <Windows.h>
    
    void cls()
    {
        // Get the Win32 handle representing standard output.
        // This generally only has to be done once, so we make it static.
        static const HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);
    
        CONSOLE_SCREEN_BUFFER_INFO csbi;
        COORD topLeft = { 0, 0 };
    
        // std::cout uses a buffer to batch writes to the underlying console.
        // We need to flush that to the console because we're circumventing
        // std::cout entirely; after we clear the console, we don't want
        // stale buffered text to randomly be written out.
        std::cout.flush();
    
        // Figure out the current width and height of the console window
        if (!GetConsoleScreenBufferInfo(hOut, &csbi)) {
            // TODO: Handle failure!
            abort();
        }
        DWORD length = csbi.dwSize.X * csbi.dwSize.Y;
        
        DWORD written;
    
        // Flood-fill the console with spaces to clear it
        FillConsoleOutputCharacter(hOut, TEXT(' '), length, topLeft, &written);
    
        // Reset the attributes of every character to the default.
        // This clears all background colour formatting, if any.
        FillConsoleOutputAttribute(hOut, csbi.wAttributes, length, topLeft, &written);
    
        // Move the cursor back to the top left for the next sequence of writes
        SetConsoleCursorPosition(hOut, topLeft);
    }
    

    Indeed, instead of redrawing the entire "frame" every time, you're much better off drawing (or erasing, by overwriting them with a space) individual characters at a time:

    // x is the column, y is the row. The origin (0,0) is top-left.
    void setCursorPosition(int x, int y)
    {
        static const HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);
        std::cout.flush();
        COORD coord = { (SHORT)x, (SHORT)y };
        SetConsoleCursorPosition(hOut, coord);
    }
    
    // Step through with a debugger, or insert sleeps, to see the effect.
    setCursorPosition(10, 5);
    std::cout << "CHEESE";
    setCursorPosition(10, 5);
    std::cout 'W';
    setCursorPosition(10, 9);
    std::cout << 'Z';
    setCursorPosition(10, 5);
    std::cout << "     ";  // Overwrite characters with spaces to "erase" them
    std::cout.flush();
    // Voilà, 'CHEESE' converted to 'WHEEZE', then all but the last 'E' erased
    

    Note that this eliminates the flicker, too, since there's no longer any need to clear the screen completely before redrawing -- you can simply change what needs changing without doing an intermediate clear, so the previous frame is incrementally updated, persisting until it's completely up to date.

    I suggest using a double-buffering technique: Have one buffer in memory that represents the "current" state of the console screen, initially populated with spaces. Then have another buffer that represents the "next" state of the screen. Your game update logic will modify the "next" state (exactly like it does with your battleField array right now). When it comes time to draw the frame, don't erase everything first. Instead, go through both buffers in parallel, and write out only the changes from the previous state (the "current" buffer at that point contains the previous state). Then, copy the "next" buffer into the "current" buffer to set up for your next frame.

    char prevBattleField[MAX_X][MAX_Y];
    std::memset((char*)prevBattleField, 0, MAX_X * MAX_Y);
    
    // ...
    
    for (int y = 0; y != MAX_Y; ++y)
    {
        for (int x = 0; x != MAX_X; ++x)
        {
            if (battleField[x][y] == prevBattleField[x][y]) {
                continue;
            }
            setCursorPosition(x, y);
            std::cout << battleField[x][y];
        }
    }
    std::cout.flush();
    std::memcpy((char*)prevBattleField, (char const*)battleField, MAX_X * MAX_Y);
    

    You can even go one step further and batch runs of changes together into a single I/O call (which is significantly cheaper than many calls for individual character writes, but still proportionally more expensive the more characters are written).

    // Note: This requires you to invert the dimensions of `battleField` (and
    // `prevBattleField`) in order for rows of characters to be contiguous in memory.
    for (int y = 0; y != MAX_Y; ++y)
    {
        int runStart = -1;
        for (int x = 0; x != MAX_X; ++x)
        {
            if (battleField[y][x] == prevBattleField[y][x]) {
                if (runStart != -1) {
                    setCursorPosition(runStart, y);
                    std::cout.write(&battleField[y][runStart], x - runStart);
                    runStart = -1;
                }
            }
            else if (runStart == -1) {
                runStart = x;
            }
        }
        if (runStart != -1) {
            setCursorPosition(runStart, y);
            std::cout.write(&battleField[y][runStart], MAX_X - runStart);
        }
    }
    std::cout.flush();
    std::memcpy((char*)prevBattleField, (char const*)battleField, MAX_X * MAX_Y);
    

    In theory, that will run a lot faster than the first loop; however in practice it probably won't make a difference since std::cout is already buffering writes anyway. But it's a good example (and a common pattern that shows up a lot when there is no buffer in the underlying system), so I included it anyway.

    Finally, note that you can reduce your sleep to 1 millisecond. Windows will actually often sleep longer, typically up 15ms, but it will prevent your CPU core from reaching 100% usage with a minimum of additional latency.

    Note that this not at all the way "real" games do things; they almost always clear the buffer and redraw everything every frame. They don't get flickering because they use the equivalent of a double-buffer on the GPU, where the previous frame stays visible until the new frame is completely finished being drawn.

    Bonus: You can change the colour to any of 8 different system colours, and the background too:

    void setConsoleColour(unsigned short colour)
    {
        static const HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);
        std::cout.flush();
        SetConsoleTextAttribute(hOut, colour);
    }
    
    // Example:
    const unsigned short DARK_BLUE = FOREGROUND_BLUE;
    const unsigned short BRIGHT_BLUE = FOREGROUND_BLUE | FOREGROUND_INTENSITY;
    
    std::cout << "Hello ";
    setConsoleColour(BRIGHT_BLUE);
    std::cout << "world";
    setConsoleColour(DARK_BLUE);
    std::cout << "!" << std::endl;
    
    0 讨论(0)
  • 2020-12-04 14:26

    system("cls") is the cause of your problem. For updating frame your program has to spawn another process and then load and execute another program. This is quite expensive. cls clears your screen, which means for a small amount of the time (until control returns to your main process) it displays completely nothing. That's where flickering comes from. You should use some library like ncurses which allows you to display the "scene", then move your cursor position to <0,0> without modifying anything on the screen and redisplay your scene "over" the old one. This way you'll avoid flickering, because your scene will always display something, without 'completely blank screen' step.

    0 讨论(0)
提交回复
热议问题