c++ 多线程学习

原创 2015年11月18日 17:49:57

最近在研究实时处理问题,学习了相关多线程的知识,这里进行简单的学习总结。

1.多线程能干啥
很多时候,一想到提高效率问题,我们便会想到多线程,那么多线程一定能提高效率吗?如果能,在什么情况下能够提高效率?
当有多个cpu的情况下,计算机可以同时处理多个事件,多线程的处理效率会快很多。如果一个任务可以细分为多个子任务,每个都需要cpu资源和内存资源。如果是单线程的话,就只能等处理完一个子任务的时候才能接下去处理下一个子任务。而在多线程的情况下,可以在第一个子任务用完cpu资源的时候,假如此时他只需要io资源(举个例子,比如说读写文件),这个时候在等待他处理的时候,可以同时处理第二个任务。
但当只有一个cpu的时候,多线程可能快也可能慢。可能慢的原因是系统在多线程之间的切换占据了一定的时间,反而增加了时耗。可能快的原因是合理利用多线程的特点,同时做不冲突的事,比如一个线程在读写文件,一个线程在处理。

2.多线程简单实现
首先,认识一下多线程创建函数

HANDLE CreateThread( 
    LPSECURITY_ATTRIBUTES lpThreadAttributes, // SD
    SIZE_T dwStackSize,                     // initial stack size
    LPTHREAD_START_ROUTINE lpStartAddress,    // thread function
    LPVOID lpParameter,                       // thread argument
    DWORD dwCreationFlags,                    // creation option
    LPDWORD lpThreadId                       // thread identifier
);

结构中的参数介绍:
lpThreadAttributes:指向SECURITY_ATTRIBUTES型态的结构的指针,通常设为NULL;
dwStackSize:设置初始栈的大小,以字节为单位,通常设为0;
lpStartAddress:指向线程函数的指针,形式:@函数名,函数名称没有限制,但是必须以下列形式声明:
DWORD WINAPI 函数名 (LPVOID lpParam) ,格式不正确将无法调用成功。
lpParameter:向线程函数传递的参数,是一个指向结构的指针,不需传递参数时,为NULL。
dwCreationFlags :线程标志。取0时表示创建立即激活线程;
lpThreadId:保存新线程的id,通常设为NULL。

下面实现一个简单的多线程例子。

#include <iostream>
#include <Windows.h>

using namespace std;

DWORD WINAPI DISPLAY1(LPVOID lpParameter)
{
    while(1)
    {
        cout<<"1"<<endl;
        Sleep(1000);

    }

}

DWORD WINAPI DISPLAY2(LPVOID lpParameter)
{
    while(1)
    {
        cout<<"2"<<endl;
        Sleep(1000);

    }

}

int main()
{
    HANDLE pThread1=CreateThread(NULL,0,DISPLAY1,NULL,0,NULL);
    HANDLE pThread2=CreateThread(NULL,0,DISPLAY2,NULL,0,NULL);

    CloseHandle(pThread1);
    CloseHandle(pThread2);

    while(1)
    {
        cout<<"0"<<endl;
        Sleep(1000);

    }

    return 0;
}

出现如下结果:
这里写图片描述
可以发现,输出行有时会存在换行和输出的交错,主要的原因是因为多线程并发运行时,多个线程之间如果公用了一些资源的话,我们并不能保证这些资源都能正确地被利用,因为这个时候资源并不是独占的。而这里的endl就是公用资源。如何解决这一问题呢?这里提供两种解决方案。
(1)将cout<<"0"<<endl换成`cout<<”0\n”;
通过多线程不共用资源的方式,解决了输出紊乱的问题;

(2)利用多线程同步
对于一个资源被多个线程共用会导致程序的混乱,我们的解决方法是只允许一个线程拥有对共享资源的独占,这样也能够解决上面的问题了。`

HANDLE CreateMutex(
    LPSECURITY_ATTRIBUTES lpMutexAttributes,  // SD
    BOOL bInitialOwner,                       // initial owner
    LPCTSTR lpName                            // object name
 );

该函数用于创造一个独占资源,第一个参数我们没有使用,可以设为NULL,第二个参数指定该资源初始是否归属创建它的进程,第三个参数指定资源的名称。

HANDLE hMutex = CreateMutex(NULL,TRUE,"screen"); 

这条语句创造了一个名为screen并且归属于创建它的进程的资源。

BOOL ReleaseMutex(
    HANDLE hMutex   // handle to mutex
  );

该函数用于释放一个独占资源,进程一旦释放该资源,该资源就不再属于它了,如果还要用到,需要重新申请得到该资源。申请资源的函数如下:

DWORD WaitForSingleObject(
    HANDLE hHandle,        // handle to object
    DWORD dwMilliseconds   // time-out interval
  );

第一个参数指定所申请的资源的句柄,第二个参数一般指定为INFINITE,表示如果没有申请到资源就一直等待该资源,如果指定为0,表示一旦得不到资源就返回,也可以具体地指定等待多久才返回,单位是千分之一秒。
下面程序是对上面例子的修改:

#include <iostream>
#include <Windows.h>
HANDLE hMutex;

using namespace std;

DWORD WINAPI DISPLAY1(LPVOID lpParameter)
{
    while(1)
    {
        WaitForSingleObject(hMutex,INFINITE);
        cout<<"1"<<endl;
        Sleep(1000);
        ReleaseMutex(hMutex);

    }

}

DWORD WINAPI DISPLAY2(LPVOID lpParameter)
{
    while(1)
    {
        WaitForSingleObject(hMutex,INFINITE);
        cout<<"2"<<endl;
        Sleep(1000);
        ReleaseMutex(hMutex);       

    }

}

int main()
{
    HANDLE pThread1=CreateThread(NULL,0,DISPLAY1,NULL,0,NULL);
    HANDLE pThread2=CreateThread(NULL,0,DISPLAY2,NULL,0,NULL);
    hMutex=CreateMutex(NULL,FALSE,NULL);

    CloseHandle(pThread1);
    CloseHandle(pThread2);

    while(1)
    {
        WaitForSingleObject(hMutex,INFINITE);
        cout<<"0"<<endl;
        Sleep(1000);
        ReleaseMutex(hMutex);

    }

    return 0;
}
版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

C++多线程学习

  • 2012-11-03 11:57
  • 33KB
  • 下载

c++多线程学习7信号

多线程同步之Semaphore (主要用来解决生产者/消费者问题) 一 信标Semaphore 信标内核对象用于对资源进行计数。它们与所有内核对象一样,包含一个使用数量,但是它们也包含另外两个...

C++/GUI/Qt学习——多线程

多线程是解决处理数据密集型的事件时,同时要需要响应用户的其它操作的一种很好的方法。 (一)创建线程 在QT应用程序中提供多线程是非常简单的,只需要子类化QThread并且重新实现它的run()函数...

c++thread学习(3)并发多线程系统开发过程

在实现一个规模较大的工程时,需要多个线程之间互相传递数据,因此,点对点的通信机制已经不能满足需要,设定一个比较合理的通信架构会提高工程开发、维护的效率以及代码的复用性,可读性。 了解ROS的小伙伴都知...

JAVA学习(13) JAVA/C++互相调用 以及多线程

#include #ifdef __cplusplus extern "C" { #endif /* * Class: com_ui_test_test * Metho...

C++基础学习之16 - 多线程与互斥锁

熟悉Linux的童鞋都对多进程比较熟,Linux下的 fork也有很多教程介绍,但这不是我们本节的重点,我们在这里主要讲的是多线程。         相对于进程来讲,线程 是一个轻量级的概念,一个进程...

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

http://www.haogongju.net/art/326731 个人觉得在学习多线程编程之前最好先了解进程和线程的关系, 然后在学习线程工作方式的过程中动手写个(我是从抄开始的)...

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

这是我第一次写技术博客, 有点紧张(呵呵...~), 我就从写学习多线程编程的心得开始吧.   个人觉得在学习多线程编程之前最好先了解进程和线程的关系, 然后在学习线程工作方式的过程中动手写个(我是...

Linux C/C++多线程学习:生产者消费者问题

生产者消费者问题 多个生产者和多个生产者的问题。生产者不断的向仓库放入产品,消费者不断的从仓库取出产品,仓库的容量是有限的。因此,当仓库处于满状态时,生产者必须等待消费者取出 1 个或多个产品后才能...

C++多线程学习---线程间的共享数据

多线程间的共享数据如果不加以约束是有问题的。最简单的方法就是对数据结构采用某种保护机制,通俗的表达就是: 确保只有进行修改的线程才能看到不变量被破坏时的中间状态。从其他访问线程的角度来看,修改不是已经...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)