关于多任务和多线程

1、基本概念

多任务是操作系统可以执行多个程序的能力。操作系统使用硬件时钟为每个程序配置时间片段。Windows 16位版本支持有限度的多任务,Windows 32位版本支持真正的多任务,还有多线程。

多线程是在程序内部实现“多任务”。

DOS 系统对多任务没多大帮助,DOS 的主要版本是基于 8086 和 8088 芯片的能力而设计的,而这些芯片的并非为多任务而设计,部分原因是内存管理不够强。而当启动和结束多个程序时,多任务操作系统需要移动内存块以收集空闲内存。不过有创意的程序员还是找到了一些办法,大多使用常驻程序,如背景打印队列程序,通过拦截硬件时钟中断来执行真正的背景处理。

Windows 1.0 已可以在物理内存中移动内存块,这是多任务的前提,虽然移动方法尚未完全对应用程序透明,但已可忍受。早期的 Windows 多任务还是非优先权式的多任务,工作切换都发生在程序完成对消息的处理后将控制权返回给 Windows 时。这也被称为“合作式的多任务”,因为它要求应用程序方面的一些合作,一个 Windows 程序可以占用整个系统,如果它要花很长时间来处理消息的话。

32位版本的 Windows 支持非序列化的消息队列,这可以避免让一个应用程序占用整个系统。(这点我也没看懂)

在多线程中,程序可以把自己分割成同时执行的片段(即执行绪)。

一个线程简单地表示为可以呼叫程序中其它函数的函数。程序从其主线程开始执行,这个主执行绪是在传统的 C 程序中叫作 main 的函数,在 Windows 中是 WinMain。一旦执行起来,程序可以通过 CreateThead 创建新线程,线程间优先权式切换。

一种多线程架构:主线程处理使用者输入消息,并建立其它线程, 这些附加的线程只进行一些背景处理,除了和主线程通讯,不和使用者交流。就像老板和职员,老板把大的工作丢给职员处理,自己保持和外界的联系。

线程共享程序的内存,所以它们共享静态变量。但它们都有自己的堆栈,因此动态变量对每个栈程是唯一的。每个线程还有自己的处理器状态,这个状态在线程切换期间被储存和恢复。

正确地设计一个复杂的多线程应用程序是 Windows 程序员可能遇到的最困难的工作, 因为优先权式多任务可以在任何时刻中断一个线程,切换控制权到另一个线程中。

多线程中一个常见的错误是“竞争状态”,为了帮助协调线程的活动,操作系统要求各种形式的同步,一种是同步信号(semaphore),它允许在程序代码某一点阻止一个线程的执行,直到另一个执行绪发信号让它继续,还有一种是临界区域(critical section),它是程序代码中不可中断的部分。

同步信号还可能产生死锁,两个线程互相阻止对方执行。

32位Windows版本(包括Windows NT和Windows 98)有一个非序列化的消息队列。这种实作似乎非常好:如果一个程序正在花费一段长时间处理一个消息,那么鼠标位于该程序的窗口上时,鼠标光标将呈现为一个时钟,但是当将鼠标移到另一个程序的窗口上时,鼠标光标将变为正常的箭头形状。只需按一下就可以将另一个窗口提到前面来。

1/10秒规则:一个消息队列线程处理任何消息都不应该超过1/10秒,任何花费更长时间的事情都应该在另一个线程中完成。

2、实战

建立新的线程的API函数是CreateThread,它的语法如下:

PHANDLE hThread = CreateThread (&security_attributes, dwStackSize, ThreadProc, pParam, dwFlags, &idThread) ;

第一个参数 &security_attributes 是指向SECURITY_ATTRIBUTES型态的结构的指针。在Windows 98中忽略该参数。在Windows NT中,它被设为NULL。

第二个参数 dwStackSize 是用于新线程的初始堆栈大小,默认值为0。在任何情况下,Windows根据需要动态延长堆栈的大小。

第三个参数 ThreadProc 是指向线程函数的指标。函数名称没有限制,但是必须以下列形式声明:

DWORD WINAPI ThreadProc (PVOID pParam) ;

第四个参数 pParam 为传递给ThreadProc的参数。这样主线程和从属线程就可以共享数据。

第五个参数 dwFlags 指定额外的标志来控制线程的创建,为0表示线程创建之后立即就可以进行调度,如果为CREATE_SUSPENDED则表示线程创建后暂停运行,这样它就无法调度,直到调用ResumeThread()。

第六个参数 &idThread 是一个指标,指向接受执行绪ID值的变量。

3、C++多线程编程必备知识

(1)CreateEvent

HANDLE CreateEvent(
LPSECURITY_ATTRIBUTES lpEventAttributes,
BOOL bManualReset,  //TRUE,使用ResetEvent()手动重置为无信号状态;FALSE,当一个等待线程被释放时,自动重置状态为无信号状态。
BOOL bInitialState, //指定事件对象的初始状态,当TRUE,初始状态为有信号状态;当FALSE,初始状态为无信号状态。
LPCSTR lpName
);
HANDLE hEvent = NULL;
hEvent = CreateEvent(NULL, TRUE, TRUE, NULL); //使用手动重置为无信号状态,初始化时为有信号状态
hEvent = CreateEvent(NULL, FALSE, TRUE, NULL); //线程释放后自动重置为无信号状态,初始化时为有信号状态
hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); //使用手动重置为无信号状态,初始化时为无信号状态
hEvent = CreateEvent(NULL, FALSE, FALSE, NULL); //线程释放后自动重置为无信号状态,初始化时为无信号状态

(2)WaitForMultipleObjects

DWORD WaitForMultipleObjects(
  DWORD nCount,             //句柄的数量 最大值为MAXIMUM_WAIT_OBJECTS(64)
  CONST HANDLE *lpHandles,  // 句柄数组的指针,类型可以为(Event,Mutex,Process,Thread,Semaphore )数组
  BOOL fWaitAll,            // 等待的类型,如果为TRUE 则等待所有信号量有效在往下执行,FALSE 当有其中一个信号量有效时就向下执行
  DWORD dwMilliseconds      // 超时时间 超时后向执行。 如果为WSA_INFINITE 永不超时。如果没有信号量就会在这死等。
);

(3)CreateSemaphore

HANDLE CreateSemaphore(
         PSECURITY_ATTRIBUTE psa, //安全控制,一般传入NULL
         LONG lInitialCount, //设置信号量的初始计数
         LONG lMaximumCount, //设置信号量的最大计数
         PCTSTR pszName //指定信号量对象的名称
         );

HANDLE OpenSemaphore(
         DWORD  dwDesiredAccess,
         BOOL bInheritHandle, //如果允许子进程继承句柄,则设为TRUE
         PCTSTR pszName //指定要打开的对象的名字
);

HANDLE ReleaseSemaphore(
         HANDLE hSemaphore, //信号量句柄
         LONG lReleaseCount, //把lReleaseCount的值增加到当前资源技术上
         PLONG plPreviousCount //返回当前资源计数的原始值
);

CloseHandle()  清理信号量

参考资料:

1、Charles Petzold. 《Windows程序设计》

2、c++CreateEvent函数在多线程中使用及实例:http://blog.csdn.net/richerg85/article/details/7471426

3、秒杀多线程:http://blog.csdn.net/morewindows/article/details/7421759
原文链接: https://www.cnblogs.com/NaughtyBaby/p/4693612.html

欢迎关注

微信关注下方公众号,第一时间获取干货硬货;公众号内回复【pdf】免费获取数百本计算机经典书籍

原创文章受到原创版权保护。转载请注明出处:https://www.ccppcoding.com/archives/220030

非原创文章文中已经注明原地址,如有侵权,联系删除

关注公众号【高性能架构探索】,第一时间获取最新文章

转载文章受原作者版权保护。转载请注明原作者出处!

(0)
上一篇 2023年2月13日 上午10:45
下一篇 2023年2月13日 上午10:45

相关推荐