Сокеты Android Java задерживают отправку данных

ОБНОВЛЕНИЕ: Спасибо за мысли. Я отладил проблему, я считаю, вплоть до соединения между андроидом и ESP8266. Я считаю, что может быть что-то не так с ESP, потому что, когда я подключаю Android к своему компьютеру, кажется, что он подключается и работает хорошо. Есть очень незначительные задержки, но не такие, как при подключении к ESP. Поэтому я думаю, что ESP вызывает задержку. При этом я не уверен, что могло вызвать это. Я пытался использовать ESP8266 версии 2.3.0 библиотек в Arduino IDE 1.6.8, но не могу понять, почему задержка. Я подключил приложение для Android к своему внутреннему маршрутизатору, а затем без проблем отправил его на ПК. Но с ESP не работает. Я не могу поверить, что я единственный, кто это видит... Если у кого-то еще это работает, пожалуйста, дайте мне знать, как вы это сделали. Моя вечная благодарность будет тебе.

Вот код ESP Arduino:

WiFiServer server(2390); // TCP/IP
SoftwareSerial mySerial( 4, 5);    // RX, TX
void setup( ) 
{
    Serial.begin( 115200 );
    WiFi.mode(WIFI_STA);
    WiFi.softAP( "mySSID", "myPASS");

    server.begin();// START TCP/IP SERVER 
}
WiFiClient client;
void loop( ) 
{
    if (client) 
    {
        while (client.connected()) 
        {
            if (client.available()) 
            {
                char command = client.read();
                Serial.println(itoa(command,buffer,10));
            }
        }
    }
    else
    {
        client = server.available();
    }
}

Я пытаюсь получить почти мгновенную связь, используя сокеты TCP/IP между Android (Java) и ESP8266. Проблема, которую я вижу, заключается в значительной задержке между нажатием кнопки и моментом, когда андроид действительно отправит данные. Кажется, иногда он отправляется сразу, когда я нажимаю кнопку «Перейти», но после первого раза он не отправляется в течение 5-10 секунд. Когда он застревает, я нажимаю кнопку «Перейти» несколько раз, и после задержки кажется, что все они отправляются сразу или начинают отправляться рывками. Я смотрю на сетевой монитор в Android Studio, и задержка заметна с момента нажатия кнопки до появления трафика на сетевой панели на вкладке «Мониторы».

Я использую отладчик Android Studio для тестирования кода.

Есть ли что-то, что я делаю не так? Или, возможно, есть лучшая реализация для почти мгновенной связи между устройствами?

Спасибо за вашу помощь заранее!

Android-устройство: Samsung Note 4 — v5.1.1 Android v2.0

Android-студия

Устройство ESP: ESP8266 с использованием Arduino Studio 1.6.8

Вот мой код:

public class MainActivity extends AppCompatActivity 
{
    private Socket socket;


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

    public void onClick(View view) 
    {
        try 
        {
            new AsyncTaskRunner().execute("");
        } 
        catch (Exception e) 
        {
            e.printStackTrace();
        }
    }

    private class AsyncTaskRunner extends AsyncTask<String, String, String> 
    {
        private String resp;
        ProgressDialog progressDialog;

        @Override
        protected String doInBackground(String... params) 
        {
            try 
            {
                DataOutputStream dataOut;
                Socket s = new Socket("192.168.4.1", 2390);

                dataOut=new DataOutputStream(s.getOutputStream());
                byte[] outputData = new byte[9];
                outputData[0] = 1;
                outputData[1] = 2;
                outputData[2] = 3;
                outputData[3] = 4;
                outputData[4] = 5;
                outputData[5] = 6;
                outputData[6] = 7;
                outputData[7] = 8;
                outputData[8] = 9;
                dataOut.write(outputData); // Red
                dataOut.flush();
                dataOut.close();
                s.close();
            }
            catch (UnknownHostException e)
            {
                e.printStackTrace();
            }
            catch (IOException e)
            {
                e.printStackTrace();
            }

            return resp;
        }

        @Override
        protected void onPostExecute(String result) {
        }

        @Override
        protected void onProgressUpdate(String... text) {
        }
    }
}

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.jered.networktest.MainActivity">

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Go"
        android:onClick="onClick"/>

</RelativeLayout>

person Edster    schedule 04.07.2016    source источник
comment
код андроида выглядит нормально. может быть проблема с кодом сервера.   -  person njzk2    schedule 04.07.2016
comment
AsyncTask может вызвать дополнительные проблемы. Попробуйте использовать поток и обработчик   -  person Mark Shen    schedule 04.07.2016
comment
Вы устанавливаете сокетное соединение каждый раз, когда хотите отправить сообщение. Почему бы вам не подумать о создании сокетного соединения в службе Intent и поддерживать его до тех пор, пока вы явно не закроете соединение.   -  person Arpit Ratan    schedule 04.07.2016
comment
@Mark Shen - я пытался использовать поток и обработчик, но вижу ту же проблему. Вроде стало лучше, правда после 3-4 нажатий есть задержка. Далее я собираюсь попробовать метод Intent Service.   -  person Edster    schedule 04.07.2016
comment
Та же проблема и у меня, но при включении режима полета все работает без задержек. Кто-нибудь знает, как это исправить?   -  person vishva vijay    schedule 01.12.2020


Ответы (2)


У меня была эта проблема для случая, когда я хотел передать некоторые данные на точку доступа Esp8266-01. Я думаю, что эта задержка произошла из-за того, что if (client. available()) внутри while (client.connected())loop ожидает получения данных, пока клиент не будет доступен. следовательно, делает задержку (у меня было ровно 5 секунд). чтобы избежать этого неприятного поведения, я добавил возврат каретки '\r' в команду записи AsyncTask в файле Android MainActivity.java, а затем изменил эскиз Arduino, чтобы дождаться, пока сервер не получит возврат каретки, я имею в виду такой код, как этот String request = client.readStringUntil('\r'); наконец, две кнопки как известные как btn1 и btn2 в приложении для Android, отправляют две строки /led/on и /led/off соответственно. к настоящему времени ESP8266 получает возврат каретки и немедленно очищает клиент, после чего нет задержки для выполнения другого раздела программы. Следующая программа включает или выключает один светодиод (контакт 0 GPIO ESP8266-01) в зависимости от нажатой кнопки в приложении Android. но не забудьте установить заземление GPIO 0 ESP8266-01 только во время его программирования и не забудьте сначала подключить мобильный Wi-Fi к сети Wi-Fi ESP, но вы также можете использовать конфигурацию Wi-Fi для автоматического подключения к Wi-Fi ESP (см. это ссылка).

Код Ардуино:

#include<ESP8266WiFi.h>
// WiFi Definitions
const char* ssid = "Esp8266TestNet";
const char* password = "Esp8266Test"; // has to be longer than 7 chars
const char* value = "";

int ledPin = 0; // GPIO 0
WiFiServer server(80);

void setup() {

   Serial.begin(115200);
   delay(10);
   pinMode(ledPin, OUTPUT);
   digitalWrite(ledPin, HIGH); // turn on

   WiFi.mode(WIFI_AP);
   WiFi.softAP(ssid, password, 1, false);
  
   server.begin();
}

void loop() {
  // Check of client has connected
  WiFiClient client = server.available();
  if(!client) {
    return;
  }

  // Read the request line
  String request = client.readStringUntil('\r');
  Serial.println(request);
  client.flush();
  
  // Match request
  if(request.indexOf("/led/on") != -1) {
     digitalWrite(ledPin, HIGH);
     value = "on";
  } else if (request.indexOf("/led/off") != -1) {
     digitalWrite(ledPin, LOW);
      value = "off";
  }
  else {
    Serial.println("not validate");
    }
  
  client.flush();
   
  // JSON response
  String s = "HTTP/1.1 200 OK\r\n";
  s += "Content-Type: application/json\r\n\r\n";
  s += "{\"data\":{\"message\":\"success\",\"value\":\"";
  s += value;
  s += "\"}}\r\n";
  s += "\n";

  // Send the response to the client
  client.print(s);
  delay(1);
  Serial.println("Client disconnected");

  // The client will actually be disconnected when the function returns and the client object is destroyed
}

XML-файл манифеста:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.myhgrtest">
    <uses-permission android:name="android.permission.INTERNET"/>

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

Activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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=".MainActivity">
    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/btn1"
        android:text="btn1"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        ></Button>
    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/btn2"
        android:text="btn2"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        ></Button>
</androidx.constraintlayout.widget.ConstraintLayout>

Основная активность.java:

package com.example.myhgrtest;

import androidx.appcompat.app.AppCompatActivity;

import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.HttpURLConnection;
import java.net.Socket;
import java.net.URL;

public class MainActivity extends AppCompatActivity {
    Button btn1,btn2;
    TcpClient mTcpClient;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn1=findViewById(R.id.btn1);
        btn2=findViewById(R.id.btn2);
        btn1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

                send_request1();
            }
        });
        btn2.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                send_request2();
            }
        });
    }
    public class ConnectTask extends AsyncTask<String, String, TcpClient> {

        @Override
        protected TcpClient doInBackground(String... message) {

            //we create a TCPClient object
            mTcpClient = new TcpClient(new TcpClient.OnMessageReceived() {
                @Override
                //here the messageReceived method is implemented
                public void messageReceived(String message) {
                    //this method calls the onProgressUpdate
                    publishProgress(message);
                }
            });
            mTcpClient.run();

            return null;
        }

        @Override
        protected void onProgressUpdate(String... values) {
            super.onProgressUpdate(values);
            //response received from server
            Log.d("test", "response " + values[0]);
            //process server response here....

        }
    }
    public void send_request1() {
        send_request1 sr = new send_request1();
        sr.execute();
    }
    public void send_request2() {
        send_request2 sr = new send_request2();
        sr.execute();
    }
    class send_request1 extends AsyncTask<Void, Void, String> {

        @Override
        protected String doInBackground(Void... voids) {
            try {             /*note : ip address must be in Quotation mark*/
                /*note : port doesn't need to be inside the Quotation mark*/
                Socket s = new Socket(/*ip address :*/"192.168.4.1",/*port :*/ 80);
                PrintWriter printWriter = new PrintWriter(s.getOutputStream());
                printWriter.write("/led/on"+"\r");
                printWriter.flush();
                printWriter.close();
                s.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return null;
        }

        @Override
        public void onPostExecute(final String Data) {
            /*Data is what you receive from your server*/
            Log.e("my_Data","Data is : " + Data);
        }
    }
    class send_request2 extends AsyncTask<Void, Void, String> {

        @Override
        protected String doInBackground(Void... voids) {
            try {
                /*note : ip address must be in Quotation mark*/
                /*note : port doesn't need to be inside the Quotation mark*/
                Socket s = new Socket(/*ip address :*/"192.168.4.1",/*port :*/ 80);
                PrintWriter printWriter = new PrintWriter(s.getOutputStream());
                printWriter.write("/led/off"+"\r");
                printWriter.flush();
                printWriter.close();
                s.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return null;
        }

        @Override
        public void onPostExecute(final String Data) {
            /*Data is what you receive from your server*/
            Log.e("my_Data","Data is : " + Data);
            //Toast.makeText(MainActivity.this, Data, Toast.LENGTH_SHORT).show();
        }
    }

}
person AmirHosein Ghenaati    schedule 09.02.2021

Я не думаю, что кто-то может сказать наверняка, в чем проблема. Чтобы вы (и другие) не гонялись за призраками, вы действительно хотите разделить свою проблему:

  • Убедитесь, что ваш 'doInBackground' не задерживается, например. войдя в систему ведения журнала Android, консоль, отладчик или выполнив что-то простое в пользовательском интерфейсе.
  • Убедитесь, что отправка на стороне Android не задерживается, написав крошечный слушатель, например. Java и запустить его на своем ПК.
  • Убедитесь, что прослушивание не задерживается, создав крошечный отправитель на Java и запустив его на своем ПК.

Таким образом, разделите свою проблему, пока не найдете основную причину :)

person bluemind    schedule 05.07.2016