C/C++ 多線程的學習心得總結

來源:互聯網
上載者:User

個人覺得在學習多線程編程之前最好先瞭解進程和線程的關係, 然後在學習線程工作方式的過程中動手寫個(我是從抄開始的)多線程的小程式, 會對學習多線程有很大的協助, 否則只有理論是很抽象的.

在學習多線程編程之前, 必須Crowdsourced Security Testing道什麼是 線程函數, 線程函數就是另一個線程的入口函數. 預設情況下一個我們所寫的代碼都是只有一個線程的, 而這個線程的入口函數就是main() 函數, 這是系統預設的. 而我們建立的另一個線程也需要一個函數來進入, 這個函數就叫做線程函數.

在C/C++中, 可以調用 '運行期庫' 函數 _beginthreadex(...), 來建立一個線程, _beginthreadex(...)函數接受6個參數, 其中第3個參數要求傳入該線程的入口函數的地址(即:傳入線程函數的函數名+&即可), 其它的每個參數的意思可以在msdn或網上查一下, 入門的話其餘參數全部傳0即可.

_beginthreadex(...)函數返回一個控制代碼, 就是新線程的控制代碼. 對於傳入的線程函數是有要求的, 線程函數必須返回 unsigned __stdcall 類型, 並且接受一個 void* 型參數, 函數的內容就可以自己定了.
先來段多線程的簡單代碼:複製代碼 代碼如下:#include<windows.h>
#include<process.h>
#include<iostream>
using namespace std;
bool stop;
unsigned Counter;
unsigned __stdcall thread(void*)
{
cout <<"In second thread..." <<endl;
while (!stop){
Sleep(200);
cout <<Counter++ <<" " <<flush;
}
//_endthreadex(0);
return 0;
}
int main()
{
HANDLE hThread;
unsigned int threadID;
stop = false;
cout <<"Creating second thread..." <<endl;
// Create the second thread.
hThread = (HANDLE)_beginthreadex(NULL, 0, &thread, NULL, 0, &threadID);
// Wait until second thread terminates. If you comment out the line
// below, Counter will not be correct because the thread has not
// terminated, and Counter most likely has not been incremented to
// 1000000 yet.
//WaitForSingleObject(hThread, INFINITE);
system("pause");
stop = true;
//cin >>stop;
cout <<"Counter is-> " <<Counter <<endl;
// Destroy the thread object.
CloseHandle(hThread);
system("pause");
return 0;
}

代碼是正確的, 複製粘貼到 Visual C++ 6.0 中, 即可編譯運行. 但是會發現 _beginthreadex(...) 函數未定義的編譯錯誤. 原因是因為我們的visual c++ 6.0 預設的是在單線程模式下編程, 如果需要進行多線程編程, 需要轉換一下編譯器的 '運行期庫' , 方法很簡單:
Project --> Settings --> C/C++ 中 選擇 Category 中的 Code generation , 然後在 Use run-time library 裡面選含有Multithread的其中一個就可以了.
這時再編譯, 就可以通過了. 有了這個例子的函數, 現在應該已經入門了吧. 不過除了這個還有很多基礎知識要補上, 推薦通讀幾遍 <windows核心編程(第四版)> 的第6章.
實際上, 在不同的編譯環境下, _beginthreadex(...)函數可能會有不同的名稱, 但是它們都是對 windows函數 CreateThread(...) 的封裝, CreateThread(...)函數用來建立一個新的線程函數, CreateThread(...)函數同樣接受6個參數, 在類似 _beginthreadex(...)的函數中對其做了一些安全的處理, 如堆棧記憶體的申請等. 雖然可以用CreateThread函數來建立一個新的線程, 但是強烈建議使用 _beginthreadex(...)函數, 對於CreateThread 函數在<windows核心編程(第四版)>第6章中有很好的講解.
當一個線程的任務結束, 要退出時, 有四種方法:
1.線程函數返回(最好使用這種方法);
2.通過調用 _endthreadex()或 ExitThread()函數,線程將自行撤消(最好不要使用這種方法);
3.同一個進程或另一個進程中的線程調用 TerminateThread()函數(應該避免使用這種方法);
4.包含線程的進程終止運行(應該避免使用這種方法).
最好通過其入口函數的返回語句(即:return)來退出線程, 也可以通過調用 C/C++ '運行期庫'函數 _endthreadex()函數來退出, 還有兩種退出的方法, 都是類似的強制退出. 最好使用線程函數的返回語句(return)來退出線程, 只有這樣才能安全的回收該線程的處理器資源和記憶體資源. 而實際上 _endthreadex()函數和 _beginthreadex(...)函數一樣, 是封裝了windows函數 ExitThread(), 如果一定要強制退出線程, 那麼強烈建議調用 _endthreadex() 函數, 這樣能安全的回收系統資源.

具體的 _beginthreadex(...)函數 和 _endthreadex()函數在<windows核心編程(第四版)>第6章中有很好的剖析. 同時還有兩個 _beginthread(...) 和 _endthread() 運行期庫函數, 也有講解.
windows還提供了一些庫函數用來獲得當前進程或者線程的控制代碼, 如 HANDLE GetCurrentProcess() 函數返回當前進程的控制代碼, HANDLE GetCurrentThread() 函數返回當前線程的控制代碼, 但是需要注意的是, 這些控制代碼都是 '偽控制代碼' , 即: 只在本進程和本線程內可用, 不可傳出取用.

如果需要在外部使用其它線程或進程的控制代碼, 則可以用 DuplicateHandle(...) 函數獲得其它進程或者線程的 '實控制代碼' , 該函數接受7個參數, 具體的用法可以查閱一下msdn和網路. 需要注意的是, 在使用完 由DuplicateHandle(...) 函數獲得的控制代碼後, 需要使用 CloseHandle() 函數來關閉該控制代碼.

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.