【问题标题】:SocketException - 'Socket is closed' even when isConnected() returns trueSocketException - 'Socket is closed' 即使 isConnected() 返回 true
【发布时间】:2012-04-15 15:25:37
【问题描述】:

我正在制作一个使用套接字通信将图像发送到计算机上运行的 java 应用程序的 android 应用程序

这是发生了什么:桌面运行服务器 java 应用程序,客户端 android 应用程序运行在设备上,它将图像传输到服务器,这部分运行良好。之后,服务器应用程序从控制台接收一行并将其传递回 android 应用程序。直到 android 应用程序收到该消息,它应该显示一个进度对话框并且它卡在那里。 android 应用程序应该使用 readLine() 读取桌面应用程序传递的字符串,但是当我尝试通过 android 应用程序中的套接字打开输入流时,它给了我异常。

以下是代码,首先是桌面服务器,然后是安卓客户端

导入 java.net.ServerSocket; 导入 java.net.Socket; 导入 java.io.*;

class ProjectServer
{
    ServerSocket serSock;
    Socket  sock;

    BufferedReader in;
    PrintWriter out;
    public static void main(String ar[])
    {
        try
        {
            ProjectServer cs=new ProjectServer();
            cs.startServer();
        }
        catch(Exception e)
        {

        }
    }

    public void startServer()
    {
        try
        {
            serSock=new ServerSocket(8070);
            System.out.println("Waiting for client...");
            sock=serSock.accept();


            System.out.println("Connections done");


            //Accept File
            System.out.println("Connected");
            System.out.println(sock.isConnected()+"1");

            //receive code

            int filesize=450660;
            int bytesRead;
            int current=0;
            // receive file
            byte [] mybytearray  = new byte [filesize];
            InputStream is = sock.getInputStream();
            FileOutputStream fos = new FileOutputStream("C:\\Project Server\\Capture.png");
            BufferedOutputStream bos = new BufferedOutputStream(fos);
            bytesRead = is.read(mybytearray,0,mybytearray.length);
            current = bytesRead;


            do {
               bytesRead =
                  is.read(mybytearray, current, (mybytearray.length-current));
               if(bytesRead >= 0) current += bytesRead;
            } while(bytesRead > -1);

            bos.write(mybytearray, 0 , current);
            bos.flush();
            System.out.println("end-start");
            bos.close();
            //sock.close();
            //receive code ends

            //System.out.println(br.readLine());


            //Matlab computation

            //Send result
            System.out.println(sock.isConnected()+"2");
            PrintWriter pr=new PrintWriter(sock.getOutputStream(),true);

            pr.println((new BufferedReader(new InputStreamReader(System.in))).readLine());
            System.out.println(sock.isConnected()+"3");
            (new BufferedReader(new InputStreamReader(System.in))).readLine();
            System.out.println(sock.isConnected()+"4");
        }
        catch(Exception e)
        {
            System.out.println(e);
            e.printStackTrace();
        }

    }
}

Android 客户端:

package com.site.custom;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;

import android.app.Activity;
import android.app.ProgressDialog;
import android.os.Bundle;
import android.util.Log;
import android.widget.TextView;


public class Act2 extends Activity
{
    private ProgressDialog pd;
    private String serverIP="58.146.100.187";
    private BufferedReader in;
    private PrintWriter out;
    private String path;
    private Socket cliSock;

    public void onCreate(Bundle onCreateInstance)
    {
        super.onCreate(onCreateInstance);
        setContentView(R.layout.act2);
        this.setTitle("This has started");
        path=getIntent().getStringExtra("path");

        //Establish Connection
        //pd=ProgressDialog.show(this, "Establishing connection", "Finding server",false,true);

        try
        {
            cliSock=new Socket(serverIP,8070);

            //pd.dismiss();
            //Log.v("MERA MSG","changing text");
            ((TextView)findViewById(R.id.tview)).setText(path);
        }
        catch(Exception e)
        {
            Log.v("MERA MSG",e.toString());
        }

        //Send file
        //Log.v("MERA MSG","changing text1");

        ProgressDialog pd=ProgressDialog.show(this, "Sending image", "Image chosen:"+path.substring(path.lastIndexOf("//")+1),false,true);
        //Log.v("MERA MSG","changing text2");

        try
        {


            File myFile = new File (path);
            System.out.println((int)myFile.length());
            byte[] mybytearray  = new byte[450560];
            FileInputStream fis = new FileInputStream(myFile);
            BufferedInputStream bis = new BufferedInputStream(fis);
            bis.read(mybytearray,0,mybytearray.length);
            OutputStream os = cliSock.getOutputStream();
            System.out.println("Sending...");
            os.write(mybytearray,0,mybytearray.length);
            os.flush();
            os.close();
            bis.close();
            //sock.close();
            //System.out.println("Completed");
            System.out.println(cliSock.isConnected()+"1");
            pd.dismiss();
            //System.out.println("Done");
            System.out.println(cliSock.isConnected()+"2");
            //Show dialog box till computation results arrive
            pd=ProgressDialog.show(this, "Recognizing...", "(waiting for server reply)",false,true);
            System.out.println(cliSock.isConnected()+"3");
            in=new BufferedReader(new InputStreamReader(cliSock.getInputStream()));
            System.out.println(in.readLine());
            pd.dismiss();

        }
        catch(Exception e)
        {
            Log.v("MERA MSG",e.toString());
            e.printStackTrace();
        }



    }
}

顺便说一句,这里是来自 LogCat 的日志: http://pastebin.com/atHMycTa 您可以在此处确认套接字仍处于连接状态,因为它在第 350 和 349 行显示为 true 查看第 351 行的错误。

【问题讨论】:

  • 我不确定这是否是原因,但您可以在从套接字读取数据后尝试将os.close() 调用移至。我不记得关闭输出流是否也会关闭套接字,或者仅在关闭输入流时才会发生这种情况..
  • @patrick 不,没有,看我的回答。
  • @EJP:哦,对不起,一定是把它和别的东西搞混了..
  • @Patrick “不,它没有”是指您的声明“isConnected 返回上一个操作的连接状态”。关闭 Socket 的任一流都会关闭套接字和另一个流。
  • @EJP:是的,我在写评论后注意到了。这就是我删除第一个的原因;)

标签: java android sockets


【解决方案1】:

isConnected() 只告诉你是否曾经连接过 this Socket, 你拥有的,所以它返回 true。它不是连接状态的指示。

【讨论】:

  • @GrowinMan 你的问题是你已经关闭了你的Socket,然后继续使用它。关闭输出流会关闭套接字。只需刷新它,然后获取输入流,然后继续。你也不需要那些ByteArrayOutputStreams。他们只是在浪费时间和空间。从文件读取,写入套接字,或反之亦然。
  • flush command() 似乎不起作用。如果我只是执行flush(),那么服务器仍然会继续等待接收图像。只有在我执行 os.close() 之后,它才会真正刷新流并且服务器完全接收图像并继续执行。有没有一种方法可以在不关闭输出流的情况下刷新它?
  • @GrowinMan 服务器正在读取 EOS。所以你需要寄一份。刷新后调用 cliSock.shutdownOutput()。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-01-10
  • 1970-01-01
  • 2019-12-15
  • 2016-11-14
  • 1970-01-01
  • 1970-01-01
  • 2018-11-29
相关资源
最近更新 更多