免费注册 查看新帖 |

Chinaunix

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

[C] ~请教:apue 互斥量foo_find例子不解,望指点 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-09-30 11:30 |只看该作者 |倒序浏览
本帖最后由 apueunix 于 2013-09-30 11:32 编辑

先谢谢,
apue 互斥量foo_find例子不解,对f_id查询前为什么结构数据不加锁(下面红色部分)?如果其它线程直接修改f_id的话,f_id 数据就不致;还是说f_id是不会修改的,但是从结构数据的保护上也要控制吧;

#include <stdlib.h>
#include <pthread.h>

#define NHASH 29
#define HASH(fp) (((unsigned long)fp)%NHASH)  

struct foo *fh[NHASH];
pthread_mutex_t hashlock = PTHREAD_MUTEX_INITIALIZER;

struct foo {
    int             f_count;
    pthread_mutex_t f_lock;
    struct foo     *f_next;
    int             f_id;
    /* ... more stuff here ... */
};

struct foo *
foo_alloc(void) /* allocate the object */
{
    struct foo  *fp;
    int         idx;

    if ((fp = malloc(sizeof(struct foo))) != NULL) {
        fp->f_count = 1;
        if (pthread_mutex_init(&fp->f_lock, NULL) != 0) {
            free(fp);
            return(NULL);
        }
        idx = HASH(fp);
        pthread_mutex_lock(&hashlock);
        fp->f_next = fh[idx];
        fh[idx] = fp;        
        pthread_mutex_lock(&fp->f_lock);
        pthread_mutex_unlock(&hashlock);
        /* ... continue initialization ... */
    }
    return(fp);

}

struct foo *fp
foo_find( int id)
{
       struct  foo *fp;
       int idx;
       idx = HASH(fp);
       pthread_mutex_lock(&hashlock);
       for ( fp=fp[idx]; fp !=NULL;fp=fp[idx]->f_next)
       {
          if (fp ->f_id == id)
           {
              fp->f_count++;
              break;
            }

         }
        pthread_mutex_unlock(&hashlock);
        return (fp);
   }
................. 后续省略

论坛徽章:
1
丑牛
日期:2013-09-29 19:04:50
2 [报告]
发表于 2013-09-30 12:27 |只看该作者
没看到  pthread_mutex_lock(&hashlock); ?

论坛徽章:
0
3 [报告]
发表于 2013-09-30 13:46 |只看该作者
回复 2# solu
谢谢回复。

hashlock 是表的锁,不是数据的锁,apue例中的数据锁是f_lock,本例中f_lock并没有上锁处理。


   

论坛徽章:
0
4 [报告]
发表于 2013-09-30 21:52 |只看该作者

望高手能解释下,我遇到的问题,为什么查询时不对数据加锁。


   

论坛徽章:
0
5 [报告]
发表于 2013-10-01 17:07 |只看该作者
假设可以更改,可能有个临界点:
foo_find()  中
  1. 59     for (fp = fh[idx]; fp != NULL; fp = fp->f_next) {
  2. 60         if (fp->f_id == id) {
  3. 61             foo_hold(fp); // 比较 id 相等了,另一线程先于此函数,将 f_id 改了,那么这里操作的就不再是等 id.的那个元素了;
  4.                               //  但如果修改的修改 f_id 的线程也是持 hashlock 的话,就没问题。
  5. 62             break;
  6. 63         }
  7. 64     }
复制代码
假设不可更改,那么必须要在初始化 foo_alloc() 时设置 f_id 完成:
也有个临界点:
  1. 31         idx = HASH(fp);
  2. 32         pthread_mutex_lock(&hashlock);
  3. 33         fp->f_next = fh[idx];
  4. 34         fh[idx] = fp->f_next;
  5. 35         pthread_mutex_lock(&fp->f_lock);
  6. 36         pthread_mutex_unlock(&hashlock);
  7. 37         /* ... continue initialization ... */
  8. 38         pthread_mutex_unlock(&fp->f_lock);
复制代码
f_id 还未初始化, 就加入 fh[] 数组中了,若是会在 37 行中省略的代码中初始化 f_id, 那 L36 就解了 hashlock 的锁,
如果另外线程此时在 foo_find() 中 pthread_mutex_lock(&hashlock) ,上边 L36 一解锁, 就进入遍历,
又恰好在 foo_alloc() 初始化 fp->f_id 之前使用新加入的节点 fp->f_id 字段时, 就悲剧了。。
愚见, 同等高人

论坛徽章:
0
6 [报告]
发表于 2013-10-02 21:05 |只看该作者
tseesing 发表于 2013-10-01 17:07
假设可以更改,可能有个临界点:
foo_find()  中假设不可更改,那么必须要在初始化 foo_alloc() 时设置 f_ ...


谢谢你的回复!
你提到的两种情况,我想都用f_lock mutex_lock来获得处理权的,不会有临界问题;还有hashlock不是用来处理内部数据的,除了跟链表相关的数据;

我不解的是APUE中foo_find 的例子(红色部分)是不是缺少f_lock控制,这样如何保持数据的一致性?

论坛徽章:
0
7 [报告]
发表于 2013-11-27 14:49 |只看该作者
我也觉得有问题,另外,查找逻辑似乎也有问题,我觉得应该遍历fh,找到与参数id匹配的foo返回,例子程序里面直接对fp对HASH(如下),这个时候的fp只是一个随机值,拿他做HASH有意义么,高手指教!

  1. struct foo *fp
  2. foo_find( int id)
  3. {
  4.        struct  foo *fp;
  5.        int idx;
  6.       [color=Red] idx = HASH(fp);[/color]
  7.        pthread_mutex_lock(&hashlock);
  8.        for ( fp=fp[idx]; fp !=NULL;fp=fp[idx]->f_next)
  9.        {
  10.           if (fp ->f_id == id)
  11.            {
  12.               fp->f_count++;
  13.               break;
  14.             }
  15.          }
  16.         pthread_mutex_unlock(&hashlock);
  17.         return (fp);
  18.    }
复制代码

论坛徽章:
0
8 [报告]
发表于 2013-11-27 15:15 |只看该作者
find逻辑的问题找到答案了:http://www.apuebook.com/errata2e.html
Page 374, end of page. In the statement idx = HASH(fp) at the beginning of foo_find, fp is undefined at this point. The correct thing to do is to search every hash chain to find a matching ID. Same problem occurs again in the middle of page 377.

但是f_id上锁的问题勘误表里面没提到

论坛徽章:
0
9 [报告]
发表于 2013-11-27 15:15 |只看该作者

重复了,删掉

本帖最后由 pirloofmilan 于 2013-11-27 16:48 编辑

重复了,删掉
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP