(一)基于阿里云的MQTT远程控制(Android 连接MQTT服务器,ESP8266连接MQTT服务器实现远程通信控制----简单的连接通信)

简介: 如果不了解MQTT的可以看这篇文章  http://www.cnblogs.com/yangfengwu/p/7764667.htmlhttp://www.cnblogs.com/yangfengwu/p/8026014.

如果不了解MQTT的可以看这篇文章  http://www.cnblogs.com/yangfengwu/p/7764667.html

http://www.cnblogs.com/yangfengwu/p/8026014.html

关于钱的问题,其实自己是花钱买的云服务,虽然自己现在能支付的起,但是呢为了尽量减少支出,自已还有好多好多文章要写,好多好多元器件要买,所以哈会在自己的淘宝上卖源码2元价格,一元捐出,一元自己留着当亲们支付云服务的费用了

如果看不懂也没关系,跟着做就可以了,做完以后您会发现原来MQTT这么好用,也如此简单.

对了我要尽量把程序写的烂一些,界面做的烂一些,因为既然是学习用的应该越直观越好.......说一下,自己的服务器因为公开了稳定性上肯定不好,

数据冲突也是可能的,这是第一篇,下面几篇慢慢的来,咱一块慢慢完善哈

实现的功能--手机和WIFI模块都连接MQTT服务器,手机用按钮实现远程控制一个继电器,然后WIFI模块采集的DHT11的温湿度,远程发给手机

不过自己这批贴片的板子要等到后天才到..........................

  

 

 

 

先看一下Android 程序怎么写,首先就是下载个MQTT的jar包

链接:https://pan.baidu.com/s/1bpjRzyB 密码:90vv

新建一个Android 工程就不说了吧...............

将下载的jar包放在一个地方

 

 

我放在了我的Android的源码的根目录

现在在Android 工程导入下载的那个jar包

 

 

 

 

 

 

 

 

 

现在把可能用到的一些权限加上 

 

 <!--     获取手机信息权限 -->  
    <uses-permission android:name="android.permission.INTERNET"></uses-permission>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" ></uses-permission>"
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"></uses-permission>  
    <uses-permission android:name="android.permission.CHANGE_CONFIGURATION" /> 
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />  
    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
     <uses-permission android:name="android.permission.WAKE_LOCK"/>
     <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
     <uses-permission android:name="android.permission.VIBRATE" />
     <uses-permission android:name="android.permission.WRITE_SETTINGS" />
     <uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE"></uses-permission>

 

现在呢先写个程序获取手机的IMEI号,因为连接的时候每一个客户端的ClientID要求不能一样,咱就用IMEI号代表ClientID

其实就这两句

TelephonyManager mTm = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);  
TelephonyIMEI = mTm.getDeviceId();

 

现在配置咱的MQTT

 

public class MainActivity extends Activity {
    
    String TelephonyIMEI="";
    
    private MqttClient client;//client
    private MqttConnectOptions options;//配置
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        TelephonyManager mTm = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);  
        TelephonyIMEI = mTm.getDeviceId();
        //Toast.makeText(getApplicationContext(), TelephonyIMEI, 500).show();
        MyMqttInit();
    }
    
    /*  初始化配置Mqtt  */
    private void MyMqttInit()
    {
        
        try
        {
            //(1)主机地址(2)客户端ID,一般以客户端唯一标识符(不能够和其它客户端重名)(3)最后一个参数是指数据保存在内存(具体保存什么数据,以后再说,其实现在我也不是很确定)
            client = new MqttClient("tcp://47.93.19.134:1883",TelephonyIMEI,new MemoryPersistence());
        } catch (MqttException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        
        options = new MqttConnectOptions();//MQTT的连接设置
        
        options.setCleanSession(true);//设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
        
        options.setUserName("username");//设置连接的用户名(自己的服务器没有设置用户名)
        
        options.setPassword("password".toCharArray());//设置连接的密码(自己的服务器没有设置密码)
        
        options.setConnectionTimeout(10);// 设置连接超时时间 单位为秒
        
        options.setKeepAliveInterval(20);// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
        
        client.setCallback(new MqttCallback() {
            @Override//获取的消息会执行这里--arg0是主题,arg1是消息
            public void messageArrived(String arg0, MqttMessage arg1) throws Exception {
                // TODO Auto-generated method stub
                
            }
            
            @Override//订阅主题后会执行到这里
            public void deliveryComplete(IMqttDeliveryToken arg0) {
                // TODO Auto-generated method stub
                
            }
            
            @Override//连接丢失后,会执行这里
            public void connectionLost(Throwable arg0) {
                // TODO Auto-generated method stub
                
            }
        });
    }

 

 现在连接咱的服务器,连接成功后打印一下连接成功,连接是阻塞的,所以放在一个任务里面执行连接

 

public class MainActivity extends Activity {
    
    String TelephonyIMEI="";
    
    private MqttClient client;//client
    private MqttConnectOptions options;//配置
    MqttConnectThread mqttConnectThread = new MqttConnectThread();//连接服务器任务
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        TelephonyManager mTm = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);  
        TelephonyIMEI = mTm.getDeviceId();
        //Toast.makeText(getApplicationContext(), TelephonyIMEI, 500).show();
        MyMqttInit();//初始化配置MQTT客户端
        mqttConnectThread.start();//执行连接服务器任务
    }
    
    /*  初始化配置Mqtt  */
    private void MyMqttInit()
    {
        .........
    }
    
    
    /*连接服务器任务*/
    class MqttConnectThread extends Thread
    {
        public void run()
        {
            try 
            {
                client.connect(options);//连接服务器,连接不上会阻塞在这
                runOnUiThread(new Runnable() {//
                    public void run() {
                        Toast.makeText(getApplicationContext(), "连接成功", 500).show();
                    }
                });
            } 
            catch (MqttSecurityException e) 
            {
                //安全问题连接失败
            } 
            catch (MqttException e) 
            {
                //连接失败原因
            }
        }
    }

 

 现在下载到手机试一试

 

 现在呢测试一下通信,测试接收消息,用调试助手发信息,然后手机端接收,然后显示出来

 调试助手链接

链接:https://pan.baidu.com/s/1qYxEeLI 密码:exfj

 

 现在先设置一下APP的订阅的主题,和接收到消息之后就显示出来

 

 

public class MainActivity extends Activity {
    
    String TelephonyIMEI="";
    
    private MqttClient client;//client
    private MqttConnectOptions options;//配置
    MqttConnectThread mqttConnectThread = new MqttConnectThread();//连接服务器任务
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        TelephonyManager mTm = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);  
        TelephonyIMEI = mTm.getDeviceId();
        //Toast.makeText(getApplicationContext(), TelephonyIMEI, 500).show();
        MyMqttInit();//初始化配置MQTT客户端
        mqttConnectThread.start();//执行连接服务器任务
    }
    
    /*  初始化配置Mqtt  */
    private void MyMqttInit()
    {
        
        try
        {
            //(1)主机地址(2)客户端ID,一般以客户端唯一标识符(不能够和其它客户端重名)(3)最后一个参数是指数据保存在内存(具体保存什么数据,以后再说,其实现在我也不是很确定)
            client = new MqttClient("tcp://47.93.19.134:1883",TelephonyIMEI,new MemoryPersistence());
        } catch (MqttException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        
        options = new MqttConnectOptions();//MQTT的连接设置
        
        options.setCleanSession(true);//设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
        
        options.setUserName("username");//设置连接的用户名(自己的服务器没有设置用户名)
        
        options.setPassword("password".toCharArray());//设置连接的密码(自己的服务器没有设置密码)
        
        options.setConnectionTimeout(10);// 设置连接超时时间 单位为秒
        
        options.setKeepAliveInterval(20);// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
        
        client.setCallback(new MqttCallback() {
            @Override//获取消息会执行这里--arg0是主题,arg1是消息
            public void messageArrived(String arg0, MqttMessage arg1) throws Exception {
                // TODO Auto-generated method stub
                final String topic = arg0;//主题
                final String msgString = arg1.toString();//消息
                
                runOnUiThread(new Runnable() {//
                    public void run() {
                        Toast.makeText(getApplicationContext(),"主题:"+topic+"消息:"+msgString, 500).show();
                    }
                });
            }
            
            @Override//订阅主题后会执行到这里
            public void deliveryComplete(IMqttDeliveryToken arg0) {
                // TODO Auto-generated method stub
                
            }
            
            @Override//连接丢失后,会执行这里
            public void connectionLost(Throwable arg0) {
                // TODO Auto-generated method stub
                
            }
        });
    }
    
    
    /*连接服务器任务*/
    class MqttConnectThread extends Thread
    {
        public void run()
        {
            try 
            {
                client.connect(options);//连接服务器,连接不上会阻塞在这
                
                client.subscribe("test",0);//设置(订阅)接收的主题,主题的级别是0
                
                runOnUiThread(new Runnable() {//
                    public void run() {
                        Toast.makeText(getApplicationContext(), "连接成功", 500).show();
                    }
                });
            } 
            catch (MqttSecurityException e) 
            {
                //安全问题连接失败
            } 
            catch (MqttException e) 
            {
                //连接失败原因
            }
        }
    }

下载到手机 

 

 现在配置一下软件,对了有些参数现在不明白没关系,后面会介绍一下相关的知识,

软件的主题名称要和APP中订阅的主题一样 都是 test

现在连接

 

 

 现在点击发布消息

 

 

 看手机端

 

 说明已经能通信了

 现在说一下关于主题哈,关于/

 现在把手机端的订阅的主题改为"/#"

 

 然后下载到手机

 

你会发现手机也能接收消息

 

手机都能接收到消息

# 是一个匹配主题中任意层次数的通配符。比如说,如果你订阅了test/device/#,你就可以接收到以下这些主题的消息。

test/device
test/device/后面随便是什么
咱们的设备可以用"/"来进行分类,咱们的APP呢可以指定接收哪一类的产品的数据"XXXX/#"....是不是很方便


对了如果现在接收两个已知主题的设备
假如说是

第一种方式

  

 

 

第二种方式

 

结果和上面一样

 

 现在呢在界面加一个按钮,按下发送消息"1",松开发送消息"0"

然后设置发布的主题是"/test/button"

 

 

 

 

public class MainActivity extends Activity {
    
    String TelephonyIMEI="";
    
    private MqttClient client;//client
    private MqttConnectOptions options;//配置
    MqttConnectThread mqttConnectThread = new MqttConnectThread();//连接服务器任务
    
    Button button;//发送消息按钮
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        button = (Button) findViewById(R.id.button1);//获取发送消息按钮
        button.setOnTouchListener(buttonTouch);//设置按钮的触摸事件
        
        TelephonyManager mTm = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);  
        TelephonyIMEI = mTm.getDeviceId();
        //Toast.makeText(getApplicationContext(), TelephonyIMEI, 500).show();
        MyMqttInit();//初始化配置MQTT客户端
        mqttConnectThread.start();//执行连接服务器任务
    }
    
    /*按钮触摸事件*/
    private OnTouchListener buttonTouch = new OnTouchListener() {
        
        @Override
        public boolean onTouch(View v, MotionEvent event) 
        {
            MqttMessage msgMessage = null;//Mqtt消息变量
            if (event.getAction() == MotionEvent.ACTION_DOWN) //按下
            {
                msgMessage = new MqttMessage("1".getBytes());
            }
            else if (event.getAction() == MotionEvent.ACTION_UP) //松开
            {
                msgMessage = new MqttMessage("0".getBytes());
            }
            
            try 
            {
                client.publish("/test/button",msgMessage);//发送主题为"/test/button"的消息
            } catch (MqttPersistenceException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (MqttException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            catch (Exception e) {
                //其余的状态msgMessage = null;所以加了这个catch (Exception e)
            }
            
            
            return false;
        }
    };

 

 现在下载到手机,

调试助手订阅一下主题 "/test/button"

 

 动作一下按钮

 

 

 

 现在把发过来的数据用文本框显示,不让他提示了

 

     接收显示的换一下

 

 

runOnUiThread(new Runnable() {//因为操作的是主界面的控件所以用刷新UI的线程,最好用handle哈,我这里怎么简单怎么写
                    public void run() {
                        //Toast.makeText(getApplicationContext(),"主题:"+topic+"消息:"+msgString, 500).show();
                        textView.setText("主题:"+topic+"\n消息:"+msgString);
                    }
                });

 

 现在试一下

 

 

 

好了,现在咱开始控制咱的WIFI模块了....用咱的手机控制WIFI板子上的继电器,WIFI模块呢采集温湿度,然后显示在手机的文本框中

自己更倾向于用lua开发,所以要刷入lua的固件哈

关于刷固件可以参考

http://www.cnblogs.com/yangfengwu/p/7514336.html

自己已经下载好的固件

链接:https://pan.baidu.com/s/1o8pAISy 密码:9zns

 

如果亲们自己下载的话别忘了,把mqtt和dht选择上哈

 

 

程序--init.lua

wifi.setmode(wifi.STATION)

RelayPin = 2;--RelayPin
gpio.mode(RelayPin,gpio.OUTPUT)--RelayPin
gpio.write(RelayPin,0)--RelayPin

LedPin = 4;--LedPin
gpio.mode(LedPin,gpio.OUTPUT)--LedPin
gpio.write(LedPin,0)--LedPin

DHT11pin = 5--DHT11 GPIO

Temperature = "0";--Storage temperature
Humidity = "0";--Store humidity


apcfg={}
apcfg.ssid="qqqqq"
apcfg.pwd="11223344"
wifi.sta.config(apcfg)
--wifi.sta.connect()
wifi.sta.autoconnect(1)


clientid = wifi.sta.getmac()
mqttClient=nil
mqttConnectedFlage = 0;

Mymqtt = mqtt.Client(clientid, 120,"user", "password");


--[[The connection serve]]
tmr.alarm(0, 1000, 1, function()
    Mymqtt:connect("47.93.19.134", 1883, 0,ConnectSuccess,ConnectFailed)
end)


--[[The connection Success]]
function ConnectSuccess(client)
     client:subscribe("/test/button", 0, subscribeSuccess)
                                         
     print("connected")
     mqttClient = client;
     tmr.stop(0);
     mqttConnectedFlage = 1;
end

--[[The connection fails]]
function mqttConnectFailed(client,reason)
   mqttConnectedFlage = 0;
   print("failed reason: " .. reason)
   tmr.start(0);
end


--[[The subscribe Success]]
function subscribeSuccess(client)
    print("subscribe success") 
end

--[[The Receive Msg]]
Mymqtt:on("message", function(client, topic, data) 
    if  string.find(data,"1") ~= nil then
        gpio.write(RelayPin,1)
    end

    if  string.find(data,"0") ~= nil then
        gpio.write(RelayPin,0)
    end
   
end)


--[[The Send Msg]]
tmr.alarm(1, 1000, 1, function()

    if mqttClient ~= nil and mqttConnectedFlage == 1 then
        mqttClient:publish("/test/yang","Temperature="..Temperature..";".."Humidity="..Humidity, 0, 0, 
          function(client) 
            gpio.write(4,1-gpio.read(4))
          end)
    end      
end)

--[[The gather humiture data]]
tmr.alarm(5, 2000, 1, function()--Every other 1S
    local status, temp, humi, temp_dec, humi_dec = dht.read11(DHT11pin)--Gathering temperature and humidity                 
    if status == dht.OK or status == dht.ERROR_CHECKSUM then
        Temperature = temp;
        Humidity = humi;
        --print("DHT Temperature:"..temp..";".."Humidity:"..humi)
    end
end)


printip = 0
wifi.eventmon.register(wifi.eventmon.STA_DISCONNECTED, function(T)
    printip = 0
end)


wifi.eventmon.register(wifi.eventmon.STA_GOT_IP, function(T)
   if printip == 0 then
      ip,netmask,gateway = wifi.sta.getip()
      print(gateway)
   end
   printip = 1
end)

 

现在说一下个个部分的功能,对了关于语法问题和其余的问题就请大家参考我的,其实上面的代码就是参考的官方给的API函数,

我希望亲们最重要的是有自学的能力,而不是需要别人灌输东西的机器.

 

 

 

 

 

 好了亲们可以自己去测试了

源码链接

https://item.taobao.com/item.htm?id=565355106226

wifi的就是上面的,直接复制粘贴过去就好啦

自己淘宝店的链接呢在最上面

 

相关实践学习
消息队列RocketMQ版:基础消息收发功能体验
本实验场景介绍消息队列RocketMQ版的基础消息收发功能,涵盖实例创建、Topic、Group资源创建以及消息收发体验等基础功能模块。
消息队列 MNS 入门课程
1、消息队列MNS简介 本节课介绍消息队列的MNS的基础概念 2、消息队列MNS特性 本节课介绍消息队列的MNS的主要特性 3、MNS的最佳实践及场景应用 本节课介绍消息队列的MNS的最佳实践及场景应用案例 4、手把手系列:消息队列MNS实操讲 本节课介绍消息队列的MNS的实际操作演示 5、动手实验:基于MNS,0基础轻松构建 Web Client 本节课带您一起基于MNS,0基础轻松构建 Web Client
目录
相关文章
|
26天前
|
Cloud Native Java 编译器
将基于x86架构平台的应用迁移到阿里云倚天实例云服务器参考
随着云计算技术的不断发展,云服务商们不断推出高性能、高可用的云服务器实例,以满足企业日益增长的计算需求。阿里云推出的倚天实例,凭借其基于ARM架构的倚天710处理器,提供了卓越的计算能力和能效比,特别适用于云原生、高性能计算等场景。然而,有的用户需要将传统基于x86平台的应用迁移到倚天实例上,本文将介绍如何将基于x86架构平台的应用迁移到阿里云倚天实例的服务器上,帮助开发者和企业用户顺利完成迁移工作,享受更高效、更经济的云服务。
将基于x86架构平台的应用迁移到阿里云倚天实例云服务器参考
|
24天前
|
编解码 前端开发 安全
通过阿里云的活动购买云服务器时如何选择实例、带宽、云盘
在我们选购阿里云服务器的过程中,不管是新用户还是老用户通常都是通过阿里云的活动去买了,一是价格更加实惠,二是活动中的云服务器配置比较丰富,足可以满足大部分用户的需求,但是面对琳琅满目的云服务器实例、带宽和云盘选项,如何选择更适合自己,成为许多用户比较关注的问题。本文将介绍如何在阿里云的活动中选择合适的云服务器实例、带宽和云盘,以供参考和选择。
通过阿里云的活动购买云服务器时如何选择实例、带宽、云盘
|
22天前
|
弹性计算 运维 安全
阿里云轻量应用服务器和经济型e实例区别及选择参考
目前在阿里云的活动中,轻量应用服务器2核2G3M带宽价格为82元1年,2核2G3M带宽的经济型e实例云服务器价格99元1年,对于云服务器配置和性能要求不是很高的阿里云用户来说,这两款服务器配置和价格都差不多,阿里云轻量应用服务器和ECS云服务器让用户二选一,很多用户不清楚如何选择,本文来说说轻量应用服务器和经济型e实例的区别及选择参考。
阿里云轻量应用服务器和经济型e实例区别及选择参考
|
23天前
|
机器学习/深度学习 存储 人工智能
阿里云GPU云服务器实例规格gn6v、gn7i、gn6i实例性能及区别和选择参考
阿里云的GPU云服务器产品线在深度学习、科学计算、图形渲染等多个领域展现出强大的计算能力和广泛的应用价值。本文将详细介绍阿里云GPU云服务器中的gn6v、gn7i、gn6i三个实例规格族的性能特点、区别及选择参考,帮助用户根据自身需求选择合适的GPU云服务器实例。
阿里云GPU云服务器实例规格gn6v、gn7i、gn6i实例性能及区别和选择参考
|
16天前
|
弹性计算 人工智能 安全
阿里云推出第九代ECS实例,最高提升30%性能
阿里云推出第九代ECS实例,最高提升30%性能
123 14
|
25天前
|
弹性计算
阿里云美国服务器需要备案吗?必看!
阿里云美国服务器无需ICP备案,适用于希望避开备案流程的用户。不同于中国大陆地区服务器,美国服务器及中国香港服务器均无需备案。用户可直接解析域名使用。阿里云提供美国云服务器ECS与轻量应用服务器两种选择,分别满足不同需求
86 9
|
16天前
|
人工智能 运维 Cloud Native
专访阿里云:AI 时代服务器操作系统洗牌在即,生态合作重构未来
AI智算时代,服务器操作系统面临的挑战与机遇有哪些?
专访阿里云:AI 时代服务器操作系统洗牌在即,生态合作重构未来
|
23天前
|
Java Android开发 数据安全/隐私保护
Android中多进程通信有几种方式?需要注意哪些问题?
本文介绍了Android中的多进程通信(IPC),探讨了IPC的重要性及其实现方式,如Intent、Binder、AIDL等,并通过一个使用Binder机制的示例详细说明了其实现过程。
120 4
|
移动开发 Android开发 HTML5
纯技术分析阿里云OS和Android之间的关系
注:转载一篇比较客观的文章。 前一篇《[多图论证]阿里云OS不太可能是Android以外的东西》用的ROM是错的,是Android的刷机包,不是官方原生的云OS的ROM,所以论据是错的。
3702 0
|
8天前
|
IDE Android开发 iOS开发
探索Android与iOS开发的差异:平台选择对项目成功的影响
【9月更文挑战第27天】在移动应用开发的世界中,Android和iOS是两个主要的操作系统平台。每个系统都有其独特的开发环境、工具和用户群体。本文将深入探讨这两个平台的关键差异点,并分析这些差异如何影响应用的性能、用户体验和最终的市场表现。通过对比分析,我们将揭示选择正确的开发平台对于确保项目成功的重要作用。
下一篇
无影云桌面