zoukankan      html  css  js  c++  java
  • MFC消息映射机制在Linux C++编程环境下的应用(转)

    #ifndef MSG_CONSTANT_H
    #define MSG_CONSTANT_H

    #define MAX_MSG_PARA_SIZE 2048 /*消息参数大小*/
    #define MAX_QUEUE_LENGTH  3000 /*队列限制长度*/


    #define EVENTMSG_QUIT      100 /*退出*/

    #define RW_SUCCESS           0
    #define EMPTY_QUE            1
    #define RW_ERROR_QUE        -1
    #define FULL_QUE            -2
    #define LOCK_ERROR          -3
    #define UNLOCK_ERROR        -4
    #define REALESE_LOCK_ERROR  -5
    #define UN_INI                   -6

    #define PER_BUFF_LENGTH     1000 /*消息输入缓冲区大小(条)*/

    /*!
    *
    * 定义消息映射段内项目的宏 参考MFC 消息映射机制
    */


    /*消息处理函数调用时,加上this,必须是消息机的成员函数*/
    #define MSG_MAP_FUN(MSG_CODE,MSG_ACTION_FUN)  case MSG_CODE:this->MSG_ACTION_FUN(msg);break;
                
    /*实现消息转发到其他消息处理对象指针,用于某些消息需要长时间处理的情况,接受对象必须为msg_machine的子类*/            
    #define MSG_MAP_OBJ(MSG_CODE,MSG_ACTION_OBJ)  case MSG_CODE:SendMessage(msg,MSG_ACTION_OBJ);break;

    /*!消息映射宏
    *
    * BEGIN_MSG_MAP为定义消息映射段开始的宏
    * 消息映射段必须出现在类体内,不是实现部分
    *形式如下 
    *  BEGIN_MSG_MAP
    *    MSG_MAP_FUN(message code ,member function that must return void and have a Msg type parameter)
    *   MSG_MAP_OBJ(message code ,message machine class instance)
    *  END_MSG_MAP
    */
                

    #define BEGIN_MSG_MAP   virtual void Dispatch(struct Msg msg) \
                   {                                      \
                                                    \
                    unsigned int msg_code;                  \
                    msg_code=msg.msg_code;                  \
                    switch(msg_code)                      \
                    {
                

    /*!
    *
    * 定义消息映射段结束的宏
    */
                
    #define END_MSG_MAP(base) default:         \
                base::Dispatch(msg);        \
                break;     \
                }\
         }

    #endif


    1. #ifndef MESSAGE_STRUCT_H
    2. #define MESSAGE_STRUCT_H
    3. #ifdef _cplusplus
    4. extern "C"
    5. {
    6. #include <pthread.h>
    7. #include <semaphore.h>
    8. #include <stdio.h>
    9. #include <stdlib.h>
    10. #include <string.h>
    11. #include <ctype.h>
    12. }
    13. #endif //_cplusplus
    14. #include "msg_macro.h"
    15. typedef unsigned char u_char;
    16. struct Msg
    17. {
    18. unsigned int msg_code;     // Message Code
    19. u_char   msg_para[MAX_MSG_PARA_SIZE]; // Message Parameter
    20. };
    21. class Msg_queue
    22. {
    23. public:
    24. Msg_queue(){ initialized=false; };
    25. ~Msg_queue();
    26. int Initialize();
    27. int GetHead(struct Msg *p_msg);
    28. int AddTail(struct Msg *p_msg);
    29. private:
    30. pthread_mutex_t mutex_wr;
    31. bool initialized; 
    32.     int msg_count;
    33. int head;
    34.     int tail; 
    35. struct Msg   msg_array[MAX_QUEUE_LENGTH];
    36. };
    37. #endif // MESSAGE_STRUCT_H
    复制代码



    1. #include "msg.h"
    2. extern "C"
    3. {
    4. #include <error.h>
    5. #include <errno.h>
    6. }
    7. int  Msg_queue::Initialize()
    8. {
    9. int error;
    10. if(initialized)return 0;
    11. error=pthread_mutex_init(&mutex_wr,NULL);
    12. switch(error){
    13.   case EAGAIN: initialized=false; return -1;
    14.   case ENOMEM: initialized=false; return -2;
    15.   case EPERM:  initialized=false;  return -3;
    16. }
    17.   initialized=true;
    18. msg_count=0;
    19. head=0;
    20. tail=0;
    21. memset(msg_array,0,sizeof(msg_array));
    22. return 0;    
    23. }
    24. int  Msg_queue::GetHead(struct Msg *p_msg)
    25. {
    26. int result;
    27. int error;
    28. if(!initialized)return UN_INI;
    29. if(p_msg==NULL) return RW_ERROR_QUE;
    30. if(error=pthread_mutex_lock(&mutex_wr))
    31. return -3;//取锁失败
    32. if(msg_count>0)
    33. {
    34.   memcpy((void*)p_msg,(void*)&msg_array[head],sizeof(struct Msg));
    35.   memset((void*)&msg_array[head],0,sizeof(struct Msg));
    36.   head++;
    37.   if(head==MAX_QUEUE_LENGTH)head=0;
    38.   msg_count--;
    39.   result= RW_SUCCESS;
    40. }
    41. else
    42. {
    43.   p_msg->msg_code=EMPTY_QUE;
    44.   result=EMPTY_QUE; 
    45. }
    46. if(error=pthread_mutex_unlock(&mutex_wr))
    47. return -4;//解锁失败
    48. return result;
    49. }
    50. int Msg_queue::AddTail(struct Msg *p_msg)
    51. {
    52. int ilock;
    53. int result;
    54. int error;
    55. if(p_msg==NULL)return RW_ERROR_QUE;
    56. if(!initialized)return UN_INI;
    57. if(error=pthread_mutex_lock(&mutex_wr))
    58. return -3;
    59. if(msg_count<MAX_QUEUE_LENGTH)
    60. {
    61.   memcpy((void*)&msg_array[tail],(void*)p_msg,sizeof(struct Msg));
    62.   tail++;
    63.   if(tail==MAX_QUEUE_LENGTH)
    64.   tail=0;
    65.   msg_count++;
    66.   result=RW_SUCCESS;
    67. }
    68. else
    69. result=FULL_QUE;
    70. if(error=pthread_mutex_unlock(&mutex_wr))
    71. result=-4;
    72. return result;
    73. }
    74. Msg_queue::~Msg_queue()
    75. {
    76.   pthread_mutex_destroy(&mutex_wr);   
    77. }
    复制代码



    1. #ifndef  MSGMACHINE_H
    2. #define  MSGMACHINE_H
    3. #include "msg.h"
    4. #include "msg_macro.h"
    5. #include "msg_input_buf.h"
    6. #ifdef _cplusplus
    7. extern "C"
    8. {
    9. #include <stdio.h>
    10. #include <stdlib.h>
    11. #include <string.h>
    12. #include <ctype.h>
    13. #include <signal.h>
    14. #include <unistd.h>
    15. }
    16. #endif //_cplusplus
    17. //----------------------------------------------------------------------------- 
    18.         
    19. class MsgMachine
    20. {
    21. public:
    22. bool cmd_run;
    23. MsgMachine();
    24. virtual ~MsgMachine(); 
    25. void Msg_Deal_Loop();
    26. /*从各一缓冲区读消息到消息队列,异步检查某个缓冲区是否可读*/
    27. int ReadInputBuf(InputBuff *pbuff) ;
    28. /*!
    29. *启动消息循环处理线程
    30. *字类如果改写Execute,必须调用父类的Execute才能启动消息处理循环
    31. */
    32. virtual void Dispatch(struct Msg msg)
    33. {
    34.   unsigned int msg_code;      
    35.       msg_code=msg.msg_code;      
    36.          switch(msg_code)       
    37.          {
    38.       default:break;  
    39.          } 
    40. private: 
    41. bool DetectorActive;
    42. Msg_queue *pMsg_que;
    43.   
    44. };               
    45.                
    46. #endif // MSGMACHINE_H
    复制代码



    1. #include "msg_machine.h"
    2. MsgMachine::~MsgMachine()
    3. {
    4. delete pMsg_que;
    5. }
    6. MsgMachine::MsgMachine()
    7. {
    8. pMsg_que=new Msg_queue;
    9. if(pMsg_que->Initialize()==-1) 
    10. {
    11.   delete pMsg_que;
    12.   //printf("Initialize message queue error!\n");
    13.   //u_log("Initialize message queue error!\n");
    14.   exit(-1);
    15. }
    16. DetectorActive=true; 
    17. }
    18. void MsgMachine::Msg_Deal_Loop()
    19. {
    20. struct Msg msg;
    21. while(cmd_run)
    22. {
    23.   
    24.   if(pMsg_que->GetHead(&msg)==0)
    25.   {
    26.    if(msg.msg_code==EVENTMSG_QUIT)
    27.    break;
    28.    Dispatch(msg); 
    29.   }
    30.   else 
    31.    continue;
    32. }
    33. }
    34. int MsgMachine::ReadInputBuf(InputBuff  *pbuff)
    35. {
    36.   struct Msg msg;
    37.   
    38.   if(pbuff->Read(&msg)==0)
    39.   {
    40.    if(pMsg_que->AddTail(&msg)==RW_SUCCESS)
    41.    { 
    42.     return 0;
    43.    }
    44.    else 
    45.    return -1;
    46.   }
    47.   else
    48.   {
    49.     return -1;
    50.   }
    51. }
    复制代码



    以上是偶们的某个项目(Redhat Linux下C++开发实现与某Portal Server的基于UDP消息的接口
    的daemon,参考MFC的消息映射机制,目前已经实现,上述代码给大家参考,请多多指教

  • 相关阅读:
    SpringBoot自定义FailureAnalyzer
    Linux 系统目录结构
    Spring Boot实战:静态资源处理
    Activiti7整合SpringBoot(十二)
    Acvitivi网关(十一)
    为什么阿里巴巴要禁用Executors创建线程池?
    lqb 基础练习 数列排序 (sort的使用)
    lqb 入门训练 A+B问题
    lqb 入门训练 序列求和 (PS:用长整数做数据的输入输出)
    lqb 入门训练 圆的面积 (PS: PI的精确计算方法 atan(1.0) * 4)
  • 原文地址:https://www.cnblogs.com/qq78292959/p/2720998.html
Copyright © 2011-2022 走看看