免费注册 查看新帖 |

Chinaunix

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

[C] 麻烦看看这段代码,线程实现生产消费者,哪儿有问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-04-04 20:02 |只看该作者 |倒序浏览
本帖最后由 spxjtu 于 2010-04-04 23:39 编辑

能运行一段时间,然后报内存的错误,double free,应该是consumer线程的函数,不知道为什么会两次free呢?

#include <stdlib.h>
#include <pthread.h>
#include <stdio.h>
struct msg{
    struct msg *next;
    int   num;
};                        //生产消费的对象
struct msg *head;
pthread_cond_t  has_product = PTHREAD_COND_INITIALIZER;
pthread_mutex_t   lock = PTHREAD_MUTEX_INITIALIZER;
void *consumer(void *p)
{
    struct msg *mp;
    int    i;
    for(i = 0; i < 10; i ++)
    {
        pthread_mutex_lock(&lock);
        while(head == NULL)
            pthread_cond_wait(&has_product, &lock);
        mp = head;
        head = mp -> next;              //消费者取出连表中第一个结构体的数字,然后指针后移
            pthread_mutex_unlock(&lock);
        printf("Consume %d\n", mp -> num);
        free(mp);                       //报错double free,应该就是这句,不知道为什么
        sleep(rand() % 5);       
    }
}
void *producer(void *p)
{
    struct msg *mp;
    struct msg *loc;
    int    i;
    for(i = 0; i < 10; i ++)
    {
        mp = malloc(sizeof(struct msg));
        mp -> num = rand() % 1000 + 1;
        printf("Produce %d\n", mp -> num);
        pthread_mutex_lock(&lock);
        if(head == NULL)               //将生产的结构依次插入head的链表里
        {
            head = mp;
            loc = mp;
        }
        loc -> next = mp;            
        loc = mp;                      //head指向生成的链表的头部,loc指向最后一个元素
        pthread_mutex_unlock(&lock);
        sleep(rand() % 5);       
    }
}
int main(int argc, char ** argv)
{
    pthread_t   pid, cid;
    srand(time(NULL));
    pthread_create(&pid, NULL, producer, NULL);
    pthread_create(&cid, NULL, consumer, NULL);
    pthread_join(pid, NULL);
    pthread_join(cid, NULL);
    return 0;
}

论坛徽章:
0
2 [报告]
发表于 2010-04-04 20:38 |只看该作者
想都不用想,肯定是同步没有做好。

论坛徽章:
0
3 [报告]
发表于 2010-04-04 21:59 |只看该作者
consumer 没问题,producer明显就是个大bug,lock了没有解锁,逻辑都有问题

论坛徽章:
0
4 [报告]
发表于 2010-04-04 22:07 |只看该作者
{:3_190:},能运行一段时间,不错啊

论坛徽章:
0
5 [报告]
发表于 2010-04-04 23:38 |只看该作者
回复 3# ydfgic


那个解锁的语句打的时候忘了,不好意思

论坛徽章:
0
6 [报告]
发表于 2010-04-05 00:12 |只看该作者
链表些的有问题。第一次,head和head->next都指向同一个,如果consumer能够连续执行当然出问题。还有你mp->next没有赋值,不一定为0,这个时候free也会有问题.

这样试试.
                mp->next        = 0;
        if(head == NULL)               //将生产的结构依次插入head的链表里
        {
            head = mp;
            loc = mp;
                       
        }else{
                        loc -> next = mp;            
                        loc = mp;                      //head指向生成的链表的头部,loc指向最后一个元素
                }

论坛徽章:
0
7 [报告]
发表于 2010-04-05 10:42 |只看该作者
回复 6# sunceenjoy


    多谢啊!我真是大脑短路了,写了if,没写else。

论坛徽章:
0
8 [报告]
发表于 2010-04-06 09:23 |只看该作者
回复 1# spxjtu


    producer没有调用pthread_cond_signal唤醒consumer啊?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP