免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2156 | 回复: 8
打印 上一主题 下一主题

[C] 为什么多线程总会出现意外?-----关于信号量的学习. [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-01-04 10:53 |只看该作者 |倒序浏览
小弟现在学习信号量,写了个简单函数,简单改遍书上的例子!


写了个简单的程序如下:



  1. typedef struct list{
  2.        
  3.         int                semop;
  4.         int                buf[MAXITEM];
  5.         int   threadid[MAXITEM];
  6.         int                nput;
  7.         int                nval;       
  8. }LOCK_LIST;

复制代码


就是 N个线程对 buf 填充数据!   buf[0] 写0,buf[1]写1 ......,依次增加!




  1. #include <sys/types.h>
  2. #include <unistd.h>
  3. #include <signal.h>
  4. #include <string.h>
  5. #include <stdio.h>
  6. #include <stdlib.h>
  7. #include <sys/stat.h>
  8. #include <sys/ipc.h>
  9. #include <sys/sem.h>
  10. #include <stdarg.h>
  11. #include <time.h>
  12. #include <fcntl.h>


  13. #define  MAXITEM                        100000
  14. #define         MAXTHRED            10
  15. #define  NBUFF                           10

  16. #define  BASE_SEM                        0X6001

  17. #define   min(X,Y)   (( X > Y )? Y : X)


  18. void * consume(void *arg);
  19. void * produce(void *arg);


  20. typedef struct list{
  21.        
  22.         int                semop;
  23.         int                buf[MAXITEM];
  24.         int   threadid[MAXITEM];
  25.         int                nput;
  26.         int                nval;       
  27. }LOCK_LIST;

  28. LOCK_LIST  lock_list;

  29. LOCK_LIST  lock_test[10];

  30. int                nitems;


  31. int main(int argc,        char **argv)
  32. {
  33.         int nthreads;
  34.         int i,count[MAXTHRED],ret;
  35.         pthread_t        tid_produce[MAXTHRED],tid_consume;
  36.        
  37.         nitems = min(atoi(argv[1]),                MAXITEM);
  38.         nthreads = min(atoi(argv[2]),        MAXTHRED);
  39.        
  40.         pthread_setconcurrency();
  41.        
  42.         memset(&lock_list,                0,        sizeof(LOCK_LIST));
  43.        
  44.         /*init sem*/
  45.         lock_list.semop = init_sem(BASE_SEM,1);
  46.         if(lock_list.semop < 0)
  47.         {
  48.                         printf("init sem error[%d] .\n",lock_list.semop);
  49.                         return -2;       
  50.         }       
  51.        
  52.         /*start produce thread*/
  53.         for(i = 0;        i < nthreads;        i++)
  54.         {
  55.                         count[i] = 0;
  56.                         pthread_create(&tid_produce[i],        NULL,        produce,        &count[i]);       
  57.         }
  58.        
  59.         printf("thread create succefully.\n");
  60.        
  61.         /*wait produce thread*/
  62.         for(i = 0; i < nthreads; i++)
  63.         {
  64.                         pthread_join(tid_produce[i],        NULL);       
  65.                         printf("count[%d]=%d\n",i,count[i]);
  66.         }
  67.        
  68.         printf("thread realease succefully.\n");
  69.        
  70.         /*start consume thread*/
  71.         pthread_create(&tid_consume,        NULL,        consume,        NULL);
  72.         printf("consume start succefully.\n");
  73.         pthread_join(tid_consume,        NULL);       
  74.         printf("consume end succefully.\n");       
  75.        
  76.         /*release sem*/
  77.         remove_sem(lock_list.semop);
  78.                
  79.         return 0;
  80. }

  81. int  init_sem(int key, int num)
  82. {
  83.                 int semid;
  84.                 int opt = 0;
  85.                 semid = semget(key,        num,        IPC_CREAT |0600);
  86.                 if(semid == -1)
  87.                 {
  88.                         printf("semget error.\n");
  89.                         return -1;       
  90.                 }       
  91.                
  92.                 semctl(key,        0,        SETVAL, opt);
  93.                
  94.                 return semid;
  95. }


  96. int remove_sem(int semid)
  97. {
  98.         if (semctl(semid, 0, IPC_RMID) < 0) {
  99.                 return -1;
  100.         }

  101.         return 0;
  102. }

  103. int lock_sem(int semid, int num)
  104. {
  105.                         struct sembuf ops[2];
  106.                        
  107.                         ops[0].sem_num = num;
  108.                         ops[0].sem_op  = 0;
  109.                         ops[0].sem_flg = 0;
  110.                        
  111.                         ops[1].sem_num        = num;
  112.                         ops[1].sem_op   = 1;
  113.                         ops[0].sem_flg  = SEM_UNDO;
  114.                        
  115.                         return semop(semid,        ops,        2);                       
  116. }

  117. int unlock_sem(int semid,        int num)
  118. {
  119.                  struct sembuf ops;
  120.                  
  121.                  ops.sem_num = num;
  122.                  ops.sem_op =  -1;
  123.                  ops.sem_flg = SEM_UNDO;
  124.                  
  125.                  return semop(semid,        &ops,        1);
  126.        
  127. }

  128. void * produce(void *arg)
  129. {
  130.                 for( ; ; )
  131.                 {
  132.                                 lock_sem(lock_list.semop,        1);
  133.                        
  134.                                 if(lock_list.nput >= nitems)
  135.                                 {
  136.                                         unlock_sem(lock_list.semop,        1);
  137.                                         return NULL;                                       
  138.                                 }       
  139.                                
  140.                                 lock_list.buf[lock_list.nput] = lock_list.nval;
  141.                                 lock_list.threadid[lock_list.nput] = pthread_self();
  142.                                 lock_list.nput ++ ;
  143.                                 lock_list.nval ++ ;
  144.                                
  145.                                
  146.                                 unlock_sem(lock_list.semop,        1);
  147.                                
  148.                                 *((int *) arg) += 1 ;                               
  149.                 }       
  150. }


  151. void *consume(void *arg)
  152. {
  153.                 int         i;
  154.                 char file[1023];
  155.                 FILE *fp;
  156.                 int flag = 0;
  157.                 char temp[1024];
  158.                 for (        i = 0; i < nitems; i++)
  159.                 {
  160.                                 if(lock_list.buf[i] != i)
  161.                                 {       
  162.                                                 flag = 1;
  163.                                                 printf("something error,id = %d ,num = %d.\n",i,lock_list.buf[i]);
  164.                                                 break;
  165.                                 }
  166.                 }
  167.                
  168.                 if(flag)
  169.                 {
  170.                                 sprintf(file,"%s","2.log");
  171.                                 if((fp = fopen(file,"w+")) == NULL)
  172.                                 {
  173.                                                 printf("open file error %s \n",file);       
  174.                                 }       
  175.                                 for (        i = 0; i < nitems; i++)
  176.                                 {
  177.                                                 memset(temp,        0,        sizeof(temp));
  178.                                                 sprintf(temp,"%04d,%04d,pthread=%d\n",lock_list.buf[i],i,lock_list.threadid[i]);
  179.                                                 fputs(temp,fp);
  180.                                 }                               
  181.                 }       
  182.                        
  183.                 return(NULL);
  184. }

复制代码




==================================================================================



编译aix5)


cc -o locklist locklist.c  -lpthread


运行:



thread create succefully.
count[0]=2412
count[1]=2805
count[2]=1420
count[3]=596
count[4]=2817
thread realease succefully.
consume start succefully.
something error,id = 137 ,num = 0.
consume end succefully.


2.log 中的记录:


0134,0134,pthread=772
0135,0135,pthread=258
0136,0136,pthread=772
0000,0137,pthread=258                     /////////////////error/////////////////?????why???????
0138,0138,pthread=258
0139,0139,pthread=772
0140,0140,pthread=258
0141,0141,pthread=772
0142,0142,pthread=258
0143,0143,pthread=772
0144,0144,pthread=258



为什么总有不对的数据呢??而且频率很高,几乎每隔一次都产生...程序那有问题,请指点!!!!!


代码附:

locklist.rar

1.4 KB, 下载次数: 43

论坛徽章:
0
2 [报告]
发表于 2008-01-04 14:15 |只看该作者
建议编译的时候加上-Wall。
我的调试结果是系统处理不过来,导致某条++语句未执行

论坛徽章:
0
3 [报告]
发表于 2008-01-04 14:26 |只看该作者
原帖由 soul_of_moon 于 2008-1-4 14:15 发表
建议编译的时候加上-Wall。
我的调试结果是系统处理不过来,导致某条++语句未执行




哦~ 请教你是怎么做的? 知道那个++ 未执行?

论坛徽章:
0
4 [报告]
发表于 2008-01-04 14:48 |只看该作者
for( ; ; )
                {
                                lock_sem(lock_list.semop,        1);
                       
                                if(lock_list.nput != lock_list.nval)
                                        printf("before: %d, %d\n", lock_list.nput, lock_list.nval);

                                if(lock_list.nput >= nitems)
                                {
                                        unlock_sem(lock_list.semop,        1);
                                        return NULL;                                       
                                }      
                              
                                lock_list.buf[lock_list.nput] = lock_list.nval;
                                lock_list.threadid[lock_list.nput] = pthread_self();
                                lock_list.nput ++ ;
                                lock_list.nval ++ ;
                              
                               if(lock_list.nput != lock_list.nval)
                                        printf("after: %d, %d\n", lock_list.nput, lock_list.nval);

                                unlock_sem(lock_list.semop,        1);
                              
                                *((int *) arg) += 1 ;                              
                }   



后来加个usleep(10)就不再现了

论坛徽章:
0
5 [报告]
发表于 2008-01-04 15:13 |只看该作者
奇怪,你的是什么环境?

我怎么usleep(1000);都还会出现呢。.

论坛徽章:
0
6 [报告]
发表于 2008-01-04 15:27 |只看该作者
你用这段程序试试看

#include <sys/types.h>
#include <unistd.h>
#include <signal.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdarg.h>
#include <time.h>
#include <fcntl.h>
#include <pthread.h>


#define  MAXITEM                        100000
#define         MAXTHRED            10
#define  NBUFF                           10

#define  BASE_SEM                        0X6001

#define   min(X,Y)   (( X > Y )? Y : X)

pthread_mutex_t mutex;

void * consume(void *arg);
void * produce(void *arg);
int  init_sem(int key, int num);
int remove_sem(int semid);

typedef struct list{
       
        int                semop;
        int                buf[MAXITEM];
        int   threadid[MAXITEM];
        int                nput;
        int                nval;       
}LOCK_LIST;

LOCK_LIST  lock_list;

int                nitems;


int main(int argc,        char **argv)
{
        int nthreads;
        int i,count[MAXTHRED];
        pthread_t        tid_produce[MAXTHRED],tid_consume;
       
        nitems = min(atoi(argv[1]),                MAXITEM);
        nthreads = min(atoi(argv[2]),        MAXTHRED);
       
        pthread_setconcurrency();
       
        memset(&lock_list,                0,        sizeof(LOCK_LIST));
       
        /*init sem*/
        lock_list.semop = init_sem(BASE_SEM,1);
        if(lock_list.semop < 0)
        {
                        printf("init sem error[%d] .\n",lock_list.semop);
                        return -2;       
        }       
       
        /*start produce thread*/
        for(i = 0;        i < nthreads;        i++)
        {
                        count = 0;
                        pthread_create(&tid_produce,        NULL,        produce,        &count);       
        }
       
        printf("thread create succefully.\n");
       
        /*wait produce thread*/
        for(i = 0; i < nthreads; i++)
        {
                        pthread_join(tid_produce,        NULL);       
                        printf("count[%d]=%d\n",i,count);
        }
       
        printf("thread realease succefully.\n");
       
        /*start consume thread*/
        pthread_create(&tid_consume,        NULL,        consume,        NULL);
        printf("consume start succefully.\n");
        pthread_join(tid_consume,        NULL);       
        printf("consume end succefully.\n");       
       
        /*release sem*/
        remove_sem(lock_list.semop);
               
        return 0;
}

int  init_sem(int key, int num)
{
        /*
                int semid;
                int opt = 0;
                semid = semget(key,        num,        IPC_CREAT |0600);
                if(semid == -1)
                {
                        printf("semget error.\n");
                        return -1;       
                }       
               
                semctl(key,        0,        SETVAL, opt);
               
                return semid;
                */
                pthread_mutex_init(&mutex, NULL);
                return 0;
}


int remove_sem(int semid)
{
//        if (semctl(semid, 0, IPC_RMID) < 0) {
//                return -1;
//        }

        pthread_mutex_destroy(&mutex);

        return 0;
}

int lock_sem(int semid, int num)
{
        pthread_mutex_lock(&mutex);

        return 0;
//                        struct sembuf ops[2];
//                       
//                        ops[0].sem_num = num;
//                        ops[0].sem_op  = 0;
//                        ops[0].sem_flg = 0;
//                       
//                        ops[1].sem_num        = num;
//                        ops[1].sem_op   = 1;
//                        ops[1].sem_flg  = SEM_UNDO;
//                       
//                        return semop(semid,        ops,        2);                       
}

int unlock_sem(int semid,        int num)
{
//                 struct sembuf ops;
//                 
//                 ops.sem_num = num;
//                 ops.sem_op =  -1;
//                 ops.sem_flg = SEM_UNDO;
//                 
//                 return semop(semid,        &ops,        1);
       
        pthread_mutex_unlock(&mutex);
        return 0;
}

void * produce(void *arg)
{
                for( ; ; )
                {
                                lock_sem(lock_list.semop,        1);
                       
                                if(lock_list.nput != lock_list.nval)
                                        printf("before: %d, %d\n", lock_list.nput, lock_list.nval);
                       
                                if(lock_list.nput >= nitems)
                                {
                                        unlock_sem(lock_list.semop,        1);
                                        return NULL;                                       
                                }       
                               
                                lock_list.buf[lock_list.nput] = lock_list.nval;
                                lock_list.threadid[lock_list.nput] = pthread_self();
                                lock_list.nput ++ ;
                                lock_list.nval ++ ;
                               
                                if(lock_list.nput != lock_list.nval)
                                        printf("after: %d, %d\n", lock_list.nput, lock_list.nval);
//                                usleep(10);
                               
                                unlock_sem(lock_list.semop,        1);
                               
                                *((int *) arg) += 1 ;                               
                }       
}


void *consume(void *arg)
{
                int         i;
                char file[1023];
                FILE *fp;
                int flag = 0;
                char temp[1024];
                for (        i = 0; i < nitems; i++)
                {
                                if(lock_list.buf != i)
                                {       
                                                flag = 1;
                                                printf("something error,id = %d ,num = %d.\n",i,lock_list.buf);
                                                break;
                                }
                }
               
                if(flag)
                {
                                sprintf(file,"%s","2.log");
                                if((fp = fopen(file,"w+")) == NULL)
                                {
                                                printf("open file error %s \n",file);       
                                }       
                                for (        i = 0; i < nitems; i++)
                                {
                                                memset(temp,        0,        sizeof(temp));
                                                sprintf(temp,"%04d,%04d,pthread=%d\n",lock_list.buf,i,lock_list.threadid);
                                                fputs(temp,fp);
                                }                               
                }       
                       
                return(NULL);
}

论坛徽章:
0
7 [报告]
发表于 2008-01-04 16:56 |只看该作者
有几个问题,
1,你的程序想达到什么效果,"N个线程对 buf 填充数据!   buf[0] 写0,buf[1]写1 ......,依次增加!",如果你不加usleep,那么实际上就第一个线程去填充,其他的就不会去填充了。
2,如果你的nitems小于线程数,那么第nitems个线程就没有什么意义了
3,不知道你什么环境,我在ES4.0下运行完全没有问题,不管有没有usleep

论坛徽章:
0
8 [报告]
发表于 2008-01-04 17:17 |只看该作者
to: soul_of_moon

用互斥锁不会出现这种情况~~ 但为啥信号量就不行呢?奇怪..

编译器:

  C for AIX Compiler, Version 5

  Usage:
     xlc [ option | inputfile ]...
     cc [ option | inputfile ]...
     c89 [ option | inputfile ]...
     xlc128 [ option | inputfile ]...
     cc128 [ option | inputfile ]...
     xlc_r [ option | inputfile ]...
     cc_r [ option | inputfile ]...
     xlc_r4 [ option | inputfile ]...
     cc_r4 [ option | inputfile ]...
     xlc_r7 [ option | inputfile ]...
     cc_r7 [ option | inputfile ]...






to : ljok30



"1,你的程序想达到什么效果,"N个线程对 buf 填充数据!   buf[0] 写0,buf[1]写1 ......,依次增加!",如果你不加usleep,那么实际上就第一个线程去填充,其他的就不会去填充了。
"

这个理解应该有误,我设置了线程并发,当数据量大,比如我测试的10000 5 的时候,肯定会好几个线程处理的。


“3,不知道你什么环境,我在ES4.0下运行完全没有问题,不管有没有usleep”

我现在怀疑是不是跟我的环境有关系,但又找不到原因~~

论坛徽章:
0
9 [报告]
发表于 2008-01-04 19:25 |只看该作者
终于找到原因了~

我犯了一个极低级的错误..

楼上的两位大哥别怪我啊。。。

锁加错了~~

没脸再发帖了~


您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP