• <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的理想與現(xiàn)實(shí)

            創(chuàng)新+實(shí)踐

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

            常用鏈接

            留言簿(8)

            我參與的團(tuán)隊(duì)

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

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

                    需要注意的是,我們必須搞清楚線程是在何時(shí)創(chuàng)建消息隊(duì)列的,微軟如是說:

            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.

                    看來,我們只需要在線程初始化時(shí)調(diào)一句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 閱讀(2865) 評(píng)論(1)  編輯 收藏 引用 所屬分類: FlagshipEngine

            Feedback

            # re: 3D引擎多線程:框架 2009-01-11 23:56 Corner Zhang
            很有參考價(jià)值!贊一個(gè)  回復(fù)  更多評(píng)論
              

            久久久久青草线蕉综合超碰| 久久亚洲精品无码AV红樱桃| 久久久久久亚洲精品不卡| 久久国产成人午夜AV影院| 欧美日韩精品久久免费| 国内精品久久久久久久97牛牛| 亚洲伊人久久大香线蕉苏妲己| 久久精品国产精品亚洲精品| 国产亚洲综合久久系列| 久久久久无码专区亚洲av| 久久久久亚洲AV无码专区体验| 久久国产影院| 久久91精品久久91综合| 亚洲色婷婷综合久久| 亚洲精品高清一二区久久| 老司机国内精品久久久久| 久久w5ww成w人免费| 亚洲国产精品无码久久久久久曰| 99久久久精品免费观看国产| 亚洲精品乱码久久久久久蜜桃不卡 | 久久综合色之久久综合| WWW婷婷AV久久久影片| 亚洲国产精品无码久久98| 久久这里有精品视频| 老司机国内精品久久久久| 成人久久久观看免费毛片| 国产产无码乱码精品久久鸭| 综合久久国产九一剧情麻豆| 久久毛片一区二区| 亚洲国产日韩综合久久精品| 性做久久久久久久久久久| 香蕉久久永久视频| 亚洲欧美一区二区三区久久| 热综合一本伊人久久精品 | 久久国产精品免费一区二区三区| 国产精品久久久久久搜索| 九九精品99久久久香蕉| 久久综合给合久久狠狠狠97色| 久久亚洲春色中文字幕久久久| 久久精品中文无码资源站 | 武侠古典久久婷婷狼人伊人|