青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

#ant

The dreams in which I'm dying are the best I've ever had...

從Win32 API封裝Thread類[1]

前幾天在學(xué)Windows多線程程序設(shè)計(jì),發(fā)現(xiàn)Win32 API用起來確實(shí)不怎么方便,特別是對(duì)于C++程序員。于是實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的封裝,技術(shù)含量當(dāng)然不高,不過用起來還是比較方便的。如果你熟悉Java,你會(huì)發(fā)現(xiàn)這個(gè)實(shí)現(xiàn)有點(diǎn)像Java的Thread,在Java中有兩種方法可以創(chuàng)建一個(gè)Thread:

1.從Thread類繼承并實(shí)現(xiàn)run方法:

1?class?MyThread?extends ?Thread?{
2?????public?void
?run()?{
3?
????????...
4?
????}?
5?
};
6?

7?//開啟線程
8?MyThread?thread?=?new ?MyThread;
9?thread.start();

2.聲明一個(gè)類實(shí)現(xiàn)Runnable接口并實(shí)現(xiàn)run方法:

1?class?MyRunnable?implements ?Runnable?{
2?????public?void
?run()?{
3?
????????...
4?
????}
5?
};
6?

7?//?開啟線程?
8?MyThread?thread?=?new?MyThread(new ?MyRunnable);
9?thread.start();


具體實(shí)現(xiàn)
Java的實(shí)現(xiàn)方式還算優(yōu)雅,我們也可以在C++中模擬,由于篇幅所限,在這里我們只關(guān)注基本功能的實(shí)現(xiàn):
thread.h
?1?#ifndef?THREAD_H
?2?
#define?THREAD_H
?3?

?4?#include?<windows.h>
?5?
?6?#define?CLASS_UNCOPYABLE(classname)?\
?7?????private
:?\
?8?????classname(const?classname&
);?\
?9?????classname&?operator=(const?classname&
);
10?

11?struct?Runnable?{
12?????virtual?void?run()?=?0
;
13?????virtual?~
Runnable()?{}
14?
};
15?

16?class?Thread?:?public?Runnable?{
17?
????CLASS_UNCOPYABLE(Thread)
18?public
:
19?????explicit?Thread(Runnable* target?=?0
);
20?????virtual?~
Thread();
21?????virtual?void
?run()?{}
22?????void
?start();
23?????void
?join();
24?private
:
25?????static?unsigned?__stdcall?threadProc(void*
param);
26?private
:
27?????Runnable*
_target;
28?
????HANDLE?_handle;
29?
};
30?

31?#endif/*THREAD_H*/

在上面的代碼中我們先定義一個(gè)Runnable類,并為Runnable類添加純虛函數(shù)run,需要特別注意的是Runnable類的虛析構(gòu)函數(shù),任何想要成為基類的class都應(yīng)該聲明析構(gòu)函數(shù)為virtual。
Thread類直接繼承自Runnable,CLASS_UNCOPYABLE(Thread) 這一行用宏聲明Thread類不可拷貝,具體細(xì)節(jié)請(qǐng)看 考慮用Macro替換Uncopyable 。start成員函數(shù)和Java中的一樣,用來開啟一個(gè)線程,join成員函數(shù)用來阻塞當(dāng)前線程直到開啟的線程執(zhí)行完畢。threadProc作為靜態(tài)成員函數(shù)用來傳遞給_beginthreadex,至于為什么用_beginthreadex代替CreateThread請(qǐng)問Google。初步的實(shí)現(xiàn)都比較簡(jiǎn)單,直接貼上代碼:
thread.cpp
?1?#include?"thread.h"
?2?#include?<process.h>?// for?_beginthreadex
?3?
?4?Thread::Thread(Runnable* target?/*=?0*/)
?5?:_target(target
)
?6?,_handle(0
)?{
?7?

?8?}
?9?

10?Thread::~Thread()?{
11?????if?(_handle?!=?0
)
12?
????????CloseHandle(_handle);
13?????if?(_target?!=?0
)
14?
????????delete?_target;
15?
}
16?

17?void?Thread::start()?{
18?????if?(_handle?!=?0
)
19?????????return
;
20?
????unsigned?id;
21?????_handle?=?reinterpret_cast<HANDLE>
(
22?????????_beginthreadex(0,?0,?threadProc,?this,?0,?&
id)
23?
????????);
24?
}
25?

26?void?Thread::join()?{
27?????if(_handle?!=?0
)?{
28?
????????WaitForSingleObject(_handle,?INFINITE);
29?
????????CloseHandle(_handle);
30?????????_handle?=?0
;
31?
????}
32?
}
33?

34?unsigned?__stdcall?Thread::threadProc(void* param)?{
35?????Thread* p?=?static_cast<Thread*>
(param);
36?????if?(p->_target?!=?0
)
37?????????p->_target->
run();
38?????else

39?????????p->run();
40?????return?0
;
41?}

下面是測(cè)試代碼:
test.cpp
?1?#include?"thread.h"
?2?#include?<iostream>
?3?
?4?using?namespace?std;
?5?

?6?//第一種方法,從Thread類繼承
?7?struct?MyThread?:?public?Thread?{
?8?????virtual?void
?run()?{
?9?????????for?(int?i?=?0;?i?<?5;?++
i)?{
10?????????????cout?<<?"MyThread?Running..."?<<?i?<<
?endl;
11?????????????Sleep(100
);
12?
????????}
13?
????}
14?
};
15?

16?//第二種方法,“實(shí)現(xiàn)”Runnable接口
17?struct?MyRunnable?:?public?Runnable?{
18?????virtual?void
?run()?{
19?????????for?(int?i?=?0;?i?<?5;?++
i)?{
20?????????????cout?<<?"MyRunnable?Running..."?<<?i?<<
?endl;
21?????????????Sleep(300
);
22?
????????}
23?
????}
24?
};
25?

26?int?main()?{
27?

28?????MyThread?thread1;
29?????Thread?thread2(new
?MyRunnable());
30?
????thread1.start();
31?
????thread2.start();
32?
????thread1.join();
33?
????thread2.join();
34?

35?????return?0;
36?}

可能的運(yùn)行結(jié)果:
MyThread Running...0
MyRunnable Running...0
MyThread Running...1
MyThread Running...2
MyRunnable Running...1
MyThread Running...3
MyThread Running...4
MyRunnable Running...2
MyRunnable Running...3
MyRunnable Running...4


如果你熟悉boost庫,你會(huì)知道boost庫的Thread構(gòu)造函數(shù)可以接受普通函數(shù)和函數(shù)對(duì)象作為參數(shù),如果你覺得從Thread類繼承或者實(shí)現(xiàn)Runnable接口還不夠簡(jiǎn)潔,下一篇會(huì)有一個(gè)比較好的改進(jìn)。

posted on 2007-08-30 10:18 螞蟻終結(jié)者 閱讀(4648) 評(píng)論(8)  編輯 收藏 引用 所屬分類: C++

Feedback

# re: 從Win32 API封裝Thread類[1] 2007-08-31 15:18 重劍

受益匪淺啊   回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-02-26 17:51 sd

太簡(jiǎn)單了  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-07-25 22:46 Robinfoxnan

看了樓主的TEA,和strlen覺得分析的很好,
但是,關(guān)于多線程,
在侯捷的《windows多線程程序設(shè)計(jì)》中說,建議不使用_beginthreadex之類純API函數(shù),原因是在使用了線程本地存儲(chǔ)的函數(shù),比如GetLastError()之類的函數(shù)后,運(yùn)行庫比如VC++,會(huì)幫你建立一些我們看不到的結(jié)構(gòu)體,會(huì)容易導(dǎo)致內(nèi)存泄露,建議使用AfxBeginThread()類,特定開發(fā)環(huán)境提供的函數(shù)或者類,比如CWinThread。
可以參考該書第8章!  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-10-10 01:23 OwnWaterloo

@Robinfoxnan
我?guī)蜆侵骰啬懔?~~

侯捷的書, 肯定說的是不要使用 CreateTread 這個(gè)API
如果他說的是不要使用 _beginthreadex , 那他該被拖出去打。

至于AfxBeginThread ,windows平臺(tái)下寫程序一定要和MFC扯上關(guān)系么?

既然你提到了AfxBeginThread, 你可以去看一下MFC的源碼, MFC80, 它就是使用的 _beginthreadex ~~

  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-10-10 01:36 OwnWaterloo

我有2個(gè)建議, 一個(gè)可以慎重考慮一下, 另外一個(gè)可以完全不理睬。
還有2個(gè)疑問 ……

1.單下劃線開始的標(biāo)識(shí)符?

2.將windows.h 從thread.h中移走
.h
class Thread {
//
private:
uintptr_t handle_;
}

.cpp
handle_ = _beginthreadex 這個(gè)可以直接賦值
CloseHandle( reinterpret_cast<HANDLE>( handle_ ); 這個(gè)要自己轉(zhuǎn)
雖然麻煩一點(diǎn), 但是將 windows.h 隔離到了頭文件之外




疑問:
1. Thread的析構(gòu)

10 Thread::~Thread() {
11 if (_handle != 0)
12 CloseHandle(_handle);
13 if (_target != 0)
14 delete _target;
15 }

CloseHandle 線程并不停止
如果主線程沒有join delete _target; 后 線程還在run

這是線程函數(shù)
34 unsigned __stdcall Thread::threadProc(void *param) {
35 Thread *p = static_cast<Thread*>(param);
36 if (p->_target != 0)
37 p->_target->run();
38 else
39 p->run();
40 return 0;
41 }

37 行, 如果被delete, 而且run訪問了數(shù)據(jù), 而不僅僅是打印一些console消息, 肯定會(huì)引發(fā)錯(cuò)誤。


疑問是: join 是慣例? 規(guī)范? 還是僅僅是boost的示例而已。


2.Thread 似乎并不需要從 Runable 繼承。
完全可以設(shè)計(jì)成 Thread是Runable的容器。

疑問是: 僅僅是為了模仿Java的行為?
  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-10-10 09:15 螞蟻終結(jié)者

@Robinfoxnan
OwnWaterloo回答的也是我想回答的,我就不多說了。  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1] 2008-10-10 09:35 螞蟻終結(jié)者

@OwnWaterloo
建議:
1.單下劃線開始的標(biāo)識(shí)符?
這個(gè)是編程風(fēng)格問題,很難說出哪個(gè)更好哪個(gè)更差,正所謂蘿卜白菜各有所愛,個(gè)人認(rèn)為成員變量無論是下劃線開始(如:_handle)還是下劃線結(jié)尾(如:handle_),又或者是用匈牙利命名法,只要在編碼中始終保持一種風(fēng)格就好。編碼風(fēng)格對(duì)程序員來說可能是一種信仰。

2.將windows.h 從thread.h中移走
這個(gè)確實(shí)是可以考慮,如果考慮到跨平臺(tái)的話。

疑問:
1. Thread的析構(gòu)
據(jù)我所知,join應(yīng)該是等待線程結(jié)束的慣用法,Java.Thread,boost.Thread都通過join來保證線程的安全結(jié)束。是否是規(guī)范我也不清楚,貌似join是POSIX中線程取消規(guī)范中的一個(gè)?

2.Thread 似乎并不需要從 Runable 繼承
是不需要繼承Runnable的,這樣只是模仿Java的行為。
  回復(fù)  更多評(píng)論   

# re: 從Win32 API封裝Thread類[1][未登錄] 2013-06-08 17:19 Jackie

這里有個(gè)問題, 是 如果你沒有join的話, 線程對(duì)象就會(huì)被馬上析構(gòu), 程序出現(xiàn)崩潰.
所以我把線程對(duì)象創(chuàng)建在堆上面了, 然后在Run調(diào)用完之后在threadProc里面delete 完成  回復(fù)  更多評(píng)論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久资源av| 午夜一区在线| 国产精品一区久久久久| 欧美日韩在线另类| 欧美日韩午夜| 国产精品视区| 狠狠久久亚洲欧美| 亚洲国产高清在线观看视频| 91久久精品www人人做人人爽| 亚洲精品国产视频| 亚洲一区日韩| 久久一区二区三区国产精品| 欧美二区在线| 99国产一区二区三精品乱码| 亚洲在线视频免费观看| 久久精品视频免费| 欧美国产精品| 国产精品欧美精品| 韩曰欧美视频免费观看| 亚洲精品视频在线观看网站| 亚洲一区二区三区精品视频| 亚洲愉拍自拍另类高清精品| 午夜视频一区在线观看| 久久精品免费观看| 欧美福利在线观看| 日韩视频免费观看高清完整版| 一区二区三区精品久久久| 性欧美video另类hd性玩具| 久久一区二区三区四区| 欧美日韩午夜| 亚洲国产精品ⅴa在线观看 | 欧美一级免费视频| 奶水喷射视频一区| 亚洲一区二区三区四区五区黄| 久久久久国产精品人| 欧美午夜不卡视频| 最近看过的日韩成人| 欧美综合激情网| 亚洲精品一二| 美女图片一区二区| 国产专区精品视频| 午夜欧美不卡精品aaaaa| 亚洲国产成人精品视频| 香蕉视频成人在线观看| 欧美视频观看一区| 亚洲人成网在线播放| 久久天天综合| 欧美一级精品大片| 国产精品日韩欧美一区| 9久草视频在线视频精品| 免费在线观看精品| 久久久91精品国产| 国内精品久久久久久久果冻传媒| 亚洲免费在线电影| 亚洲视频在线一区观看| 欧美午夜久久| 亚洲午夜在线| 一本一本a久久| 欧美日韩美女| 亚洲一区二区三区四区中文| 亚洲精品免费一区二区三区| 美女主播视频一区| 亚洲精品乱码久久久久久| 免费视频一区| 欧美1区视频| 亚洲免费观看高清在线观看 | 狠狠久久亚洲欧美| 久久在精品线影院精品国产| 欧美在线视频一区二区| 国产一区二区三区直播精品电影 | 国产精品伦子伦免费视频| 亚洲——在线| 亚洲欧美日韩一区二区三区在线观看| 亚洲国产日韩欧美| 9人人澡人人爽人人精品| 欧美日韩的一区二区| 一区二区91| 亚洲桃色在线一区| 国产日本亚洲高清| 老司机精品视频网站| 免费成人高清| 亚洲午夜视频| 翔田千里一区二区| 在线视频国产日韩| 亚洲欧洲中文日韩久久av乱码| 欧美日韩三级| 久久久国产一区二区| 久久亚洲欧美| 亚洲日韩第九十九页| 亚洲人成网站在线观看播放| 国产精品国产自产拍高清av| 久久久精品欧美丰满| 美女网站久久| 亚洲欧美在线x视频| 久久精品国产第一区二区三区最新章节 | 国产农村妇女精品一区二区| 欧美成人精品在线视频| 欧美精品久久一区二区| 亚洲资源av| 久热综合在线亚洲精品| 亚洲自拍偷拍麻豆| 久久久久国产精品厨房| 夜夜精品视频| 久久久综合网站| 亚洲综合第一| 欧美fxxxxxx另类| 欧美一区二区三区免费在线看| 久久青青草原一区二区| 亚洲欧美日本精品| 免费观看30秒视频久久| 欧美一区二区播放| 欧美日韩日本视频| 欧美11—12娇小xxxx| 国产免费一区二区三区香蕉精| 亚洲第一区色| 一区二区三区在线观看国产| 亚洲午夜激情免费视频| 99国产精品久久久久久久| 久久久久久久激情视频| 欧美亚洲视频在线观看| 欧美日韩免费网站| 亚洲丶国产丶欧美一区二区三区| 国产一区二区激情| 一道本一区二区| 亚洲美女免费精品视频在线观看| 久久精品国产清高在天天线| 欧美一站二站| 国产精品日本| 中国女人久久久| 亚洲一区3d动漫同人无遮挡| 欧美伦理影院| 亚洲美女在线观看| 亚洲美女中文字幕| 女生裸体视频一区二区三区| 免费成人网www| 另类av导航| 国产日韩精品视频一区| 一区二区欧美视频| 一区二区三区|亚洲午夜| 欧美大片网址| 亚洲国产综合视频在线观看| 亚洲精品网址在线观看| 亚洲视频电影在线| 久久午夜色播影院免费高清| 国产精品人人做人人爽| 亚洲在线不卡| 久久激情综合| 韩国三级电影一区二区| 久久久久久有精品国产| 嫩草成人www欧美| 亚洲人成人一区二区三区| 欧美第十八页| 日韩一级精品视频在线观看| 亚洲一区二区三区视频播放| 国产精品女同互慰在线看| 性8sex亚洲区入口| 美女成人午夜| 日韩亚洲精品视频| 国产精品扒开腿做爽爽爽视频 | 欧美一区二区视频97| 国产视频久久网| 久久久av网站| 亚洲精品乱码久久久久久| 中文精品在线| 国产一区三区三区| 欧美大片免费看| 欧美国产在线观看| 午夜精品久久久久久久久久久| 国产精品视频免费在线观看| 久久精品亚洲一区二区| 亚洲国产婷婷香蕉久久久久久| 亚洲自拍三区| 在线播放豆国产99亚洲| 欧美黄色一区二区| 亚洲欧美日本另类| 欧美国产在线观看| 欧美一区二区三区免费大片| 亚洲成人原创| 国产精品一区二区在线观看| 欧美xxx在线观看| 亚洲一区二区三区成人在线视频精品| 欧美自拍偷拍午夜视频| 日韩视频精品| 永久91嫩草亚洲精品人人| 欧美天堂亚洲电影院在线观看 | 136国产福利精品导航| 亚洲一区不卡| 久久深夜福利| 国产日韩欧美亚洲| 国产偷国产偷精品高清尤物| 葵司免费一区二区三区四区五区| 在线视频日韩| 欧美国产精品| 久久久久久久一区二区三区| 在线一区二区三区四区五区| 伊人久久噜噜噜躁狠狠躁| 国产精品久久久久aaaa樱花| 美女视频网站黄色亚洲| 欧美成人精品在线播放| 国产精品日韩高清|