2013-01-24 2 views
7

Ich habe eine Java-Anwendung, die Voip ist. Ich benutze die eine Buchse, um Informationen über Threads gleichzeitig zu senden und zu empfangen. Code ist unten dargestellt ..Können Sie gleichzeitig in einen Eingabe- und Ausgabestream für Sockets schreiben?

Socket clientSocket = sockList.accept(); 
OutputStream outSock = clientSocket.getOutputStream(); 
InputStream inSock = clientSocket.getInputStream(); 
new Thread(new Capture(outSock)).start(); 
new Thread(new PlayAudio(inSock)).start(); 
outSock.close(); 
clientSocket.close(); 

Das Problem, das ich bin Feststellung ist, dass, wenn ich schreibe an den Output, blockiert es auf dem ersten Schreib. Ich sende nicht viele Bytes. Bellow ist mein Schreibcode.

private class Capture implements Runnable{ 

    private OutputStream out; 
    public Capture(OutputStream out){ 
     this.out = out; 
    } 
    @Override 
    public void run() { 
     try{ 
      int numBytesRead; 
      TargetDataLine outLine = getMic(); 
      outLine.open(); 
      outLine.start(); 

      byte[] data = new byte[outLine.getBufferSize()/5]; 
      byte[] test = {0x1,0x1,0x1}; 

      while(true) {  
       //numBytesRead = outLine.read(data, 0, data.length); 
       //System.out.println(numBytesRead); 
       out.write(test, 0, test.length); 
       out.flush(); 
       /*if(numBytesRead > 0){ 
        out.write(data, 0, data.length); 
        System.out.println("C"); 
       }*/ 
      } 
     }catch(Exception ex){} 
    } 
} 

Der andere Thread, der Sound Code liest ...

ist
private class PlayAudio implements Runnable{ 

    private InputStream in; 
    public PlayAudio(InputStream in){ 
     this.in = in; 
    } 
    @Override 
    public void run() { 
     int write; 
     try{ 
     SourceDataLine inLine = getSpeaker(); 
     inLine.open(); 
     inLine.start(); 
     byte[] data = new byte[inLine.getBufferSize()]; 
     byte[] test = new byte[3]; 
     while(true){ 
      System.out.println(1); 
      //write = in.read(data, 0, data.length); 
      in.read(test, 0 , test.length); 
      System.out.println(2); 
      /*if(write > 0){ 
       inLine.write(data, 0, write); 
       System.out.println(3); 
       System.out.println(write); 
      }*/ 
     } 
     } catch(Exception ex){} 
    } 

} 

ich einen guten Teil des eigentlichen Code kommentiert habe, seit ich es zur Arbeit zu kommen gerade bin versucht. Meine Schreibfunktion blockiert beim ersten Schreiben unbegrenzt. Ist es möglich, dass dies ein Problem mit meinen Threads sein könnte? Mein einziger Gedanke ist, dass die Ausgabe- und Eingabeströme mein Socket-Objekt teilen, was zu einem Deadlock oder etwas führen kann. Bitte lassen Sie mich wissen, was los ist.

Antwort

11

Ja, Sie können gleichzeitig in einen Socket-Eingangs- und Ausgangsstrom schreiben.

von do-java-sockets-support-full-duplex

Da der Eingangsstrom und der Ausgangsstrom sind separate Objekte innerhalb der Steckdose, das einzige, was Sie könnten sich beschäftigen mit ist, was passiert, wenn man zwei Fäden hatte versucht (zu lesen oder schreiben zwei Threads, gleicher Eingangs-/Ausgangsstrom) gleichzeitig? Die Lese-/Schreibmethoden der InputStream/OutputStream-Klassen sind nicht synchronisiert. Wenn Sie eine Unterklasse von InputStream/OutputStream verwenden, ist es jedoch möglich, dass die von Ihnen aufgerufenen Lese-/Schreibmethoden synchronisiert werden. Sie können das Javadoc für jede Klasse/Methode, die Sie anrufen, überprüfen und das ziemlich schnell herausfinden.

+0

Ich dachte, dass, da In- und Out (zwei getrennte Ströme) getrennte Objekte sind, es nicht wichtig ist, auf ihnen gleichzeitig zu arbeiten. Also mein Sprichwort, ich sollte mir trotzdem Sorgen um Synchronisation machen? –

4

Ja, Sie können auf Socket beim Lesen schreiben, aber Sie müssen Socket in einem unabhängigen Thread lesen. Ich benutze dieses Konzept. Hier ist das Beispiel (lesen Sie sorgfältig unterstützt mutiple Client als auch):

public class TeacherServerSocket { 

private Logger logger = Logger.getLogger(TeacherServerSocket.class); 
public static Map<String, TeacherServerThread> connectedTeacher = new HashMap<String, TeacherServerThread>(); 
ServerSocket serverSocket;; 

@Override 
public void run() { 
    // starting teacher server socket 
    this.serverSocket = startServer(); 
    // if unable to to start then serverSocket would have null value 
    if (null != this.serverSocket) { 

     while (true) { 
      //listening to client for infinite time 
      Socket socket = listenToClient(); 
      if (null != socket) { 

       TeacherServerThread teacherServerThread = new TeacherServerThread(socket); 
       Thread thread = new Thread(teacherServerThread); 
       thread.start(); 

       //putting teacher ip address and teacher object into map 
       connectedTeacher.put(teacherServerThread.getTeacherIp(),teacherServerThread); 
       System.out.println("INFO: Teacher is connected with address "+ teacherServerThread.getTeacherIp()); 

      } 

     } 


    } 

} 

@Override 
public ServerSocket startServer() { 
    //port number on which teacher server will be run. 
    int port=12345; 

    try { 
     // throw an exception if unable to bind at given port 
     ServerSocket serverSocket = new ServerSocket(port); 
     System.out.println("Teacher server socket started on port no :"+port); 
     return serverSocket; 

    } catch (IOException e) { 

     logger.error("Unable to start Teacher Server socket"); 
     e.printStackTrace(); 

    } 

    return null; 

} 

@Override 

public Socket listenToClient() { 

    if (this.serverSocket != null) { 

     try { 
      // throw an exception is unable to open socket 
      Socket socket = this.serverSocket.accept(); 
      return socket; 

     } catch (IOException e) { 

      logger.error("Unable to open socket for teacher"); 
      e.printStackTrace(); 

     } 
    } 
    else { 

     logger.error("TeacherServerSocket has got null value please restart the server"); 

    } 

    return null; 
} 





@Override 
public Map getConnectedDevicesMap() { 

return TeacherServerSocket.connectedTeacher; 

} 

/** 
* This method will send message to connected teacher which comes form student 
* @author rajeev 
* @param message, which comes form student 
* @return void 
* * */ 
@Override 
public void publishMessageToClient(String message) { 
    if(TeacherServerSocket.connectedTeacher.size()>0){ 
     System.out.println("Total Connected Teacher: "+TeacherServerSocket.connectedTeacher.size()); 
     for (String teacherIp : TeacherServerSocket.connectedTeacher.keySet()) { 

      TeacherServerThread teacherServerThread=TeacherServerSocket.connectedTeacher.get(teacherIp); 
      teacherServerThread.publishMessageToTeacher(message); 

     } 
    } 

} 



@Override 
public void stopServer() { 

    if (this.serverSocket != null) { 

     try { 

      serverSocket.close(); 

     } catch (Exception e) { 

      e.printStackTrace(); 

     } 
    } 

} 


} 

Um in einem in unabhängigen Thread für mehrere Client zu lesen:

public class TeacherServerThread implements Runnable { 


Logger logger=Logger.getLogger(TeacherServerThread.class); 
Socket socket; 
String teacherIp; 

public TeacherServerThread(Socket socket) { 
this.socket=socket; 
this.teacherIp=socket.getInetAddress().toString(); 
} 


@Override 
public void run() { 
    //starting reading 
    ReadFromTeacherAndPublishToStudent messageReader=new ReadFromTeacherAndPublishToStudent(); 
    Thread thread=new Thread(messageReader); 
    thread.start(); 
} 





private class ReadFromTeacherAndPublishToStudent implements Runnable { 

    @Override 
    public void run() { 
     String message=null; 
     try { 
      BufferedReader readTeacherData=new BufferedReader(new InputStreamReader(socket.getInputStream())); 

      StudentServerSocket studentServerSocket=new StudentServerSocket(); 
      //sending message to student which is read by teacher 
      while((message=readTeacherData.readLine())!=null){ 
       //System.out.println("Message found : "+message); 
       // studentServerSocket.publishMessageToClient(message); // do more stuff here 

      } 
      // if message has null value then it mean socket is disconnected. 
     System.out.println("INFO: Teacher with IP address : "+teacherIp+" is disconnected"); 
     TeacherServerScoket.connectedTeacher.remove(getTeacherIp()); 
     if(null!=socket){ 
      socket.close(); 
     } 

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




} 

} //class 



public void publishMessageToTeacher(String message){ 

    if(this.socket!=null){ 

     try { 

     PrintWriter writeMessageToTeacher=new PrintWriter(this.socket.getOutputStream()); 
     writeMessageToTeacher.println(message); 
     writeMessageToTeacher.flush(); 
     System.out.println(" Message published to teacher"+message); 
     }catch(Exception e){ 
     logger.error(e.toString()); 
     logger.error("Exception In writing data to teacher"); 

     } 


    }else { 
     logger.error("Unable to publish message to teacher .Socket has Null value in publishMessageToTeacher");  
     System.out.println("ERROR: socket has null value can not publish to teacher"); 
    } 



} 

public String getTeacherIp() 
{ 
    return teacherIp; 

} 
} 

Änderungscode entsprechend Ihnen Anforderung .... ..

+0

Ich verwende Threads zum Lesen und Schreiben. Ein Thread zum Lesen und der andere zum Schreiben. Außerdem kann ich PrintWriter nicht verwenden, da ich primitive Byte-Arrays senden muss, das ist, wo meine Audiodaten sind. –

0

Der Grund, warum es scheint, dass meine write() blockiert ist, weil ich dumm die Socket() geschlossen hatte und meine Eingabe Streams es nicht realisiert. Daher werden niemals Daten gesendet. Dummer Fehler in meinem Namen.