- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我编写了我的第一个多线程程序,并且它的大部分工作正常。共享缓冲区数组最初填充为 -1,指示生产者它是空的并且需要填充。然后,生产者用随机值 1 - 10 填充共享缓冲区,然后生产者轮流填充缓冲区。然后,生产者向消费者发出信号,告知其已填充缓冲区的一个元素并开始消费它。生产者需要填写 120 个元素,消费者需要填写每个条目。程序运行得非常好,直到到达第 110 项。然后它就卡住了,我不明白为什么。我该如何解决这个问题?
这是输出的片段。
Item: 85, Consuming value 8, my thread id is: 1216
Item: 86, Consuming value 7, my thread id is: 298320
Signal
Producer thread 231296 and value: 0
Producer thread 297552 and value: 2
Producer thread 298576 and value: 0
Item: 87, Consuming value 9, my thread id is: 297808
Signal
Producer thread 960 and value: 3
Producer thread 298064 and value: 2
Item: 88, Consuming value 3, my thread id is: 231744
Item: 89, Consuming value 7, my thread id is: 298320
Item: 90, Consuming value 3, my thread id is: 1216
Item: 91, Consuming value 7, my thread id is: 298832
Signal
Producer thread 231296 and value: 3
Producer thread 297552 and value: 8
Producer thread 298576 and value: 6
Item: 92, Consuming value 2, my thread id is: 297808
Signal
Producer thread 960 and value: 9
Producer thread 298064 and value: 7
Item: 93, Consuming value 5, my thread id is: 298320
Item: 94, Consuming value 2, my thread id is: 298832
Item: 95, Consuming value 0, my thread id is: 1216
Item: 96, Consuming value 2, my thread id is: 231744
这是我的代码
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#include <stdint.h>
#define THREADS 5
#define ELEMENTS 120
pthread_t tid_producer[THREADS], tid_consumer[THREADS];
int value = 0;
int saveValue = 0;
void *produce(void *arg);
void *consume(void *arg);
int producerCount =0;
int consumerCount = ELEMENTS;
struct {
pthread_mutex_t mutex;
int index;
int value;
int MyShBuff[ELEMENTS];
} add = {PTHREAD_MUTEX_INITIALIZER, 0, 0};
struct{
pthread_mutex_t mutex;
pthread_cond_t cond;
int nready;
int value;
int empty;
int counter;
/* number ready for consumer */
} nready = {PTHREAD_MUTEX_INITIALIZER, PTHREAD_COND_INITIALIZER,0, -2, ELEMENTS};
int main()
{
int i, j, k;
//Ready buffer for producers
for (i =0; i < ELEMENTS; i++)
{
add.MyShBuff[i]=-1;
}
for(j = 0; j < THREADS; j++) {
pthread_create(&tid_producer[j], NULL, &produce, NULL);
pthread_create(&tid_consumer[j], NULL, &consume, NULL);
}
/* wait for all producers and the consumer*/
for(k = 0; k < THREADS; k++) {
pthread_join(tid_producer[k], NULL);
pthread_join(tid_consumer[k], NULL);
}
/* Clean up and exit */
pthread_mutex_destroy(&nready.mutex);
pthread_mutex_destroy(&add.mutex);
pthread_cond_destroy(&nready.cond);
pthread_exit(NULL);
exit(0);
return 0;
}
void *produce(void *arg)
{
int i = 0;
for ( ; ; )
{
pthread_mutex_lock(&add.mutex);
if(add.index >= ELEMENTS)
{
pthread_mutex_unlock(&add.mutex);
return NULL;
}
if(add.MyShBuff[add.index] == -1)
{
add.value = rand() % 10 + 0;
add.MyShBuff[add.index] = add.value;
printf("Producer thread %d and value: %d\n" ,pthread_self(), add.MyShBuff[add.index]);
add.index++;
}
pthread_mutex_unlock(&add.mutex);
pthread_mutex_lock(&nready.mutex);
if(nready.nready == 0)
{
pthread_cond_broadcast(&nready.cond);
printf("Signal\n");
}
nready.nready++;
pthread_mutex_unlock(&nready.mutex);
}
}
void *consume(void *arg)
{
pthread_mutex_lock(&nready.mutex);
while(nready.empty != 0)
{
while (nready.nready == 0)
{
pthread_cond_wait(&nready.cond,&nready.mutex);
pthread_mutex_lock(&add.mutex);
printf(" Item: %d, Consuming value %d, my thread id is: %d\n", nready.counter, add.MyShBuff[nready.counter], pthread_self());
add.MyShBuff[nready.counter] = -2;
pthread_mutex_unlock(&add.mutex);
nready.counter++;
nready.empty--;
}
nready.nready--;
pthread_mutex_unlock(&nready.mutex);
}
return NULL;
}
最佳答案
正如评论中指出的,您在消费中使用互斥体是错误的(想象一下 while 条件是错误的,线程将在不释放互斥体的情况下离开)此外,我建议限制保存相同信息的变量数量(空,计数器),这使得流程难以遵循。最后,您的 pthread_wait_cond 中还有一个错误:请参阅Does pthread_cond_wait(&cond_t, &mutex); unlock and then lock the mutex?或解释 pthread_wait_cond 的使用的任何其他问题:它们强调谓词可能在返回时为真,这意味着您必须在开始任何需要谓词为真的更多计算之前再次测试它。
我重写了一些你的代码以使其工作;我希望我没有错过其他东西,但这可能会帮助您走得更远:
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#include <stdint.h>
#define THREADS 5
#define ELEMENTS 120
pthread_t tid_producer[THREADS], tid_consumer[THREADS];
int value = 0;
int saveValue = 0;
void *produce (void *arg);
void *consume (void *arg);
int producerCount = 0;
int consumerCount = ELEMENTS;
struct
{
pthread_mutex_t mutex;
int index;
int value;
int MyShBuff[ELEMENTS];
} add = { PTHREAD_MUTEX_INITIALIZER, 0, 0 };
struct
{
pthread_mutex_t mutex;
pthread_cond_t cond;
int nready;
int value;
int empty;
int counter;
/* number ready for consumer */
} nready = { PTHREAD_MUTEX_INITIALIZER, PTHREAD_COND_INITIALIZER, 0, -2, ELEMENTS, 0 };
int main(int argc, char **argv)
{
//Ready buffer for producers
for (int i = 0; i < ELEMENTS; i++) {
add.MyShBuff[i] = -1;
}
for (int j = 0; j < THREADS; j++) {
pthread_create (&tid_producer[j], NULL, &produce, NULL);
pthread_create (&tid_consumer[j], NULL, &consume, NULL);
}
/* wait for all producers and the consumer */
for (int k = 0; k < THREADS; k++) {
pthread_join (tid_producer[k], NULL);
pthread_join (tid_consumer[k], NULL);
}
/* Clean up and exit */
pthread_mutex_destroy (&nready.mutex);
pthread_mutex_destroy (&add.mutex);
pthread_cond_destroy (&nready.cond);
return 0;
}
void *produce(void *dummy)
{
pthread_mutex_lock (&add.mutex);
while (add.index < ELEMENTS) {
if (add.MyShBuff[add.index] == -1)
{
add.value = rand () % 10 + 0;
add.MyShBuff[add.index] = add.value;
printf("Producer thread %ld and value: %d\n" ,pthread_self(), add.MyShBuff[add.index]);
add.index++;
}
pthread_mutex_unlock (&add.mutex);
pthread_mutex_lock (&nready.mutex);
{
if (nready.nready == 0)
{
pthread_cond_broadcast (&nready.cond);
printf ("Signal\n");
}
nready.nready++;
}
pthread_mutex_unlock (&nready.mutex);
pthread_mutex_lock (&add.mutex);
}
pthread_mutex_unlock (&add.mutex);
return NULL;
}
void *consume(void *dummy)
{
pthread_mutex_lock (&nready.mutex);
while (nready.empty != 0)
{
// you also need to check it is not time to leave...
while (nready.nready == 0 && nready.empty != 0) {
pthread_cond_wait (&nready.cond, &nready.mutex);
}
if (nready.empty == 0) {
break;
}
pthread_mutex_lock (&add.mutex);
printf(" Item: %d, Consuming value %d, my thread id is: %ld\n", nready.counter, add.MyShBuff[nready.counter], pthread_self());
add.MyShBuff[nready.counter] = -2;
pthread_mutex_unlock (&add.mutex);
nready.counter++;
nready.empty--;
nready.nready--;
}
pthread_mutex_unlock (&nready.mutex);
return NULL;
}
关于c - 多线程程序在完成之前卡住,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46721748/
我正在寻找一种使此打印 HTML 代码 fragment 向后兼容旧 Android 版本的简单方法: @TargetApi(Build.VERSION_CODES.KITKAT) private v
我在 GCC 终端 (centos linux) 中为 ATM 项目编译以下 c 和 .h 代码时收到以下错误。请帮忙,因为我是编程新手。 validate_acc.h #ifndef _VALIDA
在写关于 SO 的不同问题的答案时,我制作了这个片段: @import url('https://fonts.googleapis.com/css?family=Shadows+Into+Light'
试图弄清楚我应该如何在 my_div_class 之前放置一个 span 而不是替换所有它。现在它取代了 div,但我不想这样做。我假设它类似于 :before 但不知道如何使用它。 { va
我正在使用选择库 http://github.hubspot.com/select/和 noUiSlider https://refreshless.com/nouislider/ .我面临的问题如下
我是开发新手,独自工作。我正在使用 Xcode 和 git 版本控制。可能我没有适本地组织和做错事,但我通常决定做 promise 只是为了在我破坏一切之前做出安全点。在那一刻,我发现很难恰本地描述我
我想确保在同一个桶和键上读取和写入时,应该更新获取的值,也就是说,应该在对其进行写入操作之后获取它。我怎样才能做到这一点? 我想要的是,如果我更新一个键的值,如果我同时使用不同线程获取值,则更新同一个
我的问题与this有关问题,已经有了答案: yes, there is a happens-before relationship imposed between actionsof the thre
The before and after hook documentation on Relish仅显示 before(:suite) 在 before(:all) 之前调用。 我什么时候应该使用其中
我有 CSV 行,我想在其中检测所有内部双引号,没有文本限定符。这几乎可以正常工作,但我的正则表达式还可以检测双引号后的字符。 CSV 部分: "7580";"Lorem ipsum";"";"Lor
是否可以通过Youtube数据API检查广告是否可以与特定视频一起显示? 我了解contentDetails.licensedContent仅显示视频是否已上传至同一伙伴然后由其声明版权。由于第三者权
考虑一下用漂亮的彩色图表描述的“像素管道” https://developers.google.com/web/fundamentals/performance/rendering/ 我有一个元素(比
之前?
在 MVC3 中,我可以轻松地将 jQuery 脚本标签移动到页面底部“_Layout.vbhtml” 但是,在 ASP.NET MVC3 中,当您使用编辑器模板创建 Controller 时,脚手
悬停时内容被替换,但是当鼠标离开元素时我希望它变回来。我该怎么做? $('.img-wrap').hover(function(){ $(this).find('h4').text('Go
已关闭。这个问题是 not reproducible or was caused by typos 。目前不接受答案。 这个问题是由拼写错误或无法再重现的问题引起的。虽然类似的问题可能是 on-top
已关闭。这个问题是 not reproducible or was caused by typos 。目前不接受答案。 这个问题是由拼写错误或无法再重现的问题引起的。虽然类似的问题可能是 on-top
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 已关闭 9 年前。 有关您编写的代码问题的问题必须在问题本身中描述具体问题 - 并包含有效代码以重现该问题。
版本:qwt 6.0.1我尝试开发频谱的对数缩放。我使用简单的线条来启用缩放plotspectrum->setAxisScaleEngine(QwtPlot::yLeft, new QwtLog10S
我有两个相同的表,I_Subject 和 I_Temp_Subject,我想将 Temp_Subject 表复制到 Subject 表。 I_Temp_Subject 由简单用户使用,I_Subjec
我的印象是第一次绘制发生在触发 DOMContentLoaded 事件之后。特别是,因为我认为为了让第一次绘制发生,需要渲染树,它依赖于 DOM 构造。另外,我知道 DOM 构造完成时会触发 DOMC
我是一名优秀的程序员,十分优秀!