循环创建多线程时保证参数的有效性

2024-04-11 00:32

本文主要是介绍循环创建多线程时保证参数的有效性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

循环创建多线程时保证参数的有效性    
当我们需要在一个循环中传递参数时,使用使用函数的方法一般都是:    
for(int   I=0;I <100;I++)
{                  
fun(I);           //使用函数传递i   
}    


每一个循环都会等待fun(I);函数执行完后再进行下一个循环。    
但是当我们需要这个循环中创建线程,并将I的参数传递给线程时,如依然使用以上方法,会造成什么情况呢?    
DWORD   WINAPI   ThreadFun(LPVOID   lpParam)
{                             //线程函数                  
Int   *I   =   (int   *)lpParam;                  Return   0;   
}        


int   I;    
for(I=0;I <100;I++)
{    
DWORD   dwThreadId;            HANDLE   hThread;             
hThread   =   CreateThread(NULL,0,ThreadFunc,&I,0,&dwThreadId);  
}    


好了,到这里我们就可以发现,在循环中,我们创建线程并传递的参数是I=1后,主程序有可能在执到下一次循环时,第一次的ThreadFun函数仍未执行,而此时的I已经等2了,如果ThreadFun再来调用               
Int   *I   =   (int   *)lpParam;语句时,显然不是我们想要的结果。        
解决此问题的一种方法,便是可以使用静态数组来保存所要传递的参数。
如下: 
   
int   I;    
static   int   nPara[100];           //此句需定义为全局    
for(I=0;I <100;I++)
{    
DWORD   dwThreadId;    
HANDLE   hThread;    
NPara[I]=I;                 //保存参数          
hThread   =   CreateThread(NULL,0,ThreadFunc,&nPara[I],0,&dwThreadId); 
}    
此时,所有参数均保存在nPara数组中,刚才的问题是解决了。


接下来又有了新的问题,让我们一起来看看吧:  
  1、如果需要创建的线程不止100,而是非常的大,而且我们也并不知道会有多少次循环的时侯。  
  2、如果我们需要传递的参数不单单只是一个int型的I,而是一个类。那么我们声明的时侯


(假设线程数量最大为65535)则:    
static   CMYClass   myClass[65535];   


编译之后,得到的文件将会堆上一大堆的垃圾。相信任何一位程序都不想看到自己的程序上面堆了一堆垃圾在上面吧。 那么,还有没有更好的办法解决呢。答案是一定的,这里,我就讲一下我自己常用的方法:动态创建对像传递参数。
  
一提到动态创建,我们自然会想到new   与   delete   ,
对了,我想说的也正是他们的使用。    
假设参数类型为:    
typedef   struct   _PARA
{  
int   I;            
DWORD   dwNumber;            
HWND   hOther;    
}Para;  


使用new在堆栈中申请一遍空间,在使用完后必需使用delete将其释放。    


int   I;    
for(I=0;I <100;I++)
{    
DWORD   dwThreadId;    
HANDLE   hThread;     
Para   *myPara   =   new   Para;  
MyPara-> I   =   I;   MyPara-> dwNumber   =   0   ;//自定   
MyPara   -> hOther   =   GetSafeHWnd();//当前窗体句柄           
hThread   =   CreateThread(NULL,0,ThreadFunc,myPara,0,&dwThreadId); //线程函数  
}           


DWORD   WINAPI   ThreadFun(LPVOID   lpParam)
{                             
//线程函数                  
Para   *myPara   =   (Para   *)lpParam;                  //执行其他功能
delete   []   myPara;                                 //释放                 
Return   0;   
}  
  
这样的话,也就不怕传递的参数多少与线程的数量太大了。另外如有需要的话可以加上一个线程计数器,保证当前线程的最大数量。   
通常情况,我比较喜欢把线程处理放在一个类中处理,在主程序中尽量不与线程打交道。        


在循环中使用new 可以保证每次申请到的内存的地址不会是同一个(即不会重复),从而传递给每个线程的参数也互不冲突,互不干扰,相互独立,在线程内部使用完参数后,根据接收的地址,使用delete释放申请的内存块。 
用结构,然后将结构的指针传给CreateThread()的参数lpParameter。需要  注意:
结构应该是“静态的”或是“全局的”或者是“动态分配的”。
如果使用局部  变量,有可能在线程调用这个结构时,结构已经不在栈中了。  
struct   StructA  
{           
int   n1,   n2,   n3,   n4,   n5;  
};    


DWORD   WINAPI   ThreadFun(LPVOID*   param)  
{
StructA*   pa   =   (StructA*)param;          //这儿就可以访问结构中的变量了         
//......           return   1;  
}    
void   TestFunc()  
{           
DWORD   dwID;            
//下面的写法不对          
StructA   a;  
        CreateThread(NULL,   0,   ThreadFun,   &a,   0,   &dwID);            
//下面的写法是对的          
static   StructA   a;           
CreateThread(NULL,   0,   ThreadFun,   &a,   0,   &dwID);  

这篇关于循环创建多线程时保证参数的有效性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/892592

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

SpringBoot中使用 ThreadLocal 进行多线程上下文管理及注意事项小结

《SpringBoot中使用ThreadLocal进行多线程上下文管理及注意事项小结》本文详细介绍了ThreadLocal的原理、使用场景和示例代码,并在SpringBoot中使用ThreadLo... 目录前言技术积累1.什么是 ThreadLocal2. ThreadLocal 的原理2.1 线程隔离2

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

Java多线程父线程向子线程传值问题及解决

《Java多线程父线程向子线程传值问题及解决》文章总结了5种解决父子之间数据传递困扰的解决方案,包括ThreadLocal+TaskDecorator、UserUtils、CustomTaskDeco... 目录1 背景2 ThreadLocal+TaskDecorator3 RequestContextH

Python创建Excel的4种方式小结

《Python创建Excel的4种方式小结》这篇文章主要为大家详细介绍了Python中创建Excel的4种常见方式,文中的示例代码简洁易懂,具有一定的参考价值,感兴趣的小伙伴可以学习一下... 目录库的安装代码1——pandas代码2——openpyxl代码3——xlsxwriterwww.cppcns.c

Java通过反射获取方法参数名的方式小结

《Java通过反射获取方法参数名的方式小结》这篇文章主要为大家详细介绍了Java如何通过反射获取方法参数名的方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1、前言2、解决方式方式2.1: 添加编译参数配置 -parameters方式2.2: 使用Spring的内部工具类 -

使用Python在Excel中创建和取消数据分组

《使用Python在Excel中创建和取消数据分组》Excel中的分组是一种通过添加层级结构将相邻行或列组织在一起的功能,当分组完成后,用户可以通过折叠或展开数据组来简化数据视图,这篇博客将介绍如何使... 目录引言使用工具python在Excel中创建行和列分组Python在Excel中创建嵌套分组Pyt

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题

《解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题》文章详细描述了在使用lombok的@Data注解标注实体类时遇到编译无误但运行时报错的问题,分析... 目录问题分析问题解决方案步骤一步骤二步骤三总结问题使用lombok注解@Data标注实体类,编译时

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for