两个Windows - 一个由线程随机输出修改的

时间:2015-04-09 17:52:51

标签: c++ multithreading c++11 ncurses

我试图编写代码,其中屏幕分为两个窗口,其中一个由不同的线程修改,但输出似乎非常随机。有人可以帮忙吗?上部控制台应由main修改,下部由线程k修改。

#include <stdio.h>
#include <ncurses.h>
#include <unistd.h>
#include <thread>
#define WIDTH 30
#define HEIGHT 10 

int startx = 0;
int starty = 0;
void kupa (int score_size, int parent_x, int parent_y)
{
    int i = 0;
    WINDOW *dupa = newwin(score_size, parent_x, parent_y - score_size, 0);
    while(true)
    {

        i++;

        mvwprintw(dupa, 0 , 0, "You chose choice %d with choice string", i);
        wrefresh(dupa);  
        sleep(5);
        wclear(dupa);
    }
    delwin(dupa);
}
int main ()
{
      int parent_x, parent_y;
      int score_size =10;
      int counter =0 ;
      initscr();
      noecho();
      curs_set(FALSE);
      getmaxyx(stdscr, parent_y, parent_x);
      WINDOW *field = newwin(parent_y - score_size, parent_x, 0, 0);
      std::thread k (kupa, score_size, parent_x, parent_y);
      while(true) {
          mvwprintw(field, 0, counter, "Field");
          wrefresh(field);
          sleep(5);
          wclear(field);
          counter++;
      }
      k.join();
      delwin(field);
}

2 个答案:

答案 0 :(得分:3)

底层的curses / ncurses库不是线程安全的(参见例如What is meant by “thread-safe” code?讨论该术语)。对于curses,这意味着库的WINDOW结构(如stdscr)是全局变量,不受互斥或其他方法的保护。该库还具有内部静态数据,这些数据在Windows之间共享。您只能使用以下策略之一获得多线程代码的可靠结果:

  • 在一个线程中执行所有窗口管理(包括输入)
  • 使用互斥锁,信号量或任何concurrency技术似乎最好管理“拥有”单独窗口的单独线程。要在这里取得成功,线程必须在等待输入时从curses库阻止的点拥有整个屏幕,直到它更新屏幕并继续等待输入。这比听起来更难。

ncurses 5.7以及编译可以为reentrant code和一些线程应用程序提供基本支持。为此,它使用包裹在其静态数据中的互斥锁,将全局变量转换为“getter”函数,并添加显式传递许多调用中隐含的SCREEN指针的函数。有关详细信息,请参阅manual page

一些ncurses的测试程序说明了线程支持(这些是源代码的test子目录中的程序):

  • 同上显示use_screen
  • test_opaque执行WINDOW属性
  • 的“getters”
  • 下雨显示use_window
  • 蠕虫显示use_window

答案 1 :(得分:1)

我不确定你想做什么,但这种行为很正常。活动的线程写入窗口,当系统进行任务切换时,另一个线程写入窗口。正常行为是只使用一个写入窗口的线程。其他线程应该只做一些工作。 无论如何,如果您使用多个线程,则必须使用事件,互斥体,队列,信号量或其他方法来同步它们。