2

からパイプ/ターミナルを処理したいのですがstdin、同時に、UP/DOWN メニューなどの他の操作の処理を続行したいです。

fgetsパイプされた行の処理が終了し、画面がレンダリングされますが、wgetch応答しなくなるため、スイッチ内で何もできません。アプリケーションを閉じるには、私がしなければならないCTRL+c.

#include <curses.h>
#include <menu.h>

char *choices[] = { "Choice 1", "Exit" };

int main()
{  
   ITEM **my_items, *cur_item;
   MENU *my_menu;
   int i, c;

   initscr();
   cbreak();
   noecho();
   keypad(stdscr, TRUE);

   FILE *fp = stdin;
   char line [ 256 ]; 
   while ( fgets ( line, sizeof line, fp) != NULL ) {
      printw ( "%s", line);
   }

   my_items = (ITEM **)calloc(2 + 1, sizeof(ITEM *));
   for(i = 0; i < 2; ++i)
           my_items[i] = new_item(choices[i], choices[i]);
   my_items[2] = (ITEM *)NULL;

   my_menu = new_menu((ITEM **)my_items);
   mvprintw(LINES - 2, 0, "F1 to Exit");
   post_menu(my_menu);
   refresh();

   timeout(0);
   while((c = wgetch(stdscr)) != KEY_F(1))
   {   switch(c)
       { case KEY_DOWN:
              menu_driver(my_menu, REQ_DOWN_ITEM);
            break;
         case KEY_UP:
            menu_driver(my_menu, REQ_UP_ITEM);
            break;
      }
   }  

   endwin();
}
4

3 に答える 3

1

selectonを使用しfileno( stdin )て、stdin から何かを読み取る準備ができたことを通知してみてください。

例については、このリンクを参照してください。

また、stdin から読み取る準備ができているものを消費する必要があることにも注意してください。そうしないと、select が引き続き準備ができていることを示します。

また、これは Windows では機能しないことに注意してください。の使用を検討してくださいPeekConsoleInput

于 2010-12-12T01:06:29.210 に答える
1

fgets()のような stdio 関数と のようなcurses 関数を混在させるべきではありませんwgetch()

改行が入力されるまで文字を蓄積して、fgets()その呼び出しの代わりを内部的に記述します。wgetch()

于 2010-12-12T04:53:46.990 に答える
1

これにはいくつかの方法があります。私の意見では、より単純なものは次のとおりです。

  1. ノンブロッキングな方法で入力から読み取ります。

  2. I/O 多重化を使用します。Unix ではselect()poll()とそのいとこです。

  3. スレッドを起動し、stdin の読み取りをブロックします。

あなたに最適なものをグーグルで検索してください。周りにはたくさんの良い情報があります.

于 2010-12-12T02:00:01.013 に答える