Android开发之MQTT的使用

22 篇文章 1 订阅
2 篇文章 0 订阅

首先要了解MQTT是啥子东西?   之前从来没有听说过啊

没办法,许多事情是不能等你准备就绪才可以去做的,生活就是这样,这狗JB生活,淦!

这次就把刚做好的MQTT 使用的过程记录一下 分享给大家。

如果对MQTT还不太熟悉的小伙伴可以先去百度下 了解下MQTT 是个什么东西。

MQTT官网:http://mqtt.org/

MQTT介绍:http://www.ibm.com

MQTT Android github:https://github.com/eclipse/paho.mqtt.android

MQTT API:http://www.eclipse.org/paho/files/javadoc/index.html

MQTT Android API: http://www.eclipse.org/paho/files/android-javadoc/index.html

建议时间充裕的同学有顺序的阅读上文五个链接内容,不充裕的同学请看下面简单的介绍。
 

1. Android端实现消息推送的几种方式

    轮询:客户端定时向服务器请求数据。伪推送。缺点:费电,费流量。
    拦截短信消息。服务器需要向客户端发通知时,发送一条短信,客户端收到特定短信之后,先获取信息,然后拦截短信。伪推送。缺点:贵而且短信可能被安全软件拦截。
    持久连接(Push)方式:客户端和服务器之间建立长久连接。真正的推送。
        Google的C2DM(Cloudto Device Messaging)。需要科学上网,国内大多数用户无法使用。
        XMPP。XMPP(可扩展通讯和表示协议)是基于可扩展标记语言(XML)的协议。androidpn是一个基于XMPP协议的java开源Android push notification实现。它包含了完整的客户端和服务器端。
        MQTT。MQTT是一个轻量级的消息发布/订阅协议,它是实现基于手机客户端的消息推送服务器的理想解决方案。

2 MQTT 协议
客户机较小并且 MQTT 协议 高效地使用网络带宽,在这个意义上,其为轻量级。MQTT 协议支持可靠的传送和即发即弃的传输。 在此协议中,消息传送与应用程序脱离。 脱离应用程序的程度取决于写入 MQTT 客户机和 MQTT 服务器的方式。脱离式传送能够将应用程序从任何服务器连接和等待消息中解脱出来。 交互模式与电子邮件相似,但在应用程序编程方面进行了优化。

协议具有许多不同的功能:

    它是一种发布/预订协议。
    除提供一对多消息分发外,发布/预订也脱离了应用程序。对于具有多个客户机的应用程序来说,这些功能非常有用。
    它与消息内容没有任何关系。
    它通过 TCP/IP 运行,TCP/IP 可以提供基本网络连接。

    它针对消息传送提供三种服务质量:
        “至多一次”
        消息根据底层因特网协议网络尽最大努力进行传递。 可能会丢失消息。
        例如,将此服务质量与通信环境传感器数据一起使用。 对于是否丢失个别读取或是否稍后立即发布新的读取并不重要。
        “至少一次”
        保证消息抵达,但可能会出现重复。
        “刚好一次”
        确保只收到一次消息。
        例如,将此服务质量与记帐系统一起使用。

    重复或丢失消息可能会导致不便或收取错误费用。
    它是一种管理网络中消息流的经济方式。 例如,固定长度的标题仅 2 个字节长度,并且协议交换可最大程度地减少网络流量。
    它具有一种“遗嘱”功能,该功能通知订户客户机从 MQTT 服务器异常断开连接。请参阅“最后的消息”发布。
  

首先要正确的使用这个玩意儿 需要后端去负责 或者觉得自己行的可以自行查阅资料(本人只是个菜鸟android)  传送门,走你

我要说的是android 这一部分的使用过程

第一步,添加依赖

在项目根目录下的build.gradle中添加:

repositories {
    maven {
        url "https://repo.eclipse.org/content/repositories/paho-releases/"
    }
}

然后在app目录下的build.gradle中添加:

dependencies {
    compile 'org.eclipse.paho:org.eclipse.paho.android.service:1.1.1'
    compile 'org.eclipse.paho:org.eclipse.paho.client.mqttv3:1.1.1'
}

第二步,声明权限

在AndroidManifest.xml中添加:

 

 <uses-permission android:name="android.permission.WAKE_LOCK" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.INTERNET" />

第三步,开启服务:

同样是在AndroidManifest.xml中添加:

<!-- Mqtt服务 -->
    <service android:name="org.eclipse.paho.android.service.MqttService" />

到此,上面那些做完后就可以具体的去实现了

第四步,具体实现:

我们可以把MQTTD的配置放入Service中去,所以需要创建一个Service,Android studio中可以快捷创建Service,具体操作是【File→New→Service→Service】。

这样会自动在AndroidManifest.xml声明该服务,如果你是通过创建Java类的方式创建服务的,千万别忘了在AndroidManifest.xml中进行声明。

下面是Service的代码:
 

public class MQTTService extends Service {

    public static final String TAG = MQTTService.class.getSimpleName();

    private static MqttAndroidClient client;
    private MqttConnectOptions conOpt;
    //下面这一坨找后段要
    private String host = "tcp://broker.emqx.io";  //本地測試
    private static String myTopic = "******";     //要订阅的主题
    private String userName = "admin";          //账号
    private String passWord = "password";     //密码
    private String clientId = "androidId";//客户端标识
    private IGetMessageCallBack IGetMessageCallBack;


    @Override
    public void onCreate() {
        super.onCreate();
        Log.e(getClass().getName(), "onCreate");
        init();
    }

    public static void publish(String msg) {
        String topic = myTopic;
        Integer qos = 0;
        Boolean retained = false;
        try {
            if (client != null) {
                client.publish(topic, msg.getBytes(), qos.intValue(), retained.booleanValue());
            }
        } catch (MqttException e) {
            e.printStackTrace();
        }
    }

    private void init() {
        // 服务器地址(协议+地址+端口号)
        String uri = host;
        client = new MqttAndroidClient(this, uri, clientId);
        // 设置MQTT监听并且接受消息
        client.setCallback(mqttCallback);

        conOpt = new MqttConnectOptions();
        // 清除缓存
        conOpt.setCleanSession(true);
        // 设置超时时间,单位:秒
        conOpt.setConnectionTimeout(10);
        // 心跳包发送间隔,单位:秒
        conOpt.setKeepAliveInterval(20);
        // 用户名
        conOpt.setUserName(userName);
        // 密码

        conOpt.setPassword(passWord.toCharArray());

        //将字符串转换为字符串数组

        // last will message
        boolean doConnect = true;
        String message = "{\"terminal_uid\":\"" + clientId + "\"}";
        Log.e(getClass().getName(), "message是:" + message+"\n"+myTopic);
        String topic = myTopic;
        Integer qos = 0;
        Boolean retained = false;
        if ((!message.equals("")) || (!topic.equals(""))) {
            // 最后的遗嘱
            // MQTT本身就是为信号不稳定的网络设计的,所以难免一些客户端会无故的和Broker断开连接。
            //当客户端连接到Broker时,可以指定LWT,Broker会定期检测客户端是否有异常。
            //当客户端异常掉线时,Broker就往连接时指定的topic里推送当时指定的LWT消息。

            try {
                conOpt.setWill(topic, message.getBytes(), qos.intValue(), retained.booleanValue());
            } catch (Exception e) {
                Log.i(TAG, "Exception OccuMqttConnectOptionsred", e);
                doConnect = false;
                iMqttActionListener.onFailure(null, e);
            }
        }

        if (doConnect) {
            doClientConnection();
        }

    }


    @Override
    public void onDestroy() {
        stopSelf();
        try {
            client.disconnect();
        } catch (MqttException e) {
            e.printStackTrace();
        }
        super.onDestroy();
    }

    /**
     * 连接MQTT服务器
     */
    private void doClientConnection() {
        if (!client.isConnected() && isConnectIsNormal()) {
            try {
                client.connect(conOpt, null, iMqttActionListener);
            } catch (MqttException e) {
                e.printStackTrace();
            }
        }

    }

    // MQTT是否连接成功
    private IMqttActionListener iMqttActionListener = new IMqttActionListener() {

        @Override
        public void onSuccess(IMqttToken arg0) {
            Log.i(TAG, "连接成功 "+arg0.getMessageId());
            ToastUtil.showToast("Mqtt链接成功",true);
            try {
                // 订阅myTopic话题
                client.subscribe(myTopic, 1);
            } catch (MqttException e) {
                e.printStackTrace();
            }
        }

        @Override
        public void onFailure(IMqttToken arg0, Throwable arg1) {
            arg1.printStackTrace();
            // 连接失败,重连
            ToastUtil.showToast(arg1.toString(),true);
            Log.i(TAG, "连接失败 " + arg1.toString());
            doClientConnection();
        }
    };

    // MQTT监听并且接受消息
    private MqttCallback mqttCallback = new MqttCallback() {

        @Override
        public void messageArrived(String topic, MqttMessage message) throws Exception {
            Log.i(TAG, "收到消息: " + new String(message.getPayload()));
            String str1 = new String(message.getPayload());
            if (IGetMessageCallBack != null) {
                IGetMessageCallBack.setMessage(str1);
            }
            String str2 = topic + ";qos:" + message.getQos() + ";retained:" + message.isRetained();
            Log.i(TAG, "messageArrived:" + str1);
            Log.i(TAG, str2);
        }

        @Override
        public void deliveryComplete(IMqttDeliveryToken arg0) {
            Log.e(TAG,arg0.toString());
        }

        @Override
        public void connectionLost(Throwable arg0) {
            // 失去连接,重连
        }
    };

    /**
     * 判断网络是否连接
     */
    private boolean isConnectIsNormal() {
        ConnectivityManager connectivityManager = (ConnectivityManager) this.getApplicationContext()
                .getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkInfo info = connectivityManager.getActiveNetworkInfo();
        if (info != null && info.isAvailable()) {
            String name = info.getTypeName();
            Log.i(TAG, "MQTT当前网络名称:" + name);
            return true;
        } else {
            Log.i(TAG, "MQTT 没有可用网络");
            return false;
        }
    }


    @Override
    public IBinder onBind(Intent intent) {
        Log.e(getClass().getName(), "onBind");
        return new CustomBinder();
    }

    public void setIGetMessageCallBack(IGetMessageCallBack IGetMessageCallBack) {
        this.IGetMessageCallBack = IGetMessageCallBack;
    }

    public class CustomBinder extends Binder {
        public MQTTService getService() {
            return MQTTService.this;
        }
    }

    public void toCreateNotification(String message) {
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 1, new Intent(this, MQTTService.class), PendingIntent.FLAG_UPDATE_CURRENT);
        NotificationCompat.Builder builder = new NotificationCompat.Builder(this);//3、创建一个通知,属性太多,使用构造器模式

        Notification notification = builder
                .setTicker("测试标题")
                .setSmallIcon(R.mipmap.ic_launcher)
                .setContentTitle("")
                .setContentText(message)
                .setContentInfo("")
                .setContentIntent(pendingIntent)//点击后才触发的意图,“挂起的”意图
                .setAutoCancel(true)        //设置点击之后notification消失
                .build();
        NotificationManager notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
        startForeground(0, notification);
        notificationManager.notify(0, notification);

    }
}

之前看过的博客中都是通过startService去启动服务的,这里改成了通过BindService去启动服务,所以没有onStartCommond方法。

同时,当获取从服务器推送过来的消息时,是使用回调去更新UI,这样做是为了方便代码的迁移。

上面的代码中还需要注意一个MqttAndroidClient.publish()方法,相当于服务器发布一个消息。

下面是回调接口

public interface IGetMessageCallBack {
    public void setMessage(String message);
}

 为了实现通过这个回调去传递从服务端获取到的消息,我们需要实现一个ServiceConnection类,并且通过onBind来从Service和Activity之间传递数据
 

public class MyServiceConnection implements ServiceConnection {

    private MQTTService mqttService;
    private IGetMessageCallBack IGetMessageCallBack;

    @Override
    public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
        mqttService = ((MQTTService.CustomBinder)iBinder).getService();
        mqttService.setIGetMessageCallBack(IGetMessageCallBack);
    }

    @Override
    public void onServiceDisconnected(ComponentName componentName) {

    }

    public MQTTService getMqttService(){
        return mqttService;
    }

    public void setIGetMessageCallBack(IGetMessageCallBack IGetMessageCallBack){
        this.IGetMessageCallBack = IGetMessageCallBack;
    }
}
 

来到了最后一步 那就是测试一下子 上面的代码对不对

因为我的项目中是kt,so  我就把kt 和java 的两种测试方式都写出来  大家自行选择

java版本:::::
注:xml文件只有一个 button 和一个 textview  我就不贴出来  自己添加吧

public class MainActivity extends AppCompatActivity implements IGetMessageCallBack {

    private TextView textView;
    private Button button;
    private MyServiceConnection serviceConnection;
    private MQTTService mqttService;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        textView = (TextView) findViewById(R.id.text);
        button = (Button) findViewById(R.id.button);

        serviceConnection = new MyServiceConnection();
        serviceConnection.setIGetMessageCallBack(MainActivity.this);

        Intent intent = new Intent(this, MQTTService.class);

        bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE);


        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                MQTTService.publish("TEST");
            }
        });
    }

    @Override
    public void setMessage(String message) {
        textView.setText(message);
        mqttService = serviceConnection.getMqttService();
        mqttService.toCreateNotification(message);
    }

    @Override
    protected void onDestroy() {
        unbindService(serviceConnection);
        super.onDestroy();
    }
}
 

KT版本:;;;;;;; 

注:这个也上面的差不多 应为是GS的项目 一些关键性的代码已经去掉了

class AAAAAA(context: Context) :
    AAAAAA(context), IGetMessageCallBack {
    private var serviceConnection: MyServiceConnection? = null
    private var mqttService: MQTTService? = null

    override fun getImplLayoutId(): Int {
        return R.layout.AAAAAA
    }


    @SuppressLint("SetTextI18n")
    override fun onCreate() {
        super.onCreate()
        serviceConnection = MyServiceConnection()
        serviceConnection!!.setIGetMessageCallBack(this)

        val intent = Intent(context, MQTTService::class.java)
        context.bindService(intent, serviceConnection!!, Context.BIND_AUTO_CREATE)
        // TODO: 2021/10/15  测试
//        MQTTService.publish("Test") 
    }

    override fun setMessage(message: String?) {
        // TODO: 2021/10/19    修改顯示隱藏的view
        Log.e("MQTTService",  "$message::::")
        mqttService = serviceConnection?.mqttService
        mqttService!!.toCreateNotification(message)
    }

    override fun onDestroy() {
        context.unbindService(serviceConnection!!)
        super.onDestroy()
    }
}

---------------------------------------------------------------------------------------------分割线------------------------------------------------------------------------------------------------------------------------------------------

 SO: 那么到了最后大家来一起总结下:

有人应该会问,问什么要把mqtt放到service里面, 我直接在MainActivity里面写mqtt的代码不是更简单吗

这是因为要放到后台啊!你放到activity里,当activity销毁的时候不就收不到消息了

                        有一点一直不清楚,toCreateNotification的作用是什么

调用通知栏,当收到消息的时候,嘟的一下不就数来个通知么

暂时就这吧,我继续搬砖了。。。。

如果你们看到最后的话  随文赠上一个 调试工具的地址(MQTTX)  自行下载 

  • 2
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Jonly_W

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值