在mqtt中, 可以通过订阅主题, 来接收服务器的消息通知. 那是不是我们在单片机中也可以模拟一个呢? 我们可以用这个来获取我们感兴趣的事件.
#include <stdio.h>
#include <string.h>
#include <stdarg.h>
typedef enum {
INIT = 0,
PRESS,
RELEASE,
}keyst_t;
typedef struct {
const char* topic;
keyst_t st;
}msg_t;
typedef void(*fn)(keyst_t st);
typedef struct event{
fn fn;
}event_t;
event_t init_list[5] = { 0 };
event_t press_list[5] = { 0 };
event_t release_list[5] = { 0 };
void sub(fn fn, int i, int arg_num, ...)
{
const char* topic;
va_list ap;
va_start(ap, arg_num);
for (int i = 0; i < arg_num; i++){
topic = va_arg(ap, char*);
if (strcmp(topic, "init") == 0) {
init_list[i].fn = fn;
}
else if (strcmp(topic, "press") == 0) {
press_list[i].fn = fn;
}
else if (strcmp(topic, "release") == 0) {
release_list[i].fn = fn;
}
}
va_end(ap);
}
fn hello;
void main(void)
{
keyst_t keyst = INIT;
sub(hello, 1, 3, "init", "press", "release");
switch (keyst){
case INIT:
init_list[1].fn(INIT);
break;
case PRESS:
init_list[1].fn(PRESS);
break;
case RELEASE:
init_list[1].fn(RELEASE);
break;
}
while (1);
}
这里是以按键消息来做为订阅主题, 为了偷懒, 我没有使用链表, 使用链表会更加灵活, 并且如果是按键有一个消息主题就可以了, 没必要整这么多.
每一条消息有一个数组(实际用链表更灵活), 当订阅时, 就将回调函数注册到该链表下.
sub()这个不定参函数, 可以让订阅消息更灵活, 可以一次订阅多个消息(最近开发的mqtt就可以一个主题将服务器所有主题订阅)
这里应该再次优化, 将sub()函数抽出来, 变成一个公共函数, 其他消息类型也可以订阅.
由于新工作都是维护工作, 所有不会去动代码架构, 因此很多想法不能去做.