c - 在随机条件下的无限 while 循环中创建读取器和写入器线程时出现段错误

标签 c multithreading gdb pthreads

我正在尝试在无限 while 循环中随机创建读取器和写入器线程(每个线程 5 个)。编写者应该等到所有读者都完成执行。为此,我正在使用互斥量和条件变量。

// Global variables
int readerCount=0, writerCount=0, sharedVariable=1001;

pthread_mutex_t m = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t read_cond = PTHREAD_COND_INITIALIZER;
pthread_cond_t write_cond = PTHREAD_COND_INITIALIZER;

读者线程运行下面的函数

void *readShared(void* param)
{
    int shVariable, readers; //To copy and print global variables locally

    pthread_mutex_lock(&m);

        readerCount++;
        while(readerCount > 1)
        {
            pthread_cond_wait(&read_cond, &m);
        }
        readerCount--;
        readers = readerCount;
        shVariable = sharedVariable;

    pthread_mutex_unlock(&m);

    printf("[Reader]:  Value in shared Variable = %d\n", shVariable);
    printf("Other \"Reader\" threads present = %d\n", readers);

    pthread_mutex_lock(&m);

        if(readerCount > 0)
        {
            pthread_cond_signal(&read_cond);
        }
        else if(writerCount > 0)
        {
            pthread_cond_signal(&write_cond);
        }

    pthread_mutex_unlock(&m);
}

编写线程在函数下面运行:

void *writeShared(void *valueToWrite)
{
        //To copy and print global variables
    int value = *(int *)(valueToWrite), readers, shVariable;

    pthread_mutex_lock(&m);

        writerCount++;
        while(readerCount > 0 || writerCount > 1)
        {
            pthread_cond_wait(&write_cond, &m);
        }
        sharedVariable = value;
        readers = readerCount;
        shVariable = sharedVariable;
        writerCount--;

    pthread_mutex_unlock(&m);

    printf("[Writer]:  Value in shared Variable = %d\n", shVariable);
    printf("Value to write = %d\n", value);
    printf("Other \"Reader\" threads present = %d\n", readers);


    pthread_mutex_lock(&m);

        if(readerCount > 0)
        {
            pthread_cond_signal(&read_cond);
        }
        else if(writerCount > 0)
        {
            pthread_cond_signal(&write_cond);
        }

    pthread_mutex_unlock(&m);
}

主要功能:

int main()
{
    pthread_t readers[5], writers[5];
    int readerIndex=0, writerIndex=0;
    while(1)
    {
        if(5 == readerIndex && 5 == writerIndex)
        {
            break;
        }
        if(rand() % 2)
        {
            if(readerIndex < 5)
            {
                pthread_create(&readers[readerIndex++], NULL, readShared, NULL);
            }
        }
        else
        {
            if(writerIndex < 5)
            {
                pthread_create(&writers[writerIndex++], NULL, writeShared, (void*)((rand() % 1000) + 100));
            }
        }
    }
    for(int i=0; i<5; i++)
    {
        pthread_join(readers[i], NULL);
        pthread_join(writers[i], NULL);
    }
    return 0;
}

我尝试使用 gdb 运行并获得以下信息:

Starting program: /home/abhijeet/threads/readWrite 
[Thread debugging using libthread_db enabled]
Using host libthread_db library "/lib/x86_64-linux-gnu/libthread_db.so.1".

[New Thread 0x7ffff77c4700 (LWP 3232)]
[Reader]:  Value in shared Variable = 1001
Other "Reader" threads present = 0

[New Thread 0x7ffff6fc3700 (LWP 3233)]
[Thread 0x7ffff77c4700 (LWP 3232) exited]

[New Thread 0x7ffff67c2700 (LWP 3234)]
[Reader]:  Value in shared Variable = 1001
Other "Reader" threads present = 0

[New Thread 0x7ffff5fc1700 (LWP 3235)]
[Thread 0x7ffff67c2700 (LWP 3234) exited]

Thread 3 "readWrite" received signal SIGSEGV, Segmentation fault.

[Switching to Thread 0x7ffff6fc3700 (LWP 3233)]
0x000055555555494a in writeShared (valueToWrite=0x36d) at PriorityReadWrite.c:13
13      int value = *(int *)(valueToWrite), readers, shVariable;

最佳答案

这一行:

pthread_create(&writers[writerIndex++], NULL, writeShared, (void*)((rand() % 1000) + 100));

将 100 到 1099 之间的随机整数作为参数传递给 writeShared 函数。在您使用 GDB 调试的特定调用中,随机值为 0x36d877

这一行:

int value = *(int *)(valueToWrite)

获取传递的整数并将其解释为整数的地址(事实并非如此),导致段错误。

要解决此问题,您需要传入一个实际整数的地址(在 main 中创建一个 int valueToWrite[5] 数组,然后将 &valueToWrite[writerIndex]pthread_create(小心增加 writerIndex after 你已经创建了线程)。

或者,您可以修复 writeShared 以期待一个整数而不是地址:

int value = (int)valueToWrite;

关于c - 在随机条件下的无限 while 循环中创建读取器和写入器线程时出现段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57455658/

相关文章:

c - 如何使用 C 创建动态 HTML 页面?

c - 稀疏矩阵线性和非线性方程求解器

java - 如何从 Java 在 Android 设备上运行 adb screenrecord 并结束屏幕录制?

c - GDB错误我不认识: "Program received signal EXC_BAD_ACCESS"

c - _beginthread 的第二个参数 stack_size 是什么意思?

无法在 c 中打印我的函数中的任何内容

Linux 中的 C++ 线程

C# UWP 应用程序调用了为不同线程编码的接口(interface)

c++ - 通过 gdb 查看 STL vector 的内容

c++ - gdb:如何将共享库的日志文件重定向到 gdb 输出