从零开始搭建属于自己的物联网平台(二)实现基于订阅发布的消息总线_自己搭建物联网平台

public abstract class EventMessageBroker {

    /\*\*
 \* 消息多播
 \*
 \* 将集群间消息在自己服务内部广播,是的所有订阅消息都可以收到
 \*
 \* @param message
 \* @return
 \*/
    public abstract void multicastEvent(AdapterMessage message) throws IllegalAccessException, InstantiationException, InvocationTargetException, IOException;

    public void doMulticastEvent(AdapterMessage message, Map<TopicMatcher, Map<String, MessageAdapter>> subscribesPool, ApplicationContext applicationContext) throws InvocationTargetException, IllegalAccessException, InstantiationException, IOException {
        // 查找相匹配的Adapter
        List<MessageAdapter> adapterList = TopicUtil.getAllMatchedAdapter(message.getTopic(), subscribesPool);

        for (MessageAdapter adapter : adapterList) {
            if (adapter != null && adapter.isActive()) {
                // 手动订阅消息
                if (adapter.getCustomer() != null) {
                    adapter.getCustomer().consume(message);
                } else {
                    // 取得对象
                    Object instance = applicationContext.getBean(adapter.getClazz());
                    if (instance != null) {
                        // 将消息转化为所需要的类型
                        if (adapter.getMethod().getParameterCount() > 0) {
                            Class<?> param = adapter.getMethod().getParameterTypes()[0];
                            adapter.getMethod().invoke(instance, JSON.parseObject(message.getPayload(), param));
                        } else {
                            adapter.getMethod().invoke(instance);
                        }

                    }
                }
            }

        }

    }
}

使用redis实现消费者
/\*\*
 \* 消息总线-集群间消息redis实现
 \*
 \* 实现规则
 \* 根据不同的MQ机制实现消息的分组消费,将原始消息转化为AdapterMessage,然后使用继承于EventMessageBroker进行消息多播
 \*/
@Slf4j
@Component
@ConditionalOnProperty(prefix = "stream.broker", name = "type", havingValue = "redis", matchIfMissing = true)
public class RedisMessageBroker  extends EventMessageBroker implements StreamListener<String, MapRecord<String, String, String>> {

    @Value("${stream.topic:/stream}")
    String topic;

    @Value("${stream.consumer.id}")
    String consumerId;

    @Value("${stream.timeout:10}")
    Integer timeout;

    @Value("${stream.consumer.group}")
    String groupName = "stream.redis";

    private final RedisTemplate<String, String> streamRedisTemplate;
    private final ApplicationContext applicationContext;
    private final EventBus eventBus;

    public RedisMessageBroker(RedisTemplate<String, String> streamRedisTemplate, ApplicationContext applicationContext, EventBus eventBus) {
        this.streamRedisTemplate = streamRedisTemplate;
        this.applicationContext = applicationContext;
        this.eventBus = eventBus;
    }


    @SneakyThrows
    @Override
    public void onMessage(MapRecord<String, String, String> message) {
        log.info("消息内容-->{}", message.getValue());
        StreamOperations<String, String, String> streamOperations = streamRedisTemplate.opsForStream();

        // 服务内消息多播
        AdapterMessage adapterMessage = new AdapterMessage();
        adapterMessage.setTopic(message.getValue().get("topic"));
        adapterMessage.setPayload(message.getValue().get("payload"));
        try {
            this.multicastEvent(adapterMessage);
        } catch (Exception e) {
            log.info("消息多播失败:" + e.getLocalizedMessage());
        }
        //消息应答
        streamOperations.acknowledge(topic, groupName, message.getId());
    }


    @Bean
    public StreamMessageListenerContainer.StreamMessageListenerContainerOptions<String, MapRecord<String, String, String>> emailListenerContainerOptions() {

        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();

        return StreamMessageListenerContainer.StreamMessageListenerContainerOptions
                .builder()
                //block读取超时时间
                .pollTimeout(Duration.ofSeconds(timeout))
                //count 数量(一次只获取一条消息)
                .batchSize(1)
                //序列化规则
                .serializer(stringRedisSerializer)
                .build();
    }

    /\*\*
 \* 开启监听器接收消息
 \*/
    @Bean
    public StreamMessageListenerContainer<String, MapRecord<String, String, String>> emailListenerContainer(RedisConnectionFactory factory,
                                                                                                            StreamMessageListenerContainer.StreamMessageListenerContainerOptions<String, MapRecord<String, String, String>> streamMessageListenerContainerOptions) {
        StreamMessageListenerContainer<String, MapRecord<String, String, String>> listenerContainer = StreamMessageListenerContainer.create(factory,
                streamMessageListenerContainerOptions);

        //如果 流不存在 创建 stream 流
        if (!streamRedisTemplate.hasKey(topic)) {
            streamRedisTemplate.opsForStream().add(topic, Collections.singletonMap("", ""));
            log.info("初始化集群间通信Topic{} success", topic);
        }

        //创建消费者组
        try {
            streamRedisTemplate.opsForStream().createGroup(topic, groupName);
        } catch (Exception e) {
            log.info("消费者组 {} 已存在", groupName);
        }

        //注册消费者 消费者名称,从哪条消息开始消费,消费者类
        // > 表示没消费过的消息
        // $ 表示最新的消息
        listenerContainer.receive(
                Consumer.from(groupName, consumerId),
                StreamOffset.create(topic, ReadOffset.lastConsumed()),
                this
        );

        listenerContainer.start();
        return listenerContainer;
    }

    @Override
    public void multicastEvent(AdapterMessage message) throws IllegalAccessException, InstantiationException, InvocationTargetException, IOException {
        super.doMulticastEvent(message, eventBus.getSubscribesPool(), applicationContext);
    }
}

配套@Subscribe注解

/\*\*
 \* 自定义消息订阅
 \*
 \* @author liuhl
 \*/
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
@SuppressWarnings("unused")
public @interface Subscribe {

    /\*\*
 \* 订阅topic
 \* @return
 \*/
    @AliasFor("value")
    String topic() default "";

    @AliasFor("topic")
    String value() default "";

}

实现BeanPostProcessor监听所有的bean创建
@Component
@Slf4j
public class SpringMessageSubscribe implements BeanPostProcessor {

    private final EventBus eventBus;

    private final MessageAdapterCrater messageAdapterCrater = new DefaultMessageAdapterCrater();

    public SpringMessageSubscribe(EventBus eventBus) {
        this.eventBus = eventBus;
    }

    /\*\*
 \* 为注解添加监听处理
 \*
 \* @param bean
 \* @param beanName
 \* @return
 \* @throws BeansException
 \*/
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        Class<?> type = ClassUtils.getUserClass(bean);

        ReflectionUtils.doWithMethods(type, method -> {
            // 取得所有订阅方法
            AnnotationAttributes subscribes = AnnotatedElementUtils.getMergedAnnotationAttributes(method, Subscribe.class);
            if (CollectionUtils.isEmpty(subscribes)) {
                return;
            }
            // 生成订阅Adapter
            MessageAdapter sub = messageAdapterCrater.createMessageAdapter(type, method, subscribes.getString("topic"));
            // 注册订阅Adapter
            eventBus.addAdapter(sub, subscribes.getString("topic"));
        });
        return bean;
    }
}

EventBus对象

EventBus实体,主要负责三个功能,发布、订阅的入口,以及管理维护MessageAdapter

/\*\*
 \* 消息总线
 \*/
@Component
public class EventBus {

    private final MessagePublisher messagePublisher;

    private Map<TopicMatcher, Map<String, MessageAdapter>> subscribesPool = new ConcurrentHashMap();

    public EventBus(MessagePublisher messagePublisher) {
        this.messagePublisher = messagePublisher;
    }

    /\*\*
 \* 集群间消息发布
 \*
 \* @param message 消息体
 \* @return 发布结果
 \*/
    public Boolean publish(AdapterMessage message) {
        return messagePublisher.publish(message);
    }

    /\*\*
 \* 将Adapter注册到EventBus
 \*
 \* @param adapter
 \* @param topic
 \*/
    public void addAdapter(MessageAdapter adapter, String topic) {
        if (subscribesPool.get(topic) != null) {
            subscribesPool.get(topic).put(adapter.getId(), adapter);
        } else {
            Map<String, MessageAdapter> adapterMap = new HashMap<>();
            adapterMap.put(adapter.getId(), adapter);
            subscribesPool.put(new TopicMatcher(topic), adapterMap);
        }
    }

    /\*\*
 \* 将Adapter注册到EventBus
 \*
 \* @param adapter
 \* @param topic
 \*/
    public void removeAdapter(MessageAdapter adapter, String topic) {
        if (subscribesPool.get(topic) != null) {
            subscribesPool.get(topic).remove(adapter.getId());
## 最后

**自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

**深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。**

**因此收集整理了一份《2024年嵌入式&物联网开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。**

![img](https://img-blog.csdnimg.cn/img_convert/7f4d61fff54df420ae301ae98c6d242f.png)

![img](https://img-blog.csdnimg.cn/img_convert/9b818099c9e4561a0e3f84fbe4debea8.jpeg)

![img](https://img-blog.csdnimg.cn/img_convert/967c21f1bcc3c69e5dc42491626a6584.png)

 ![img](https://img-blog.csdnimg.cn/img_convert/ef1a08f443c2ee297a9715a9035e3c7b.png)

![img](https://img-blog.csdnimg.cn/img_convert/1a515f436b6b4029907874e390752506.png)

![img](https://img-blog.csdnimg.cn/img_convert/c4959b0e642a791e4aa45b2861b67119.png)

![](https://img-blog.csdnimg.cn/img_convert/c21b264e8feb134f39100ed3d8d3aa82.png)

 

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上嵌入式&物联网开发知识点,真正体系化!**

[**如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!**](https://bbs.csdn.net/topics/618654289)

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**!!


存中...(img-aybsYoP1-1715676846502)]

[外链图片转存中...(img-BFGaQ9PQ-1715676846503)]

 

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上嵌入式&物联网开发知识点,真正体系化!**

[**如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!**](https://bbs.csdn.net/topics/618654289)

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**!!


  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要基于Spring Boot搭建物联网平台实现COAP协议接入,你需要遵循以下步骤: 1. 首先需要了解COAP协议,COAP是Constrained Application Protocol(受限应用协议)的缩写,它是一种轻量级的Web传输协议,专门用于连接受限环境下的设备。COAP基于UDP协议,支持多播和组播,具有低延迟和低能耗等特点。在Spring Boot中可以使用Eclipse Californium库来实现COAP协议的接入。 2. 在Spring Boot项目中引入Eclipse Californium库,可以通过Maven或Gradle来引入,具体方法可以参考Eclipse Californium的官方文档。 3. 实现COAP协议的服务端,可以在Spring Boot项目中创建一个COAP服务端类,并添加COAP资源。在COAP资源中定义资源路径、请求方法和响应内容等信息。COAP服务端类需要继承Californium的CoapServer类。 4. 实现COAP协议的客户端,可以通过Eclipse Californium提供的CoapClient类来实现。在Spring Boot项目中创建一个COAP客户端类,通过CoapClient类发送COAP请求,获取响应信息。 5. 在物联网平台中使用COAP协议进行设备接入,可以通过将设备与COAP服务端进行绑定,实现设备信息的采集和控制。在平台中定义COAP资源路径和请求方法,实现设备信息的获取和控制。 综上所述,基于Spring Boot搭建物联网平台实现COAP协议接入需要掌握COAP协议的基本知识,熟悉Eclipse Californium库的使用方法,并具备Java编程能力。 ### 回答2: 物联网平台是一种用于连接和管理物联网设备的软件平台,它允许设备之间相互通信,并与云端应用进行数据交互和控制操作。基于Spring Boot搭建物联网平台可以实现COAP协议的接入。 COAP(Constrained Application Protocol)是一种轻量级的应用层协议,专为物联网设备设计。它具有低开销、低带宽和低功耗的特点,适用于资源受限的设备和网络环境。COAP协议可以通过UDP和DTLS(Datagram Transport Layer Security)进行数据传输。 为了在Spring Boot中实现COAP协议接入,可以使用Eclipse Californium项目作为COAP协议的实现库。该项目提供了COAP协议的Java实现,可以方便地嵌入到Spring Boot应用中。 首先,在Spring Boot项目的依赖管理文件(例如pom.xml)中添加Eclipse Californium库的依赖。然后,在Spring Boot的配置文件中设置COAP服务器的监听端口和相关参数。 在Spring Boot中编写COAP的处理器类,用于处理COAP请求和响应。可以定义不同的URI来映射到不同的处理器方法,根据具体需求进行业务逻辑处理和数据交互。处理器方法可以使用COAP的API来处理COAP消息,例如解析请求、发送响应等。 另外,在物联网平台中,还可以与数据库进行交互,将设备数据进行持久化存储和查询。在Spring Boot中,可以使用Spring Data库来简化数据库访问的操作。可以定义实体类来表示设备数据,使用Spring Data提供的注解和API来进行数据库的操作。 通过以上步骤,基于Spring Boot搭建物联网平台就可以实现COAP协议的接入。该平台可以接收来自物联网设备的COAP请求,处理请求并返回相应的COAP响应。同时,可以将设备数据存储到数据库中,并提供API接口供云端应用访问和控制。这样,可以实现物联网设备的远程监控和管理。 ### 回答3: 基于Spring Boot搭建物联网平台可以实现COAP(Constrained Application Protocol)协议的接入,以下是该过程的简要解释。 首先,Spring Boot是一个开源的Java框架,用于快速构建基于Java的应用程序。它提供了一种简单易用的方式搭建RESTful风格的Web服务,并且具有良好的扩展性和模块化的特性,非常适合用于构建物联网平台。 COAP是一种专为物联网设备设计的应用层协议,它基于HTTP协议,但比HTTP更适合于资源受限的设备。COAP协议可以实现对设备的低功耗连接、高效的数据传输和灵活的资源管理。 在基于Spring Boot搭建物联网平台中,要实现COAP协议的接入,首先需要引入COAP协议相关的依赖。这可以通过在项目的pom.xml文件中添加COAP协议的Java实现库,如Eclipse Californium,来实现。 接下来,可以创建COAP服务器端的资源。在Spring Boot中,可以使用@Controller和@RequestMapping注解来定义COAP资源的访问路径和处理方法。通过处理方法,可以实现对设备的读取、修改、删除等操作。 另外,在COAP协议中,通信的双方都有一个COAP客户端和COAP服务器的身份。因此,物联网平台也需要实现COAP客户端,用于与COAP服务器进行通信。可以使用RestTemplate类或者其他COAP客户端工具来发送COAP请求和接收COAP响应。 最后,基于Spring Boot搭建物联网平台可以实现COAP协议的接入,通过COAP服务器和客户端的交互,实现物联网设备的管理和控制。此外,Spring Boot还提供了丰富的特性和扩展性,可以方便地与其他模块进行集成,为物联网平台开发提供更多的选择。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值