11

コンピューターの C# で記述されたサーバーに Android で UDP パケットを送信しようとしています。携帯電話でアプリを実行すると、不正な状態の例外が発生します。メインアクティビティでネットワーク操作を実行することに関係があるのではないかと思いますが、その問題を解決する方法がわかりません。これが私のクライアントです:

public class MainActivity extends Activity {

WifiManager wifi;
InetAddress dev_ip;
final int serverPort = 31337;
Thread drawThread = new Thread(new drawer());

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

    //set up wifi and connection
    wifi = (WifiManager) this.getSystemService(Context.WIFI_SERVICE);
    WifiInfo info = wifi.getConnectionInfo();
    int ip = info.getIpAddress();        
    String ipaddr = (ip & 0xff) + "." + (ip >> 8 & 0xff) + "." + (ip >> 16 & 0xff) + "." + (ip >> 24 & 0xff);

    try {
        dev_ip = InetAddress.getByName(ipaddr);
    } catch (UnknownHostException e) {
        Toast.makeText(this, "host error", Toast.LENGTH_LONG).show();
    }

    if (!wifi.isWifiEnabled())
       wifi.setWifiEnabled(true);

    Toast.makeText(this, "IP: " + ipaddr, Toast.LENGTH_LONG).show();     
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.activity_main, menu);
    return true;
}

public void draw(View view) throws IOException, SocketException
{
    drawThread.start();
}

public class drawer implements Runnable {

    public void run() {

        //transmit data
        try {
            DatagramSocket socket = new DatagramSocket(serverPort, /*myip*/);
            String test_data = "It works!";
            byte btest[] = new byte[50];
            btest = test_data.getBytes();
            DatagramPacket p1 = new DatagramPacket(btest, btest.length, /*myip*/, serverPort);
            socket.send(p1);
            socket.close();
        }   
        catch (IOException e) {
        }
    }
}
}

LogCat:

07-27 00:10:17.155: D/CLIPBOARD(1711): Hide Clipboard dialog at Starting input: 

finished by someone else... !
07-27 00:10:18.020: W/System.err(1711): java.net.BindException: bind failed: EADDRNOTAVAIL (Cannot assign requested address)
07-27 00:10:18.020: W/System.err(1711):     at libcore.io.IoBridge.bind(IoBridge.java:89)
07-27 00:10:18.020: W/System.err(1711):     at java.net.PlainDatagramSocketImpl.bind(PlainDatagramSocketImpl.java:68)
07-27 00:10:18.020: W/System.err(1711):     at java.net.DatagramSocket.createSocket(DatagramSocket.java:133)
07-27 00:10:18.020: W/System.err(1711):     at java.net.DatagramSocket.<init>(DatagramSocket.java:95)
07-27 00:10:18.020: W/System.err(1711):     at com.ls.styloid.MainActivity$drawer.run(MainActivity.java:67)
07-27 00:10:18.025: W/System.err(1711):     at java.lang.Thread.run(Thread.java:856)
07-27 00:10:18.025: W/System.err(1711): Caused by: libcore.io.ErrnoException: bind failed: EADDRNOTAVAIL (Cannot assign requested address)
07-27 00:10:18.025: W/System.err(1711):     at libcore.io.Posix.bind(Native Method)
07-27 00:10:18.025: W/System.err(1711):     at libcore.io.ForwardingOs.bind(ForwardingOs.java:39)
07-27 00:10:18.025: W/System.err(1711):     at libcore.io.IoBridge.bind(IoBridge.java:87)
07-27 00:10:18.025: W/System.err(1711):     ... 5 more
07-27 00:10:42.090: D/CLIPBOARD(1711): Hide Clipboard dialog at Starting input: finished by someone else... !
07-27 00:11:30.150: W/System.err(2535): java.net.BindException: bind failed: EADDRNOTAVAIL (Cannot assign requested address)
07-27 00:11:30.155: W/System.err(2535):     at libcore.io.IoBridge.bind(IoBridge.java:89)
07-27 00:11:30.155: W/System.err(2535):     at java.net.PlainDatagramSocketImpl.bind(PlainDatagramSocketImpl.java:68)
07-27 00:11:30.155: W/System.err(2535):     at java.net.DatagramSocket.createSocket(DatagramSocket.java:133)
07-27 00:11:30.155: W/System.err(2535):     at java.net.DatagramSocket.<init>(DatagramSocket.java:95)
07-27 00:11:30.155: W/System.err(2535):     at com.ls.styloid.MainActivity$drawer.run(MainActivity.java:67)
07-27 00:11:30.155: W/System.err(2535):     at java.lang.Thread.run(Thread.java:856)
07-27 00:11:30.155: W/System.err(2535): Caused by: libcore.io.ErrnoException: bind failed: EADDRNOTAVAIL (Cannot assign requested address)
07-27 00:11:30.155: W/System.err(2535):     at libcore.io.Posix.bind(Native Method)
07-27 00:11:30.155: W/System.err(2535):     at libcore.io.ForwardingOs.bind(ForwardingOs.java:39)
07-27 00:11:30.155: W/System.err(2535):     at libcore.io.IoBridge.bind(IoBridge.java:87)
07-27 00:11:30.155: W/System.err(2535):     ... 5 more
07-27 00:11:36.515: D/CLIPBOARD(2535): Hide Clipboard dialog at Starting input: finished by someone else... !

編集: サーバーには、以前は気付かなかった多くの問題があるようです。回答の1つに従ってリスナーを書き直したときに、それらが発生し始めました。label3、ソケット例外 0x80004005 で「破棄されたオブジェクトにアクセスできません」というエラーが発生することがありますが、まだパケットを受信して​​いません。ただし、ソケットの状態を確認すると、読み取り可能のように見えます。私はおそらくスレッドを台無しにしました。これを修正してください。サーバ:

public partial class Form1 : Form
    {
        Socket listener;
        Thread udp_listener;

        public Form1()
        {
            InitializeComponent();

            //set up listener thread
            udp_listener = new Thread(listen);
            udp_listener.IsBackground = true;
            udp_listener.Start();
        }

        protected override void OnFormClosing(FormClosingEventArgs e)
        {
            base.OnFormClosing(e);
            listener.Close();
            udp_listener.Join();
        }

        private void listen()
        {
            //set up UDP
            const int serverPort = 31337;
            bool terminate = false;
            IPHostEntry iphost = Dns.GetHostEntry(Dns.GetHostName());
            IPAddress ipaddr = iphost.AddressList[0];
            IPEndPoint endpoint = new IPEndPoint(ipaddr, serverPort);
            listener = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
            label3.Text = ipaddr.ToString();

            try
            {
                do
                {
                    byte[] buffer = new byte[100];
                    listener.Receive(buffer);
                    label3.Text = "Connected";
                    label3.ForeColor = Color.Red;
                    label3.Text = Encoding.UTF8.GetString(buffer);
                }
                while (!terminate);
            }
            catch (Exception e)
            {
                MessageBox.Show(e.ToString());
            }
            finally
            {
                listener.Close();
            }

            listener.Close();
        }
    }

EDIT2:

パソコンでC#でクライアントを作ってみました。パケットは送信されましたが、サーバーは何も受信しませんでした。

EDIT3: サーバーは正常に動作しますが、Android アプリは実行を拒否します。コードは次のとおりです。

   package com.tests.contest;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.net.Socket;
import java.net.UnknownHostException;
import android.app.Activity;

public class MainActivity extends Activity {
      private Socket sock;
      private BufferedWriter out;
      private Thread thrd;

      @Override
      public void onResume() {
        super.onResume();
        thrd = new Thread(new Runnable() {
            public void run() {
              while (!Thread.interrupted()) {
                runOnUiThread(new Runnable() {
                  @Override
                  public void run() {
                      try {
                        sock = new Socket("THEIP", 31337);
                    } catch (UnknownHostException e1) {
                        // TODO Auto-generated catch block
                        e1.printStackTrace();
                    } catch (IOException e1) {
                        // TODO Auto-generated catch block
                        e1.printStackTrace();
                    }
                      try {
                        out = new BufferedWriter(new OutputStreamWriter(sock
                              .getOutputStream()));
                        out.write("WORKS");
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                  }
                });
              }
            }
          });
          thrd.start();
      }

      @Override
      public void onPause() {
        super.onPause();
        if (thrd != null)
          thrd.interrupt();
        try {
          if (sock != null) {
            sock.getOutputStream().close();
            sock.getInputStream().close();
            sock.close();
          }
        } catch (IOException e) {}
        thrd = null;
      }

      /*private void sendText() {
        String text = "HI";
        try {
          out.write(text + "\n");
          out.flush();
        } catch (IOException e) {}
      }*/
    }

この問題は、明らかに実行していないメイン スレッドでネットワーク操作を実行しているために発生します。

4

2 に答える 2

7

例外は、何をすべきかを正確に伝えることです:Socketオブジェクトを別の に作成しますThread。これにもを使用できますAsyncTask

Socketsメイン UI スレッドで許可しない理由Application Not Respondingは、Socket.

編集: http://thinkandroid.wordpress.com/2010/03/27/incorporating-socket-programming-into-your-applications/

すでにデスクトップ サーバーがあるので、クライアントの例までスキップできます。

edit2 : Android アプリ用の C# サーバーでも作業しているので、デスクトップ アプリでリスナーを作成する方法は次のSocketとおりです。

IPHostEntry ipHostInfo = Dns.GetHostEntry(Dns.GetHostName());
IPAddress ipAddress = ipHostInfo.AddressList[0];
IPEndPoint localEndPoint = new IPEndPoint(ipAddress, 27015); //Port 27015
Socket listener = new Socket(AddressFamily.InterNetwork,
    SocketType.Stream, ProtocolType.Tcp); // Create a TCP/IP socket.

問題がサーバー/ネットワーク内にあることはほぼ確実です。単純な UDP パケットを送信する Android コードは正しいようです。また、簡単な C# クライアント プログラムを作成して接続することもlocalhostできます。127.0.0.1

編集3:

この基本的なアプリでは、正しい IP アドレスを入力すると、サーバーに接続できるはずです。[送信] を押すと、サーバーに数バイトが送信されます。これを使用して、接続が機能していることを確認します。私はそれが私の側で動作することを確認しました。これが機能する場合は、サーバーの問題について新しい SO の質問を開きます。それ以外の場合は、ネットワーク構成に問題があります。

于 2012-07-26T19:50:47.147 に答える
2

ループ内で UI コンポーネント ( label3 ) を変更しようとしているため、問題は do while ループから発生していると思います。これは、終了変数が常に falseであるという無限ループの罪によるものです。UI を変更するコード ( label3.*) をループから外してみてください。

于 2012-07-29T03:11:36.210 に答える