C/C++ 多线程学习心得

来源:互联网 发布:js 删除元素节点 编辑:程序博客网 时间:2024/04/29 07:51
这是我第一次写技术博客, 有点紧张(呵呵...~), 我就从写学习多线程编程的心得开始吧.

  个人觉得在学习多线程编程之前最好先了解进程和线程的关系, 然后在学习线程工作方式的过程中动手写个(我是从抄开始的)多线程的小程序, 会对学习多线程有很大的帮助, 否则只有理论是很抽象的.

  在学习多线程编程之前, 必须先知道什么是 线程函数, 线程函数就是另一个线程的入口函数. 默认情况下一个我们所写的代码都是只有一个线程的, 而这个线程的入口函数就是main() 函数, 这是系统默认的. 而我们创建的另一个线程也需要一个函数来进入, 这个函数就叫做线程函数.

  在C/C++中, 可以调用 '运行期库' 函数 _beginthreadex(...), 来创建一个线程, _beginthreadex(...)函数接受6个参数, 其中第3个参数要求传入该线程的入口函数的地址(即:传入线程函数的函数名+&即可), 其它的每个参数的意思可以在msdn或网上查一下, 入门的话其余参数全部传0即可.

  _beginthreadex(...)函数返回一个句柄, 就是新线程的句柄. 对于传入的线程函数是有要求的, 线程函数必须返回 unsigned __stdcall 类型, 并且接受一个 void* 型参数, 函数的内容就可以自己定了.

  先来段多线程的简单代码:

Code
复制代码
#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() 函数来关闭该句柄.

  至此, 入门的基础知识应该有了.


  欢迎各位大牛 补充 + 讨论 ~~~~~!!!!

From zhuocheng (cnblogs.com/zhuocheng)

原创粉丝点击