Socket Client + Threading + Android + статус соединения - PullRequest
2 голосов
/ 16 марта 2012

Я очень новичок как в Java, так и в разработке Android, и я разрабатываю клиент для Android, чтобы отправлять сообщения на сервер и получать сообщения обратно с сервера. Он выполняет только несколько функций, сначала он запросит у сервера исходное состояние, затем он использует это состояние для обновления кнопки и символ зарядки, чтобы сказать, заряжается зарядное устройство или нет. У меня есть весь код связи для работы, и поток, кажется, отправляет и получает сообщения достаточно хорошо, но переменные вне потока не обновляются, и если я отключу соединение с сервером, клиент будет держать соединение открытым как будто сервер все еще там. Кто-нибудь может дать мне какой-нибудь совет, чтобы исправить мои проблемы?

package charger.app;

import java.io.IOException;
import java.io.UnsupportedEncodingException;

import android.app.Activity;
import android.os.Bundle;
import android.os.StrictMode;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ImageButton;
import android.widget.Toast;


public class ChargeTabActivity extends Activity {

String serverIp = "192.168.6.120"; //server Ip address  
private static String preqon = "control:Charger:1\r\n"; //sets connection to turn charger on
private static String preqoff = "control:Charger:0\r\n";    //Turn charger off
public boolean connected = false; //connection 

//set up buttons
private ImageButton chargeOnOff;
private ImageButton connectButton;

//Set it so that Client Functions can be accessed from main program
ClientFunctions functions = new ClientFunctions();



@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    //initialize UI
    setContentView(R.layout.main);

    //declare buttons for listeners, and display changes        
    connectButton = (ImageButton)findViewById(R.id.connectButton);       
        connectButton.setOnClickListener(connectButtonListener);    
    chargeOnOff = (ImageButton)findViewById(R.id.chargeOnOff);  
        chargeOnOff.setOnClickListener(chargeOnOffListener);
    Button chargeStatus = (Button)findViewById(R.id.chargeStatus);



    ////set up strict mode off/////
        StrictMode.ThreadPolicy policy = new StrictMode.
                ThreadPolicy.Builder().permitAll().build();
                StrictMode.setThreadPolicy(policy);


}



/*Listens to see if the connect button is pressed, if it is connected it will 
 * display connected, else it will try and connect*/
private OnClickListener connectButtonListener = new OnClickListener() 
{
    public void onClick(View v)
    {
        if (connected) 
        {
            toast("I am connected");
        }
        else 
        {
            toast("I am not connected");
        }
    }
};


/*Listens to see if the Charge on Off Button is pressed, if it is charging it will 
 * send a request to charge, else it will send a request to stop the charge*/
private OnClickListener chargeOnOffListener = new OnClickListener() 
{
    public void onClick(View v)
    { 
        //ask server
        try {
            functions.updateStatus();  //updates the charger status
        } catch (IOException e) {
            Log.e("Client","Trouble querying",e);
            e.printStackTrace();
        }

        //send message to turn server on or off
        if (functions.chargeStatNumber != 0) 
        {
            try {
                functions.sendMessage(preqoff);
                chargeOnOff.setImageResource(R.drawable.charge_on);
                toast("Turning Charger off");
            } catch (UnsupportedEncodingException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }


        }//end if


        else 
        {
            try {
                functions.sendMessage(preqon);
                chargeOnOff.setImageResource(R.drawable.charge_off);
                toast("Turning Charger on");

            } catch (UnsupportedEncodingException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }


        } //end else

        Log.d("ClientActivity", "Charge Button hit" );


        //ask server about its status
        try {
            functions.updateStatus();  //updates the charger status
        } catch (IOException e) {
            Log.e("Client","Trouble querying",e);
            e.printStackTrace();
        }

    }
};


/////When Application Starts
@Override
public void onStart() 
{
    super.onStart();

    Thread cThread = new Thread(new serverThread()); //Starts the thread that initiates communications
    cThread.start();
};

////////On Pause//////This is when a user makes another app, the center point of attention 
@Override
public void onPause() {
    super.onPause();
    functions.closeConnection();

};

//////On Resume/////This is used for when an app exits, and then the user brings it back
@Override
public void onResume() {
    super.onResume();


};

//////On Destroy//////////
@Override
public void onDestroy() {
    super.onDestroy();
    functions.closeConnection();

};



//toast messages
public void toast(String toastMessage) 
{
    Toast.makeText(DelphiChargeTabActivity.this, toastMessage,
            Toast.LENGTH_LONG).show();
};



//Thread to run and receive network connections

public class serverThread implements Runnable 
{
    public boolean connected = false; //connection
    public void run() //runs the thread
    {

        try 
        {
            functions.connectToServer(serverIp); //connects to server
            connected = true;

            while (connected) 
            {
                try 
                {
                    connectButton.setImageResource(R.drawable.blue_car);
                    functions.getStreams(); //establish stream connections
                    functions. updateStatus(); //updates the status of the server
                    //checkChangeChargeButton(); //sets initial state for charge button



                    try
                    {
                        Thread.sleep(2500);
                    }
                          catch ( InterruptedException interruptedException ) 
                          {
                             functions.displayMessage( "\nThread exception" );
                          } // end catch

                }
                    catch (IOException e) 
                    {
                        Log.e("Client","Trouble Getting Streams",e);

                    }



            }//end while
        }//end try
            catch (Exception e)
            {
                Log.e("Client", "Error Connecting", e);
                connected = false;
                connectButton.setImageResource(R.drawable.red_car);

            }
    }; //end run
}; //end serverThread

public void checkChangeChargeButton() 
{

    if (functions.chargeStatNumber != 0) 
    {
        chargeOnOff.setImageResource(R.drawable.charge_on);
    }
    else 
    {
        chargeOnOff.setImageResource(R.drawable.charge_off);
    }

};



} //This is the Activity close bracket

Это код для моих методов функций

    package charger.app;
import java.io.BufferedWriter;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.UnsupportedEncodingException;
import java.net.InetAddress;
import java.net.Socket;

import android.util.Log;


public class ClientFunctions
{

private String message;
private String server;
public Socket sock;
private PrintWriter writer;
private DataInputStream inStream;
public Boolean connected;
private String encode = "UTF8";
public byte[] chargerstat;
public int  chargeStatNumber; //Used to set the array byte 15 to an int


//inquires server for status of charge
public void updateStatus() throws IOException 
{
    //Prompt user for server command
    System.out.println("Sending request for Charger status data.");
        String serverMessage = "meas:Charger?\r\n";

    // send the message
        sendMessage("meas:Charger?\r\n");
        System.out.println();
        displayMessage(serverMessage + "sent");

    // now receive the message
    receiveMessage();
        displayMessage("Receive complete");


};


public void connectToServer(String serverIp) throws IOException  //Server IP is sent in main    app as String "serverIp"
{
    //set server IP

    //display logcat trying to connect
    displayMessage("\nTrying to Connect");
    //Start Socket Connection
    sock = new Socket( InetAddress.getByName( serverIp ), 18088);
    //Display connected
    displayMessage( "\nConnected to: " + 
             sock.getInetAddress().getHostName() );



};

public void getStreams() throws IOException 
{
    //Set up PrintWriter
    writer = new PrintWriter(new BufferedWriter(new            OutputStreamWriter(sock.getOutputStream())),true);
    writer.flush();
    //Set Up I
    inStream = new DataInputStream(sock.getInputStream());
    //display Message Got I/O
    displayMessage("\nGot IO Streams");

};

public void sendMessage(final String serverMessage) throws UnsupportedEncodingException 
{


    //set variable for message
    message = serverMessage;
    //sends message
    writer.printf("%s", message);
    writer.flush();
    //display message sent
    displayMessage("\nsent" /*+ serverMessage*/);




};

public static int getInt(byte[] array, int offset) 
  {
    return
      ((array[offset+3]   & 0xff) << 24) |
      ((array[offset+2] & 0xff) << 16) |
      ((array[offset+1] & 0xff) << 8) |
       (array[offset] & 0xff);
  }


public void receiveMessage() throws IOException 
{
    {
        int c=0;
        int i;          
        int seqcountval;

        chargerstat = new byte[96];
        i=0;
        while(i < 96)
        {
            try
            {
                c = inStream.read();
                chargerstat[i] = (byte)c;
            }
            catch ( IOException ioException ) 
            {
                ioException.printStackTrace();
            } // end catch
// uncomment this to see each byte          
            //displayMessage(":" + i + " " + chargerstat[i]);
            i++;
        }

        seqcountval = getInt(chargerstat, 76);
        displayMessage("SeqCntr = " + seqcountval);
        chargeStatNumber = chargerstat[15];

    };
};

public  void closeConnection() 
{
    //closes connections
    displayMessage("\n\nClosing Connection");

    try 
    {
        writer.close();
        inStream.close();
        sock.close();


    }
    catch ( IOException ioException ) 
      {
         ioException.printStackTrace();
      } // end catch


};



public void displayMessage( final String messageToDisplay )
   {

         new Runnable()
         {
            public void run() // updates displayArea
            {
              //change to log.d for android
                System.out.println(messageToDisplay);
            } // end method run
         };  // end anonymous inner class

   } // end method displayMessage





};    

Ответы [ 3 ]

1 голос
/ 17 марта 2012

Почему бы вам не использовать AsyncTask вместо Thread, вам будет намного легче делиться информацией и, кажется, это применимо к тому, что вы пытаетесь сделать.

Использование AsyncTask

AsyncTask позволяет выполнять асинхронную работу над вашим пользователем. интерфейс. Он выполняет операции блокировки в рабочем потоке и затем публикует результаты в потоке пользовательского интерфейса, не требуя от вас обрабатывать темы и / или обработчики самостоятельно.

http://developer.android.com/guide/topics/fundamentals/processes-and-threads.html

Ваша «подключенная» переменная в ChargerActivity отличается от переменной в вашем ServerThread, поэтому подключенная ChargerActivity не будет обновляться из потока.

1 голос
/ 25 июня 2013
Функция сокета

не может быть использована в потоке пользовательского интерфейса после Android 4.0, например, функция соединения и sendMessage

0 голосов
/ 17 марта 2012

Вы не можете обновить пользовательский интерфейс из другого потока. Вы должны обновить пользовательский интерфейс в основном потоке. Если вы хотите обновить пользовательский интерфейс из другого потока, вы можете сделать что-то вроде этого

public void changeButtonImage(){
    runOnUiThread(new Runnable(){

        @Override
        public void run() {
            connectButton.setImageResource(R.drawable.blue_car);
        }});
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...