Rumah > Tutorial sistem > LINUX > Barisan Mesej IPC POSIX Linux: Cara Mudah untuk Penghantaran Mesej Boleh Dipercayai

Barisan Mesej IPC POSIX Linux: Cara Mudah untuk Penghantaran Mesej Boleh Dipercayai

PHPz
Lepaskan: 2024-02-10 14:45:25
ke hadapan
790 orang telah melayarinya

Sistem Linux ialah sistem pengendalian yang menyokong pelaksanaan serentak pelbagai tugas Ia boleh menjalankan berbilang proses pada masa yang sama, dengan itu meningkatkan penggunaan dan kecekapan sistem. Walau bagaimanapun, jika pertukaran data dan kerjasama diperlukan antara proses ini, beberapa kaedah komunikasi antara proses (IPC) perlu digunakan, seperti isyarat, memori dikongsi, semaphore, dsb. Antaranya, baris gilir mesej POSIX adalah kaedah IPC yang agak mudah dan boleh dipercayai Ia membenarkan dua atau lebih proses untuk menghantar mesej melalui baris gilir tanpa mengambil berat tentang kandungan dan format mesej. Artikel ini akan memperkenalkan kaedah baris gilir mesej POSIX dalam sistem Linux, termasuk penciptaan, pembukaan, penghantaran, penerimaan, penutupan dan pemadaman baris gilir mesej.

Linux IPC POSIX 消息队列:一种实现可靠消息传递的简单方式

Model:

#include
#include 
#include 
mq_open()   //创建/获取消息队列fd       
mq_get()    //设置/获取消息队列属性   
mq_send()/mq_receive()    //发送/接收消息 
mq_close()      //脱接消息队列            
mq_unlink()     //删除消息队列            
Salin selepas log masuk

Kelebihan POSIX mq VS Sys V mq

  • Antara muka aplikasi berasaskan fail yang lebih ringkas
  • Sokongan penuh untuk keutamaan mesej (keutamaan akhirnya menentukan kedudukan mesej dalam baris gilir)
  • Menyokong sepenuhnya pemberitahuan tak segerak bagi ketibaan mesej, yang dilaksanakan melalui isyarat atau penciptaan benang
  • Mekanisme tamat masa untuk menyekat operasi hantar dan terima

Mesej nama baris gilir

Ia diketahui daripada $man mq_overview: Baris gilir mesej dikenal pasti secara unik dengan nama dalam bentuk '/somename' Panjang maksimum rentetan nama tidak boleh ke arah NAME_MAX (iaitu, 255 dua proses berkomunikasi dengan menggunakan mesej yang sama nama beratur

mq_open()

//创建一个POSIX消息队列或打开一个已经存在的消息队列,成功返回消息队列描述符mqdes供其他函数使用,失败返回

(mqd_t)-1设errno
//Link with -lrt.
mqd_t mq_open(const char *name, int oflag);
mqd_t mq_open(const char *name, int oflag, mode_t mode, struct mq_attr *attr);
Salin selepas log masuk

benderakan
mesti termasuk salah satu daripada:

  • O_RDONLY bermaksud membuka baris gilir mesej untuk menerima mesej sahaja
  • O_WRONLY bermaksud membuka baris gilir mesej untuk menghantar mesej sahaja
  • O_RDWR bermaksud membuka baris gilir mesej dalam bentuk yang boleh diterima dan dihantar

boleh Bitwised ORed:

  • O_NONBLOCKBuka baris gilir mesej dalam mod tanpa sekatan
  • O_CREATJika baris gilir mesej tidak wujud, buatnya UID pemilik baris gilir mesej ditetapkan kepada UID berkesan proses panggilan, dan GID ditetapkan kepada GID proses panggilan yang berkesan
  • O_EXCLPastikan baris gilir mesej dibuat, jika baris gilir mesej sudah wujud, ralat berlaku

modeJika terdapat O_CREAT dalam oflag, mod digunakan untuk mewakili kebenaran baris gilir mesej yang baru dibuat
attrJika terdapat O_CREAT dalam oflag, maka attr mewakili atribut baris gilir mesej , ia akan menjadi lalai Tetapkan baris gilir mesej konfigurasi (mq_overview(7) untuk butiran.)

mq_setattr() / mq_getattr()

//设置/修改 / 获取消息队列属性,成功返回0,失败返回-1设errno
//Link with -lrt.
int mq_setattr(mqd_t mqdes, const struct mq_attr *newattr, struct mq_attr *oldattr);
int mq_getattr(mqd_t mqdes, struct mq_attr *attr);
Salin selepas log masuk

struktur mqattr

struct mq_attr {
    long mq_flags;      /* Flags: 0 or O_NONBLOCK */
    long mq_maxmsg;     /* Max. # of messages on queue */
    long mq_msgsize;    /* Max. message size (bytes) */
    long mq_curmsgs;    /* # of messages currently in queue */
};
Salin selepas log masuk

mq_send() / mq_timesend()

//发送消息到mqdes指向的消息队列。成功返回0,失败返回-1设errno
//Link with -lrt.
int mq_send(mqd_t mqdes, const char *msg_ptr,size_t msg_len, unsigned int msg_prio);

//如果消息队列满
#include        //额外的header
int mq_timedsend(mqd_t mqdes, const char *msg_ptr,size_t msg_len, unsigned int msg_prio,
const struct timespec *abs_timeout);
Salin selepas log masuk

msg_len msg_ptr指向的消息队列的长度,这个长度必须msg_prio 一个用于表示消息优先级的非0整数,消息按照优先级递减的顺序被放置在消息队列中,同样优先级的消息,新的消息在老的之后,如果消息队列满了,就进入blocked状态,新的消息必须等到消息队列有空间了进入,或者调用被signal中断了。如果flag里有O_NOBLOCK选项,则此时会直接报错
abs_timeout:如果消息队列满了,那么就根据abs_timeout指向的结构体表明的时间进行锁定,里面的时间是从970-01-01 00:00:00 +0000 (UTC)开始按微秒计量的时间,如果时间到了,那么mq_timesend()立即返回

struct timespec {
    time_t tv_sec;        /* seconds */
    long   tv_nsec;       /* nanoseconds */
};
Salin selepas log masuk

mq_receive()/mq_timedreceive()

//从消息队列中取出优先级最高的里面的最老的消息,成功返回消息取出消息的大小,失败返回-1设errno
//具体功能参照mq_send()/mq_timesend()
//Link with -lrt.
ssize_t mq_receive(mqd_t mqdes, char *msg_ptr, size_t msg_len, unsigned int *msg_prio);
#include        //额外的header
ssize_t mq_timedreceive(mqd_t mqdes, char *msg_ptr, size_t msg_len,
 unsigned int *msg_prio, const struct timespec *abs_timeout);
Salin selepas log masuk

mq_notify()

//允许调用进程注册或去注册同步来消息的通知,成功返回0,失败返回-1设errno
//Link with -lrt.
int mq_notify(mqd_t mqdes, const struct sigevent *sevp);
Salin selepas log masuk

sevp指向sigevent的指针

  • 如果sevp不是NULL,那么这个函数就将调用进程注册到通知进程,只有一个进程可以被注册为通知进程
  • 如果sevp是NULL且当前进程已经被注册过了,则去注册,以便其他进程注册
union sigval {                  /* Data passed with notification */
    int     sival_int;          /* Integer value */
    void*   sival_ptr;          /* Pointer value */
};
struct sigevent {
    int     sigev_notify;       /* Notification method */
    int     sigev_signo;        /* Notification signal */
    union sigval    sigev_value;    /* Data passed with notification */
    void(*sigev_notify_function) (union sigval); //Function used for thread notification
 (SIGEV_THREAD)
    void*   sigev_notify_attributes;    // Attributes for notification thread (SIGEV_THREAD)
    pid_t   sigev_notify_thread_id;     /* ID of thread to signal (SIGEV_THREAD_ID) */
};
Salin selepas log masuk

sigev_notify使用下列的宏进行配置:

  • SIGEV_NONE调用进程仍旧被注册,但是有消息来的时候什么都不通知
  • SIGEV_SIGNAL通过给调用进程发送sigev_signo指定的信号来通知进程有消息来了
  • SIGEV_THREAD一旦有消息到了,就激活sigev_notify_function作为新的线程的启动函数

mq_close()

//关闭消息队列描述符mqdes,如果有进程存在针对这个队列的notification request,那么也会被移除

//成功返回0,失败返回-1设errno
//Link with -lrt.
int mq_close(mqd_t mqdes);
Salin selepas log masuk

mq_unlink():

//移除队列名指定的消息队列,一旦最后一个进程关闭了针对这个消息队列的描述符,就会销毁这个消息队列
//成功返回0,失败返回-1设errno
//Link with -lrt.
int mq_unlink(const char *name);
Salin selepas log masuk

本文介绍了Linux系统中POSIX 消息队列的方法,包括消息队列的创建、打开、发送、接收、关闭和删除等方面。通过了解和掌握这些知识,我们可以更好地使用POSIX 消息队列来实现进程间通信,提高系统的稳定性和效率。当然,Linux系统中POSIX 消息队列还有很多其他的特性和用法,需要我们不断地学习和研究。希望本文能给你带来一些启发和帮助。

Atas ialah kandungan terperinci Barisan Mesej IPC POSIX Linux: Cara Mudah untuk Penghantaran Mesej Boleh Dipercayai. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:lxlinux.net
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan