用Android Studio 创建Socket客户端向单片机发送数据

  • 首先要跟单片机通信得用到ESP8266WiFi模块


首先要跟单片机通信得用到ESP8266WiFi模块

因为我是新手小白,所以ESP8266模块的相关介绍直接贴大佬的博客了
1、ESP8266串口WiFi模块的基本使用
http://www.shaoguoji.cn/2017/01/15/ESP8266-usage/ 2、51单片机通过ESP8266模块与手机进行通讯(单片机)

之所以写这博客主要是想记个笔记哈,在网上找了好多大佬写的博客大大小小有些问题,应该是方法过时了吧,所以我要把我总结出来的方法贴出来,emmmm

1.先创建界面

android studio socket如何断开重连 android studio socket 客户端_android


1、界面代码部分

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.wulianprojects.MainActivity"
    android:orientation="vertical">

    <!--android:orientation="vertical",垂直布局,水平布局的话是horizontal-->
    <!--android:layout_width="match_parent",宽度最大化-->
    <!--android:layout_height="wrap_content",高度根据自身-->
    <!--android:background="@mipmap/titlebar_bg",背景-->
    <!--android:gravity="center",水平垂直居中-->
    <!--android:text="登录",控件文字-->
    <!--android:textSize="20sp",控件文字大小-->

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@mipmap/titlebar_bg"
        android:gravity="center">
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:gravity="center"
            android:text="客户端"
            android:textColor="@color/themeTextColor"
            android:textSize="20sp" />
    </LinearLayout>

    <EditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="服务器IP:"
        android:id="@+id/edGateIp" />

    <EditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="通信端口:"
        android:id="@+id/edGateSn" />


    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="20dp"
        android:layout_marginRight="20dp"
        android:orientation="horizontal"
        android:gravity="center">

        <Button
            android:id="@+id/btnSearch"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:textSize="24sp"
            android:text="自动填写"
            android:background="@drawable/btn_bg_round_click"
            android:textColor="@drawable/btn_click_text_color"
            />

        <Button
            android:id="@+id/btnLogin"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:textSize="24sp"
            android:text="连接服务器"
            android:background="@drawable/btn_bg_round_click"
            android:textColor="@drawable/btn_click_text_color"/>
    </LinearLayout>

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textSize="16sp"
        android:layout_marginTop="20dp"
        android:text="温馨提示:请在连接服务器WiFi的情况下进行操作哦"
        />



    <!--临时测试专用区域-->
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:orientation="horizontal">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="16sp"
            android:text="数据接收:"/>

        <EditText
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="内容"
            android:id="@+id/sjjs" />
    </LinearLayout>


    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:orientation="horizontal">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="16sp"
            android:text="数据发送:"/>

        <EditText
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="内容"
            android:id="@+id/sjfs" />
    </LinearLayout>

    <Button
        android:id="@+id/fs"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textSize="24sp"
        android:text="发送数据"
        android:background="@drawable/btn_bg_round_click"
        android:textColor="@drawable/btn_click_text_color"/>
        
</LinearLayout>

注意:因为按钮我加了美化,所以要在res文件下的drawable中建俩个.xml布局
具体方法可以参考这位大佬的博客

新建方法如下

android studio socket如何断开重连 android studio socket 客户端_android_02


1、btn_bg_round_click.xml按钮美化布局代码

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="false">
        <shape android:shape="rectangle" >
            <solid android:color="@color/color_white" />
            <corners android:radius="5dp" />
            <stroke android:width="1dp" android:color="#acacac" />
        </shape>
    </item>

    <item android:state_pressed="true" >
        <shape android:shape="rectangle">
            <solid android:color="@color/color_blue" />
            <corners android:radius="5dp" />
            <stroke android:width="1dp" android:color="#acacac" />
        </shape>
    </item>


</selector>

2、btn_click_text_color.xml文字美化布局代码

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_pressed="false" android:color="@color/color_blue"/>
    <item android:state_pressed="true" android:color="@color/color_white"/>

</selector>

3、在res文件下的values中的colors.xml颜色文件中添加两种颜色

<color name="color_blue">#FFFF00</color>
    <color name="color_white">#00BFFF</color>

总体图片如下:

android studio socket如何断开重连 android studio socket 客户端_单片机_03

2、在AndroidManifest.xml文件中添加WiFi相关的权限

<uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />

android studio socket如何断开重连 android studio socket 客户端_单片机_04

  1. MainActivity.java代码部分
public class MainActivity extends AppCompatActivity {

    //控件定义
    EditText edGateIp, edGateSn;
    Button btnSearch, btnLogin;

    //消息机制
    private Handler messageHandler;

    //开辟一个socket
    Socket socket = null;

    OutputStream OutputStream = null;//定义数据输出流,用于发送数据
    BufferedReader bufferedReader;//声明输入流对象
    InputStream InputStream = null;//定义数据输入流,用于接收数据

    //定义一个逻辑变量,用于判断服务器连接状态
    boolean isConnected = false;

    //用于控制读数据线程是否执行
    boolean RD = false;


    //测试区域
    Button fs;
    EditText sjfs, sjjs;



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

        //控件绑定
        edGateIp = findViewById(R.id.edGateIp);
        edGateSn = findViewById(R.id.edGateSn);
        btnSearch = findViewById(R.id.btnSearch);
        btnLogin = findViewById(R.id.btnLogin);

        fs=findViewById(R.id.fs);
        sjfs=findViewById(R.id.sjfs);
        sjjs=findViewById(R.id.sjjs);


        //自动填写按钮事件
        btnSearch.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                //自动填写服务器固定的IP地址跟端口号
                edGateIp.setText("192.168.4.1");
                edGateSn.setText("5000");
            }
        });

        //连接服务器按钮事件
        btnLogin.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                //容错机制
                //取出文本框内容,用来判断输入框是否为空
                String ip = edGateIp.getText().toString();
                String sn = edGateSn.getText().toString();

                if((ip== null || ip.length() == 0 )&& (sn== null || sn.length() == 0))
                    Toast.makeText(MainActivity.this, "IP地址、端口号不能为空", Toast.LENGTH_SHORT).show();
                else if(ip== null || ip.length() == 0)
                    Toast.makeText(MainActivity.this, "IP地址不能为空", Toast.LENGTH_SHORT).show();
                else if(sn== null || sn.length() == 0)
                    Toast.makeText(MainActivity.this, "端口号不能为空", Toast.LENGTH_SHORT).show();

                else {
                    //判断服务器连接状态
                    if (isConnected != true) {
                        //创建一条新的Socket连接
                        new ClientThread().start();
                        //按钮文字改变
                        btnLogin.setText("断开连接");
                        //页面消息
                        Toast.makeText(MainActivity.this, "服务器连接成功!", Toast.LENGTH_SHORT).show();
                        //二次判断,服务器是否已连接上
                        if (socket == null) {
                            //没连接的话,按钮文字改为连接服务器,页面消息提示
                            btnLogin.setText("连接服务器");
                            Toast.makeText(MainActivity.this, "连接错误,请检查WiFi是否连上,IP、端口是否输入正确!", Toast.LENGTH_SHORT).show();
                        }

                    } else {
                        //按钮按下的时候已在连接情况下,服务器断开连接
                        if (socket != null) {
                            try {
                                //退出服务器
                                socket.close();
                                //服务器状态改为空
                                socket = null;
                                //服务器连接转态改为空
                                isConnected = false;
                                //读数据线程不执行
                                RD = false;

                                btnLogin.setText("连接服务器");
                                //页面文字显示
                                Toast.makeText(MainActivity.this, "与服务器断开连接!", Toast.LENGTH_SHORT).show();
                            } catch (IOException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }
        });

        //发送按钮按下状态
        fs.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                    //这里可以用线程也可以不用线程
                    String zt = btnLogin.getText().toString();


                    if(zt == "断开连接"){
                        new sj().start();
                        if(sjfs != null && sjfs.length() > 0){
                            //handlerMessage处理发送信息刷新UI界面
                            Message msgMessage =new Message();
                            msgMessage.obj=sjfs.getText().toString();
                            msgMessage.what=0;
                            messageHandler.sendMessage(msgMessage);
                        }
                        else //页面文字显示
                            Toast.makeText(MainActivity.this, "发送的数据不能为空!", Toast.LENGTH_SHORT).show();
                    }
                    else //页面文字显示
                        Toast.makeText(MainActivity.this, "未连接服务器,请先连接!", Toast.LENGTH_SHORT).show();
            }
        });


        //消息处理机制
        messageHandler = new Handler() { // 等待socket连接成功
            @Override
            public void handleMessage(android.os.Message msgMessage) {
                String sendString="";
                String receiveString="";

                switch (msgMessage.what) {
                    case 0:
                        //append追加显示数据,之前数据不会被替换
                        sendString="用户端发送:"+msgMessage.obj.toString()+"\n";
                        sjjs.append(sendString);
                        break;

                    case 1:
                        receiveString="服务器端发送:"+msgMessage.obj.toString()+"\n";
                        //append追加显示数据,之前数据不会被替换
                        sjjs.append(receiveString);
                        break;
                    default:
                        break;
                }
            }
        };
    }

    //用线程创建Socket连接,线程不允许更新UI(用Handler实现)
    public class ClientThread extends Thread{
        public void run(){
            //定义两个变量用于储存ip跟端口号
            InetAddress GateIp;
            int GateSn;
            try {
                //判断socket的状态,防止重复执行
                if(socket == null){
                    //获取输入的IP地址
                    GateIp = InetAddress.getByName(edGateIp.getText().toString());
                    //获取输入的端口
                    GateSn = Integer.valueOf(edGateSn.getText().toString());
                    //新建一个socket,连接
                    socket = new Socket(GateIp,GateSn);
                    ///获取socket的输出流,接收数据
                    InputStream = socket.getInputStream();


                    //取得输入流、输出流
                    //取得输入流、输出流
                    bufferedReader=new BufferedReader(new InputStreamReader(socket.getInputStream()));
                    OutputStream=socket.getOutputStream();

                    //接收数据可用子线程也可直接在此线程操作
                    char[] buffer=new char[256];//定义数组接收输入流数据
                    String bufferString="";//定义一个字符接收数组数据
                    int conut =0;//初始化buffer数组长度为0
                    int tag=0;//初识写入数组的位置

                    isConnected=true;

                    //死循环重复接收输入流数据并进行处理
                    while (true) {
                        //当输入流写入buffer数组的长度大于0时即接收到数据时
                        while ((conut=bufferedReader.read(buffer))>0) {
                            //将buffer数组的数据全部写入bufferString字符类型
                            while ( tag<buffer.length) {
                                bufferString=bufferString+buffer[tag];
                                tag++;
                            }
                            //将数据给messageHandler刷新UI界面
                            Message msgMessage =new Message();
                            msgMessage.obj=bufferString;
                            msgMessage.what=1;
                            messageHandler.sendMessage(msgMessage);
                            //初始化数据,以便处理下一条输入流信息
                            tag=0;
                            bufferString="";
                        }
                    }
                }
                //出错提示
                //UnknownHostExceptionDNS解析出错
                //IOException读写文件异常
            } catch (UnknownHostException e) {
                //在命令行打印异常信息在程序中出错的位置及原因
                e.printStackTrace();
            } catch (IOException e) {
                //在命令行打印异常信息在程序中出错的位置及原因
                e.printStackTrace();
            }
        }
    }


//向服务器发送数据子程序
    public class sj extends Thread {
        public void run() {
            //判断连接状态
            if (socket != null) {
                try {
                    //判断输入框是否为空
                    if (sjfs != null && sjfs.length() > 0) {
                        //输入框内容转码后向服务器发送
                        OutputStream.write((sjfs.getText().toString()+"\n").getBytes("utf-8"));
                        //清空缓冲区
                        OutputStream.flush();
                    }
                } catch (IOException e) {
                    e.printStackTrace();
                } // 连接输出流
            }
        }
    }



}

注意,向模块发送和接收代码必须放在一个子线程中,否者会闪退运行不出来,代码加了很多容错机制,解决了空数据闪退问题哈,虽然有点乱,但还是能用的哈,现学现卖,emmmm,勿喷哈,演示图我就不贴了。。。有点麻烦,懒得调试,另外要用到两个串口调试软件,一般买模块会发资料你,这个我也不贴了,百度好像没存,嗯就是这样。