MQTT与Android

接上文:MQTT协议入门

首先应部署EMQX到公网上以便后续的访问操作。

MQTT与Android_第1张图片

等会儿的结果信息可以在这里查看。


MQTT回顾:

MQTT协议有三种身份:发布者、代理、订阅者,发布者和订阅者都为客户端,代理为服务器,同时消息的发布者也可以是订阅者(为了节约内存和流量发布者和订阅者一般都会定义在一起)。
MQTT传输的消息分为主题(Topic,可理解为消息的类型,订阅者订阅后,就会收到该主题的消息内容(payload))和负载(payload,可以理解为消息的内容)两部分。


JAVA中:阿里专门针对Android客户端的实现方式MqttAndroidClient

implementation 'org.eclipse.paho:org.eclipse.paho.client.mqttv3:1.2.0'

MQTT与Android_第2张图片

这里有个简单的UI,

先想清楚做什么:

Android端充当订阅者,订阅消息主题,然后MQTT.fx工具模拟发布者发布消息,让Android端接收消息。

然后反过来Android端充当发布者发布消息。然后MQTT.fx工具模拟订阅者,订阅消息主题,收到来自Android端的消息。

总体的步骤:

MQTT与Android_第3张图片


1.Android端充当订阅者,订阅消息主题,然后MQTT.fx工具模拟发布者发布消息,让Android端接收消息。

MQTT与Android_第4张图片

程序启动,android端连接、订阅主题:

MQTT与Android_第5张图片

并设置了消息的回调:

MQTT与Android_第6张图片

 ,然后是MQTT.fx:

MQTT与Android_第7张图片

MQTT与Android_第8张图片 发送者向订阅者Android发送了1条消息。

MQTT与Android_第9张图片

客户端根据回调处理消息结果并进行UI展示:


2.Android端充当发布者发布消息。然后MQTT.fx工具模拟订阅者,订阅消息主题,收到来自Android端的消息。

MQTT与Android_第10张图片

MQTT与Android_第11张图片

这里是客户端推送给另一端:

MQTT与Android_第12张图片

订阅了android端的主题后,当android端点击按钮发送一条消息后,订阅者就可以接收消息。


全部源码:

 

package com.yinlei.mqtt_project

import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.os.Handler
import android.os.Message
import android.util.Log
import android.widget.Toast
import kotlinx.android.synthetic.main.activity_main.*
import org.eclipse.paho.client.mqttv3.*
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence
import java.util.concurrent.Executors
import java.util.concurrent.ScheduledExecutorService
import java.util.concurrent.TimeUnit
import kotlin.Exception
import kotlin.concurrent.thread

/**
 * 1.连接成功
 * 2. 开始订阅topic
 * 3.拿到数据做逻辑、UI处理
 * 4.发布消息到指定topic
 * 5. 客户端发布消息给硬件。
 * 6.硬件得到消息并做相应的处理。
 */
class MainActivity : AppCompatActivity() {


    private val host ="填写自己的broker"
    private val userNames = "android"//用户名
    private val passWord = "android"//密码
    private val mqtt_id = "1099129793"//客户端id
    private val mqtt_sub_topic = "1099129793"//主题
    private val mqtt_pub_topic = "YL1099129793"//发布的主题topic
    private val scheduler: ScheduledExecutorService by lazy {
        Executors.newSingleThreadScheduledExecutor()
    }


    private val client: MqttClient by lazy{
        //host为主机名,test为clientid即连接MQTT的客户端ID,一般以客户端唯一标识符表示,MemoryPersistence设置clientid的保存形式,默认为以内存保存
         MqttClient(host, mqtt_id, MemoryPersistence())
    }
    private  val options: MqttConnectOptions by lazy {
        //MQTT的连接设置
         MqttConnectOptions()
    }
    private lateinit var  handler: Handler

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        btn_1.setOnClickListener {
            Toast.makeText(this@MainActivity,"Hello MQTT", Toast.LENGTH_SHORT).show()
        }
        image_1.setOnClickListener {
            Toast.makeText(this@MainActivity,"图片1", Toast.LENGTH_SHORT).show()
            PublishMessagepWrapper(mqtt_pub_topic, "Android客户端发布的消息")
        }


        MqttInit()
        startReconnect()

        handler = Handler{
            when(it.what){
                1 ->{
                    Log.d("yinlei","开机校验更新回传")
                }
                2 ->{
                    Log.d("yinlei","反馈回传")
                }
                3 -> {
                    Log.d("yinlei","MQTT 收到消息回传   UTF8Buffer msg=new UTF8Buffer(object.toString());")
                    Toast.makeText(this@MainActivity,it.obj.toString(), Toast.LENGTH_SHORT).show()
                    text_test.text = it.obj.toString()
                }
                30 -> {
                    Log.d("yinlei","连接失败   UTF8Buffer msg=new UTF8Buffer(object.toString());")
                    Toast.makeText(this@MainActivity,"连接失败", Toast.LENGTH_SHORT).show()
                }
                31 -> {
                    Log.d("yinlei","连接成功")
                    Toast.makeText(this@MainActivity,"连接成功", Toast.LENGTH_SHORT).show()
                    try {
                        client.subscribe(mqtt_sub_topic,1) //订阅主题
                    }catch (e: Exception){
                        e.printStackTrace()
                    }
                }
                else -> {
                    Log.d("yinlei","未知错误")
                }
            }
            false
        }

    }


    private fun MqttInit() {
        try{
            with(options){
                //设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
                isCleanSession = false
                //设置连接的用户名
                userName = userNames
                //设置连接的密码
                password = passWord.toCharArray()
                // 设置超时时间 单位为秒
                connectionTimeout = 10
                // 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
                keepAliveInterval = 20
            }

            //设置回调
            client.setCallback(object : MqttCallback {
                override fun messageArrived(topic: String?, message: MqttMessage?) {
                    //subscribe后得到的消息会执行到这里面
                    Log.d("yinlei","messageArrived")
                    val msg = Message()
                    msg.what = 3
                    msg.obj = "${topic}___${message.toString()}"
                    handler.sendMessage(msg) //发送消息
                }

                override fun connectionLost(cause: Throwable?) {
                    //连接丢失后,一般在这里面进行重连
                    Log.d("yinlei","connectionLost")

                }

                override fun deliveryComplete(token: IMqttDeliveryToken?) {
                    //publish后会执行到这里
                    Log.d("yinlei","deliveryComplete")
                }

            })
        }catch (e: Exception){
            e.printStackTrace()
        }
    }


    private fun startReconnect() {
        scheduler.scheduleAtFixedRate({
            if (!client.isConnected){//下线了
                Mqtt_connect()
            }
        }, 0 * 1000, 10 * 1000, TimeUnit.MILLISECONDS)

    }

    private fun Mqtt_connect(){
        thread {
            try{
               if (!client.isConnected){
                   //还未连接
                   client.connect(options)
                   val msg = Message().apply { what = 31 }
                   handler.sendMessage(msg)
               }
            }catch (e: Exception){
                e.printStackTrace()
                val msg = Message().apply { what = 30 }
                handler.sendMessage(msg)
            }
        }
    }

    private fun PublishMessagepWrapper(topic: String, message2: String){
        if (!client.isConnected){
            return
        }
        val message = MqttMessage().apply { payload = message2.toByteArray() }
        try {
            client.publish(topic, message)//客户端发布消息主题
        }catch (e: MqttException){
            e.printStackTrace()
        }

    }


}

MQTT.fx的简单介绍:

MQTT与Android_第13张图片

 

MQTT与Android_第14张图片

 

MQTT与Android_第15张图片

 

MQTT与Android_第16张图片

 

MQTT与Android_第17张图片

MQTT与Android_第18张图片

MQTT与Android_第19张图片

 

 

 

你可能感兴趣的:(物联网,Android)