• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            flagship的理想與現實

            創新+實踐

              C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
              8 Posts :: 0 Stories :: 27 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(8)

            我參與的團隊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

                    現在我們已經有了三個可獨立工作的線程:資源加載線程、邏輯線程、渲染線程,下一步我們需要決定它們如何在實際的項目中相互配合,也就是所謂的應用程序框架了,該框架需要解決以下兩個問題
                    首先,資源讀取線程可以簡單設計為一個循環等待的線程結構,每隔一段時間檢查加載隊列中是否有內容,如果有則進行加載工作,如果沒有則繼續等待一段時間。這種方式雖然簡單清晰,但卻存在問題,如果等待時間設得過長,則加載會產生延遲,如果設得過短,則該線程被喚醒的次數過于頻繁,會耗費很多不必要的CPU時間。
                    然后,主線程是邏輯線程還是渲染線程?因為邏輯線程需要處理鍵盤鼠標等輸入設備的消息,所以我起初將邏輯線程設為主線程,而渲染線程另外創建,但實際發現,幀數很不正常,估計與WM_PAINT消息有關,有待進一步驗證。于是掉轉過來,幀數正常了,但帶來了一個新的問題,邏輯線程如何處理鍵盤鼠標消息?
                  
                    對于第一個問題,有兩種解決方案:
                    第一,我們可以創建一個Event,資源讀取線程使用WaitForSingleObject等待著個Event,當渲染線程向加載隊列添加新的需加載的資源后,將這個Event設為Signal,將資源讀取線程喚醒,為了安全,我們仍需要在渲染線程向加載隊列添加元素,以及資源加載線程從加載隊列讀取元素時對操作過程加鎖。
                    第二,使用在渲染線程調用PostThreadMessage,將資源加載的請求以消息的形式發送到資源價值線程,并在wParam中傳遞該資源對象的指針,資源加載線程調用WaitMessage進行等待,收到消息后即被喚醒,這種解決方案完全不需要加鎖。
                    對于第二個問題,我們同樣可以用PostThreadMessage來解決,在主線程的WndProc中,將邏輯線程需要處理的消息發送出去,邏輯線程收到后進行相關處理。

                    需要注意的是,我們必須搞清楚線程是在何時創建消息隊列的,微軟如是說:

            The thread to which the message is posted must have created a message queue, or else the call to PostThreadMessage fails. Use one of the following methods to handle this situation.

            • Call PostThreadMessage. If it fails, call the Sleep function and call PostThreadMessage again. Repeat until PostThreadMessage succeeds.
            • Create an event object, then create the thread. Use the WaitForSingleObject function to wait for the event to be set to the signaled state before calling PostThreadMessage. In the thread to which the message will be posted, call PeekMessage as shown here to force the system to create the message queue.
              PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE)
              Set the event, to indicate that the thread is ready to receive posted messages.

                    看來,我們只需要在線程初始化時調一句PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE)就可以了,然后在主線程中如此這般:

            switch ( uMsg )
                    
            {
                    
            case WM_PAINT:
                        
            {
                            hdc 
            = BeginPaint(hWnd, &ps);
                            EndPaint(hWnd, 
            &ps);
                        }

                        
            break;
                    
            case WM_DESTROY:
                        
            {
                            m_pLogic
            ->StopThread();
                            WaitForSingleObject( m_pLogic
            ->GetThreadHandle(), INFINITE );
                            PostQuitMessage(
            0);
                        }

                        
            break;
                    
            default:
                        
            {
                            
            if ( IsLogicMsg( uMsg ) )
                            
            {
                                PostThreadMessage( m_pLogic
            ->GetThreadID(), uMsg, wParam, lParam );
                            }

                            
            else
                            
            {
                                
            return DefWindowProc( hWnd, uMsg, wParam, lParam );
                            }

                        }

                        
            break;
                    }

                    在邏輯線程中這般如此:
            MSG msg;
                    
            while ( m_bRunning )
                    
            {
                        
            if ( PeekMessage( &msg, NULL, 00, PM_NOREMOVE ) )
                        
            {
                            
            if ( ! GetMessageW( &msg, NULL, 00 ) )
                            
            {
                                
            return (int) msg.wParam;
                            }


                            MessageProc( msg.message, msg.wParam, msg.lParam );
                        }


                        LogicTick();
                    }
                    完成!
            posted on 2009-01-10 23:37 flagship 閱讀(2853) 評論(1)  編輯 收藏 引用 所屬分類: FlagshipEngine

            Feedback

            # re: 3D引擎多線程:框架 2009-01-11 23:56 Corner Zhang
            很有參考價值!贊一個  回復  更多評論
              

            97久久精品无码一区二区天美| 午夜精品久久久久久久无码| 亚洲国产精品无码久久久不卡| 色欲久久久天天天综合网 | 理论片午午伦夜理片久久 | 精品精品国产自在久久高清| 精品久久久久久无码中文字幕一区 | 久久AAAA片一区二区| 日本加勒比久久精品| av无码久久久久久不卡网站| 久久精品国产一区二区三区| 久久亚洲日韩精品一区二区三区 | 国产欧美久久久精品影院| 久久久久亚洲AV无码麻豆| 久久久WWW成人免费精品| 国产精品禁18久久久夂久| 亚洲国产综合久久天堂| 丰满少妇人妻久久久久久4| 亚洲香蕉网久久综合影视| 国产精品久久久99| 97精品伊人久久大香线蕉app| 亚洲国产成人乱码精品女人久久久不卡| 狠狠色婷婷久久一区二区| 精品欧美一区二区三区久久久 | 久久国产精品国语对白| 成人资源影音先锋久久资源网| 婷婷久久综合九色综合绿巨人| 婷婷久久综合九色综合98| 色欲久久久天天天综合网| 国内精品久久久久久久久电影网| 青青青青久久精品国产h久久精品五福影院1421 | 97精品伊人久久大香线蕉| 久久婷婷五月综合成人D啪| 91精品观看91久久久久久| 久久97精品久久久久久久不卡| 亚洲精品国产美女久久久| 久久久久久久久久久精品尤物| 久久久久99这里有精品10| 久久综合亚洲鲁鲁五月天| 国产精品久久婷婷六月丁香| 久久久久人妻一区二区三区|