通过练习掌握TCP在进行传输过程中的问题



练习1:创建一个英文大写转换server

client输入字母数据,发送给服务端,服务端收到后显示到控制台,并将该数据转成大写返回client,知道client输入over,转换结束

public class Main {

	public static void main(String[] args) throws IOException{
Text_Transform_Client();
Text_Transform_Server();
} public static void Text_Transform_Server() throws IOException {
//文本转服务端 /* 转换服务端
* 1.创建ServerSocket服务端对象
* 2.获取Socket对象
* 3.源:Socket,读取client发过来须要转换的数据
* 4.汇:显示在控制台
* 5.将数据转成大写返回client
*/
//创建服务端对象
ServerSocket ss = new ServerSocket(6534); //获取socket对象
Socket socket = ss.accept(); //获取ip,明白是谁连进来的
String ip = socket.getInetAddress().getHostAddress();
System.out.println("ip : "+ip); //获取socket读取流,并装饰
BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
//获取socket输出流,并装饰
PrintWriter pw = new PrintWriter(socket.getOutputStream(),true);
//new PrintWriter(socket.Outputtream())
String line = null;
while((line = br.readLine())!=null){
System.out.println(line);
pw.println(line.toUpperCase());//pw.print(line.tpUpperCase+"\r\n");
} //pw.flush();
socket.close();
ss.close();
} public static void Text_Transform_Client() throws IOException{
//文本转换client
/*
* 转换client:
* 1.创建Socketclient对象
* 2.获取键盘录入
* 3.将录入的信息,发送给Socket输出流
*/
Socket socket = new Socket("127.0.0.1",6534);
BufferedReader br =
new BufferedReader(new InputStreamReader(System.in)); //源是:键盘,汇:Socket输出流
//new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
PrintWriter pw = new PrintWriter(socket.getOutputStream(),true);
//new PrintWriter(socket.getOutstream());
//Socket输入流,读取服务端返回的大写数据
BufferedReader br2 = new BufferedReader(new InputStreamReader(socket.getInputStream())); String line = null;
while((line = br.readLine())!=null){
if("over".equals(line))break; pw.println(line);//pw.print(line+"\r\n")
//pw.flush();
//读取服务端返回的大写信息
String str = br2.readLine();
System.out.println("up : "+str);
}
socket.close();
}
}

常见问题:

一、上述代码有一个问题,就是client输入over后client结束,服务端有没有结束?

    结束,readline()方法是堵塞式方法,可是在client输入over后,client的socket关闭返回一个-1,服务端的readline()方法中的read方法读取-1,所以readline()方法就读取到null,所以会结束。



二、假设在client和服务端的PrintWriter pw = new PrintWriter(socket.getOutputStream())的自己主动刷新去掉,pw.print()的自己主动换行去掉,会发生什么?

   client没有收到转换后的数据,服务端没有显示数据

由于在client,pw.print()写入的数据,都写到了PrintWriter中,并没有刷新到socket输入流中

PS:这就是TCP在传输过程中出现两端都在等待的情况,非常可能是数据没有发出去,最大的可能就是有堵塞式方法。

当然,能够在pw.print();下加pw.flush(),可是问题依然,由于readline读取结束的标记是换行,所以在client的pw.print(+"\r\n"),所以要想 解决这个问题,就要在client和服务端都加上刷新动作,和换行符。

一旦遇到上述问题,一般都是由于堵塞式方法造成的服务端、client都在等待的情况,所以依照上述代码演示样例所写,比較好



练习2:上传文本文件

public class Main {

	public static void main(String[] args)throws IOException{
UpText_Client();
UpText_Server();
} public static void UpText_Server() throws IOException { ServerSocket ss = new ServerSocket(6534);
Socket socket = ss.accept();
BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream())); BufferedWriter bw = new BufferedWriter(new FileWriter("c:\\server.txt")); String line = null;
while((line = br.readLine())!=null){
//if("over".equals(line))break;//*
bw.write(line);
bw.newLine();//*
bw.flush();//*
} PrintWriter pw = new PrintWriter(socket.getOutputStream(),true);
pw.println("上传成功"); br.close();
bw.close();
socket.close();
ss.close();
} public static void UpText_Client() throws IOException { Socket socket = new Socket("127.0.0.1",6534);
BufferedReader br = new BufferedReader(new FileReader("c:\\data.txt"));
PrintWriter out = new PrintWriter(socket.getOutputStream(),true);
String line = null;
while((line = br.readLine())!=null){
out.println(line);
}
//out.println("over");//*,开发的时候一般都是应用时间戳,做结束标记,先发给服务器一下时间戳,输入结束后,再发一次
//socket里有方法
socket.shutdownOutput();//告诉服务端数据写完了 //读取socket流
BufferedReader brin = new BufferedReader(new InputStreamReader(socket.getInputStream()));
String string = brin.readLine();
System.out.println(string);
br.close();
socket.close();
}
}

*号处要注意,漏写easy造成,等待清理,client输入完成后,服务端还在等待,不知道client已经输入完成,堵塞,等待

演示的时候,分为两个主函数演示

练习3:上传图片

上传图片到client

public static void main(String[] args)throws IOException{
UpText_Client();
}
public static void UpText_Client() throws IOException { //创建client
Socket socket = new Socket("127.0.0.1",6534); //读取client要上传的图片文件
FileInputStream fis = new FileInputStream("c:\\1.jpg"); //获取socket输出流,将得到的图片数据发给服务端
OutputStream out = socket.getOutputStream();
byte[] buf = new byte[1024];
int len = 0;
while((len = fis.read(buf))!=-1){
out.write(buf, 0, len);
} //告诉服务端,client数据发送完成,使其读取结束
socket.shutdownOutput(); InputStream in = socket.getInputStream();
byte[] buf2 = new byte[1024];
int len2 = in.read(buf2);
String result = new String(buf2,0,len2);
System.out.println(result);
socket.close();
fis.close(); }

上传图片到服务端

public static void main(String[] args)throws IOException {
UpText_Server();
}
public static void UpText_Server() throws IOException {
//创建服务端
ServerSocket ss = new ServerSocket(6534); //获取client
Socket socket = ss.accept();
//读取client发来的数据
InputStream in = socket.getInputStream(); String ip = socket.getInetAddress().getHostAddress();
System.out.println("IP : "+ip+"....connect");
//将读取的数据存储到文件里
File dir = new File("c:\\CopyPic111111111");
if (!(dir.exists())) {
dir.mkdirs();
}
File file = new File(dir,ip+".jpg");
FileOutputStream fos = new FileOutputStream(file); byte[] buf = new byte[1024];
int len = 0;
while((len = in.read(buf))!=-1){
fos.write(buf, 0, len);
}
//获取socket输出流,显示上传结果
OutputStream out = socket.getOutputStream();
out.write("上传成功".getBytes());
fos.close();
socket.close();
ss.close();
}

上述代码的服务端仅仅能获取一个client上传,多个则不行。

服务端获取了1号client正在处理1号client,那么2号client就必需要等待,等待时间过长,就会连接超时,所以服务端结合线程,获取client对象为一个线程,处理client信息为一个线程,不停的切换,就能够实现多个client上传图片到服务端



服务端结合线程,改进

client部分不变

服务端

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
public class Up { private static ServerSocket ss;
public static void main(String[] args)throws IOException {
UpText_Server();
}
public static void UpText_Server() throws IOException { ss = new ServerSocket(6534); while(true){
Socket socket = ss.accept();//不停的接收client对象
new Thread(new UPtask(socket)).start();//创建多个线程运行不同client的信息
}
}
}

服务端线程

public class UPtask implements Runnable {
private Socket socket;
public UPtask(Socket socket){
this.socket = socket;
}
public void run() {
int count = 1;
try {
String ip = socket.getInetAddress().getHostAddress();
System.out.println("IP : "+ip+"....connect");
InputStream in = socket.getInputStream(); File dir = new File("c:\\CopyPic111111111");
if (!(dir.exists())) {
dir.mkdirs();
}
File file = new File(dir,ip+".jpg");
//假设已经存在
while(file.exists()) {
file = new File(dir,ip+"("+(count++)+").jpg");
}
FileOutputStream fos = new FileOutputStream(file); byte[] buf = new byte[1024];
int len = 0;
while((len = in.read(buf))!=-1){
fos.write(buf, 0, len);
}
//获取socket输出流,显示上传结果
OutputStream out = socket.getOutputStream();
out.write("上传成功".getBytes());
fos.close();
socket.close();
} catch (Exception e) {
// TODO: handle exception
throw new RuntimeException("server异常,请稍等");
}
}
}

UDP和TCP的差别:

UDP:将数据打包,有限制,不连接,效率高,不安全,easy丢包

TCP:建立数据通道,无限制,效率低,安全





JAVA学习第六十二课 — TCP协议练习的更多相关文章

  1. JAVA学习第五十二课 — IO流(六)File对象

    File类 用来给文件或者目录封装成对象 方便对文件与目录的属性信息进行操作 File对象能够作为參数传递给流的构造函数 一.构造函数和分隔符 public static void FileDemo( ...

  2. JAVA学习第六十四课 — 反射机制

       Java反射机制是在执行状态中,对于随意一个类,都可以知道这个类的全部属性和方法,对于随意一个对象,都可以调用它的随意一个方法和属性,这样的动态获取的信息以及动态调用对象的方法的功能称为java ...

  3. JAVA学习第三十二课(经常使用对象API)- 基本数据类型对象包装类

    将基本数据类型(8种:int..)封装成对象的优点就是能够在对象中封装很多其它的功能和方法来操控该数据 常见的操作就是:用于基本数据类型与字符串之间的转换 基本数据类型对象包装类一般用于基本类型和字符 ...

  4. JAVA学习第六十五课 — 正則表達式

    正則表達式:主要应用于操作字符串.通过一些特定的符号来体现 举例: QQ号的校验 6~9位.0不得开头.必须是数字 String类中有matches方法 matches(String regex) 告 ...

  5. java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

    java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...

  6. JAVA学习第五十九课 — 网络编程概述

    网络模型 OSI(Open System Interconnection)开放系统互连:參考模型 TCP/IP 网络通讯要素 IP地址 port号 传输协议 网络參考模型 七层OSI模型的基本概念要了 ...

  7. JAVA学习第四十八课 — IO流(二):文件的复制 & 缓冲区1

    一.复制文本文件 将G盘的文本文件拷贝到D盘上 也就是 读取G盘中文本文件的数据.写入D盘中->连读带写 而剪切呢.就是连读带写后,删除原盘的文件 <span style="fo ...

  8. Java学习笔记【十二、网络编程】

    原计划的学习结束时间是3月4日,目前看来已经延迟了,距离低标还差一些,多方面原因,也不找借口,利用周末赶赶进度,争取本周末把低标完成吧! 参考: http://www.runoob.com/java/ ...

  9. JAVA学习第三十四课 (经常使用对象API)—List集合及其子类特点

    整个集合框架中最经常使用的就是List(列表)和Set(集) 一.List集合 && Set的特点 Collection的子接口: 1.List:有序(存入和取出的顺序一致),元素都有 ...

随机推荐

  1. SplitContainer如何实现左侧导航,正确显示和导航内容

    关于这种类型的设计有很多的实现,这样,我首先解释一下我使用: 我的原则是实现方式,将form嵌panel在,作为一个子窗口. 如下面的代码细节: Form form = new DataSelect( ...

  2. 基本调试命令 - u/ub/uf

    原:http://www.cnblogs.com/developersupport/p/windbgcommand-u.html 在调试过程中难免会遇到须要反编译代码来分析逻辑的时候.在windbg中 ...

  3. Eclipse+Maven创建webapp项目<一> (转)

    Eclipse+Maven创建webapp项目<一> 1.开启eclipse,右键new——>other,如下图找到maven project 2.选择maven project,显 ...

  4. hdoj 2102 A计画 【BFS】

    称号:hdoj 2102 A计画点击打开链接 意甲冠军:文的就不说了.求救出公主所须要的最短时间,所以用广搜. 分析:读题之后不难做,比一般的题目多了一个条件就是能够传送,那么我们能够在广搜里面加一个 ...

  5. Lua学习 1) —— Android呼叫变量值和分配

    2014-07-08 Lua脚本语言,嵌入在App中扩展开发是非常不错的. 关于Android与Lua的环境搭配,我直接下载别人编好的.so与.jar(放到libs下就好了) 以下简介一下Androi ...

  6. 【C语言的日常实践(十四)】constkeyword详细解释

    const是C语言keyword,它定义一个变量不同意变更.使用const在一定程度上,可以提高节目的安全性和可靠性.其他.解const的作用,在看别人的代码时,对理解对方的程序有一定帮助. 1.co ...

  7. Appium0.18.x迁移到Appium1.x须知事项(灰常实用,解答了本人几个疑问)

    英文原版:https://github.com/appium/appium/blob/master/docs/en/advanced-concepts/migrating-to-1-0.md Migr ...

  8. ExtJS4 根据分配不同的树形菜单在不同的角色登录后

    继续我的最后.建立cookie后,带他们出去 var userName = Ext.util.Cookies.get('userName'); var userAuthority = Ext.util ...

  9. iOS 7 新特性

      iOS7更新了很多引人注目的功能.用户界面完全重新设计了.iOS7为开发2D,2.5D游戏引入了全新的动画系统.加强多线程,点对点连接,以及许多其他重要的功能让iOS7成为有史以来最有意义的一次发 ...

  10. 信息二战炸弹:中国到美国咨询公司Say no

    疯抢/文 在禁止Windows8操作系统參与政府採购,以及在中国销售的全部外国IT产品和服务都须通过新的安全审查之后,英国<金融时报>今天报料中国已禁止国企使用美国咨询公司服务. 这则消息 ...