关于socket.getOutputStream() 的一些问题, OutputStream的flush是一个空方法,所以需要另一个实现了Flush的流来包装一下

这里为什么使用PrintWriter,而不使用BufferedWriter

原因是在接收方使用BufferedReader 的readLine,而BufferedWriter.write并不会自动换行,所以会导致读取阻塞,需要手动换行,代码如下:

            BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
bw.write("你好啊");
// 因为在服务端使用的是readLine,所以如果不调用newLine,那么会一直阻塞
bw.newLine();
bw.flush();

以下两个测试类代码,在输出数据的时候,输出空行为结束符,在读取输入流的时候都在循环内判断了readLine长度是否为0(当然规范的做法是约定长度,根据长度判断是否结束),原因如下摘抄☞:点击这里

对于socket,不能认为把某次写入到流中的数据读取完了就算流结尾了,但是socket流还存在,还可以继续往里面写入数据然后再读取。所以用BufferedReader封装socket的输入流,调用BufferedReader的readLine方法是不会返回null的

所以在循环内如果不判断   msg!=null&&msg.length()>0  那么程序将会一直阻塞在这里(程序是因为readLine阻塞,并不是死循环)

关于流的关闭会影响socket的使用,而且对一次连接关闭流以后,没有办法再次打开,哪怕只关闭输入流,也会导致输出流不能使用.反之亦然.

所以如果在一次IO操作以后,还有另一次IO,那么就先不关闭.等全部用完再关闭.

ServerSocket

@RunWith(JUnit4.class)
public class ServerSocketTest {
@Test
public void testServer(){
ServerSocket serverSocket = null;
try {
serverSocket = new ServerSocket();
// serverSocket.setReuseAddress(true);
// System.out.println(InetAddress.getLocalHost());//获取的本机地址不一定正确
serverSocket.bind(new InetSocketAddress(8000)); while(true){
//一旦连接,返回的socket包含客户端信息的socket
Socket socket = serverSocket.accept();
PrintWriter pw = new PrintWriter(socket.getOutputStream(),true);
pw.println("host:"+socket.getInetAddress()+":"+socket.getPort()+"建立链接");
//这里发送空行作为结束符,当然规范做法是根据长度作为标识
pw.println("");
//因为new PrinWriter的时候指定了autoFlush的参数为true所以不用手动flush
// pw.flush();
BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
for(String msg = br.readLine();msg!=null&&msg.length()>0;msg = br.readLine()){
System.out.println(msg);
}
pw.close();
br.close();
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} } }

ClientSocket

/**
*
* @author lzw
*
*/
@RunWith(JUnit4.class)
public class SocketClient {
@Test
public void testClient() throws UnknownHostException, IOException{
//表示连接到服务器的 地址以及端口
SocketAddress address = new InetSocketAddress("19.95.103.112",8000);
Socket socket = new Socket();
socket.connect(address,60000);//连接
//读取服务端返回的数据
getMsb(socket);
sendMsg(socket);
socket.close();
} private void sendMsg(Socket socket){
PrintWriter pw = null;
// BufferedWriter bw = null;
try { // bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
// bw.write("你好啊");
//因为在服务端使用的是readLine,所以如果不调用newLine,那么会一直阻塞
// bw.newLine();
// bw.flush();
// OutputStream os = socket.getOutputStream();
// //这个是一个空方法
// os.flush();
pw = new PrintWriter(socket.getOutputStream(),true);
pw.println("你好啊");
       //输出空行作为结束标识
       pw.println("");
//因为new PrinWriter的时候指定了autoFlush的参数为true所以不用手动flush
// pw.flush();
} catch (IOException e) {
e.printStackTrace();
}finally{
//因为本次连接,到这个方法以后没有更多交互,所以可以关闭
if(pw!=null){
pw.close();
}
}
} private void getMsb(Socket socket){
InputStream is = null;
BufferedReader br = null;
try {
is = socket.getInputStream();
br = new BufferedReader(new InputStreamReader(is));
for(String msg = br.readLine();msg!=null&&msg.length()>0;msg = br.readLine()){
System.out.println(msg+"--");
}
} catch (IOException e) {
e.printStackTrace();
}finally{
//这里不能关闭流,否则会把socket也关闭了(因为后面还要发送数据,所以不能关闭流,不管是关闭输入输入其中之一,都会导致输入和输出都不能使用)
// try {
// if(br!=null)
// br.close();
// } catch (IOException e) {
// e.printStackTrace();
// }
}
} }

记录 serverSocket socket 输入,输出流,关闭顺序,阻塞,PrintWriter的一些问题.的更多相关文章

  1. DatagramSocket(邮递员):对应数据报的Socket概念,不需要创建两个socket,不可使用输入输出流。

    UDP编程: DatagramSocket(邮递员):对应数据报的Socket概念,不需要创建两个socket,不可使用输入输出流. DatagramPacket(信件):数据包,是UDP下进行传输数 ...

  2. Java 输入输出流 转载

    转载自:http://blog.csdn.net/hguisu/article/details/7418161 1.什么是IO Java中I/O操作主要是指使用Java进行输入,输出操作. Java所 ...

  3. java输入输出流总结 转载

    一.基本概念 1.1 什么是IO?     IO(Input/Output)是计算机输入/输出的接口.Java中I/O操作主要是指使用Java进行输入,输出操作.     Java所有的I/O机制都是 ...

  4. Java输入输出流(转载)

    转自http://blog.csdn.net/hguisu/article/details/7418161 目录(?)[+] 1.什么是IO Java中I/O操作主要是指使用Java进行输入,输出操作 ...

  5. java socket 以及 流 关闭的问题

    首先我一下几个提出问题:稍后再做出解答. 问题一:A如果仅仅将输入流关闭(inA.close()),对A与B之间的连接是否有影响? A能否再次获得输入流(inA = socketA.getInputS ...

  6. Java 输入输出流 (七)

    1.什么是IO Java中I/O操作主要是指使用Java进行输入,输出操作. Java所有的I/O机制都是基于数据流进行输入输出,这些数据流表示了字符或者字节数据的流动序列.Java的I/O流提供了读 ...

  7. Java基础学习总结(47)——JAVA输入输出流再回忆

    一.什么是IO Java中I/O操作主要是指使用Java进行输入,输出操作. Java所有的I/O机制都是基于数据流进行输入输出,这些数据流表示了字符或者字节数据的流动序列. Java的I/O流提供了 ...

  8. Java自学第10期——File类与IO流(输入输出流、处理流、转换流、缓冲流、Properties集合、打印流)

    1.IO简介 IO(输入输出)通过java.io包下的类和接口来支持,包下包括输入.输出两种IO流,每种输入输出流又可分为字符流和字节流两大类. 2.File类 File类是io包下与平台无关的文件和 ...

  9. 关于java socket中的read方法阻塞问题

    客户端: public class TCPClient { public static void main(String[] args) throws IOException { FileInputS ...

随机推荐

  1. <select>设置multiple="multiple"属性后 下拉框全部展开了 不再是折叠的怎么回事

    1 <select multiple="multiple" size="5"> <option>1</option> < ...

  2. Eclipse运行Java简单实例

    运行eclipse前首先配置好JDK环境变量等  双击这句话可跳转配置环境变量详细步骤 运行eclipse软件 1.File菜单-New - project 2.Java Project - Next ...

  3. 程序管理与SElinux

    一.程序: 1.在Linux中,触发任何一个事件是,系统都会将他定义为一个程序,并且给予这个程序一PID,同时依据启发这个程序的使用者与相关属性关系,给予这个PID一组有效的权限设定,从此以后,这个P ...

  4. python使用上下文管理器实现sqlite3事务机制

    如题,本文记录如何使用python上下文管理器的方式管理sqlite3的句柄创建和释放以及事务机制. 1.python上下文管理(with) python上下文管理(context),解决的是这样一类 ...

  5. JS前端验证代码

    手机号码正则表达式验证: function checkPhone(){ var phone = document.getElementById('phone').value; if(!(/^1[345 ...

  6. Python CRM项目六

    自定义Django Admin的action 在Django Admin中,可以通过action来自定义一些操作,其中默认的action的功能是选中多条数据来进行删除操作 我们在king_admin中 ...

  7. sql集锦

    1. emp表中取出1981年入职的员工信息--sql select * from emp where extract(year from emp.hiredate)='1981'; ...陆续添加

  8. 16_Python闭包

    一.什么是闭包 什么是闭包:内函数对外函数非全局变量的引用,并且外函数的返回值是内函数的引用(地址). def wrapper(): name = 'zhangsan' def inner(): na ...

  9. AWWWB.COM网站克隆器

    AWWWB.COM建议收费软件实行免费化的倡议书:AWWWB.COM原打算对软件中的高级功能收取260元的注册费,但是,考虑到网友使用软件的同时也是对软件的支持和传播,所以,放弃收费计划,实施一种免费 ...

  10. Spring源码情操陶冶-AnnotationConfigBeanDefinitionParser注解配置解析器

    本文承接前文Spring源码情操陶冶-自定义节点的解析,分析spring中的context:annotation-config节点如何被解析 源码概览 对BeanDefinitionParser接口的 ...