- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有一个小的 C++ 程序,其中主进程正在“创建数据”并将它们发送到应该读取该数据的子(fork)进程。我的问题是,在学校我的代码运行良好,但在我自己的笔记本电脑上,两个进程在程序启动后立即卡住。具体来说,他们都在等待 channel “do_msgrcv”中。
这是我的代码:
#define VYROBA 1 // Manufacturer
#define PREPRAVA 2 // Transport
void manufacturer ( ) {
static int count = 0;
int rcv [ 2 ];
while ( 1 ) {
int snd [ 2 ] = { VYROBA, count };
int ret = msgsnd ( glb_msg_id, &snd, sizeof ( int ), 0 );
ret = msgrcv ( glb_msg_id, &rcv, sizeof ( int ), PREPRAVA, 0 );
printf ( "Got crate\n" );
}
}
void consumer ( ) {
static int count = 0;
int rcv [ 2 ];
while ( 1 ) {
int ret = msgrcv ( glb_msg_id, &rcv, sizeof ( int ), VYROBA, 0 );
usleep ( 500000 );
if ( ret < 0 ) {
printf ( "Can't read message.\n" );
}
printf ( "Got product: %d\r\n", rcv [ 1 ] );
fflush ( stdout );
rcv [ 1 ]++;
if ( rcv [ 1 ] == 10 ) {
int snd [ 2 ] = { PREPRAVA, rcv [ 1 ] };
ret = msgsnd ( glb_msg_id, &snd, sizeof ( int ), 0 );
} else {
ret = msgsnd ( glb_msg_id, &rcv, sizeof ( int ), 0 );
}
}
}
如果它对学校有帮助,我们有 Ubuntu 12.04,我正在使用 Ubuntu 16.04。
感谢您的帮助。
最佳答案
你使用的是system V消息队列,linux内核支持since version 2.6 ,因此您考虑的两个 Ubuntu 版本。
现在,如果您查看手册,msgrcv()
and msgsnd()
使用消息缓冲区,hwich 应具有以下结构:
struct msgbuf {
long mtype; /* message type, must be > 0 */
char mtext[1]; /* message data */
};
一旦您开始不使用此结构而是使用一些手动管理的缓冲区,您就有获得不可移植代码的风险(例如,您必须确保正确的大小和填充,考虑字节顺序等...... ).这肯定是这里发生的事情。
消息结构以编码为 long
的消息类型开始,您假设它与 int
(数组的第一个元素)相同。但是 C++ 标准没有固定 int
和 long
的大小(只有它们的最小大小):这个大小可能是 vary between platforms, compilers and CPU architecture .例如:
int
和 long
都是 32 位的并且具有相同的大小。您的代码有效。 long
将是 64 位,而 int
仍将是 32 位。所以你的缓冲区会太短,导致 msgsnd()
和 msgrcv()
溢出缓冲区。这将是未定义的行为,更不用说消息类型将被破坏的事实。 我认为更改您的代码并为缓冲区使用正确的结构应该可以解决问题。
补充说明
顺便说一句,您的 fork 逻辑使 manufacturer() 被执行两次:在原始进程上,但在消费者完成后也在 fork 进程上执行!
if ( !fork ( ) ) {
consumer ( );
}
manufacturer ( );
最好使用 else
来确保它只用在它应该用的地方。
关于C++ msgsnd 和 msgrcv 陷休眠眠,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36823552/
我是一名优秀的程序员,十分优秀!