JAVA学习第六十二课 — TCP协议练习
通过练习掌握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协议练习的更多相关文章
- JAVA学习第五十二课 — IO流(六)File对象
File类 用来给文件或者目录封装成对象 方便对文件与目录的属性信息进行操作 File对象能够作为參数传递给流的构造函数 一.构造函数和分隔符 public static void FileDemo( ...
- JAVA学习第六十四课 — 反射机制
Java反射机制是在执行状态中,对于随意一个类,都可以知道这个类的全部属性和方法,对于随意一个对象,都可以调用它的随意一个方法和属性,这样的动态获取的信息以及动态调用对象的方法的功能称为java ...
- JAVA学习第三十二课(经常使用对象API)- 基本数据类型对象包装类
将基本数据类型(8种:int..)封装成对象的优点就是能够在对象中封装很多其它的功能和方法来操控该数据 常见的操作就是:用于基本数据类型与字符串之间的转换 基本数据类型对象包装类一般用于基本类型和字符 ...
- JAVA学习第六十五课 — 正則表達式
正則表達式:主要应用于操作字符串.通过一些特定的符号来体现 举例: QQ号的校验 6~9位.0不得开头.必须是数字 String类中有matches方法 matches(String regex) 告 ...
- java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)
java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...
- JAVA学习第五十九课 — 网络编程概述
网络模型 OSI(Open System Interconnection)开放系统互连:參考模型 TCP/IP 网络通讯要素 IP地址 port号 传输协议 网络參考模型 七层OSI模型的基本概念要了 ...
- JAVA学习第四十八课 — IO流(二):文件的复制 & 缓冲区1
一.复制文本文件 将G盘的文本文件拷贝到D盘上 也就是 读取G盘中文本文件的数据.写入D盘中->连读带写 而剪切呢.就是连读带写后,删除原盘的文件 <span style="fo ...
- Java学习笔记【十二、网络编程】
原计划的学习结束时间是3月4日,目前看来已经延迟了,距离低标还差一些,多方面原因,也不找借口,利用周末赶赶进度,争取本周末把低标完成吧! 参考: http://www.runoob.com/java/ ...
- JAVA学习第三十四课 (经常使用对象API)—List集合及其子类特点
整个集合框架中最经常使用的就是List(列表)和Set(集) 一.List集合 && Set的特点 Collection的子接口: 1.List:有序(存入和取出的顺序一致),元素都有 ...
随机推荐
- 工作经常使用的SQL整理,实战篇(三)
原文:工作经常使用的SQL整理,实战篇(三) 工作经常使用的SQL整理,实战篇,地址一览: 工作经常使用的SQL整理,实战篇(一) 工作经常使用的SQL整理,实战篇(二) 工作经常使用的SQL整理,实 ...
- Django操作model时刻,一个错误:AttributeError:’ProgrammingError’ object has no attribute ‘__traceback__’
原因:在Django项目下对应的应用以下的models.py配置的model(也就是class)没有创建成对应的表. 这是怎么回事呢? 首先,将models.py里面的model创建成相应的数据库表的 ...
- asp.net访问网络路径方法(模拟用户登录)
public class IdentityScope : IDisposable { // obtains user token [DllImport("advapi32.dll" ...
- Android运用自己的标题栏
Android程序的标题栏TitleBar区域很单调,如果想个性化一些可以通过下面的方法来为自己软件的标题定制一个layout布局文件,比如浏览器的标题栏,它包含了网站的Favicon,自定义的进度条 ...
- Unity3D方法来隐藏和显示对象
Unity3D作 在使用unity3d开发游戏的过程中.我们经常会遇到须要隐藏或者显示的操作,针对这一点,以下做了一些总结. 一.设置Renderer状态 在游戏的开发中,全部可以被渲染的物体都包括有 ...
- 大约SQL现场“这包括”与“包括在”字符串的写法
1.字段查找表值"这包括"方法一字符串的所有记录 如果表中有一name场,查询name这包括"乔 - 史密斯"所有记录.能够写sql: Stirng st ...
- Android-管理Activity生命周期 -开始一个Activity
很多程序都是从main()方法开始启动的,和其他程序不同,android是在activity生命周期的特定状态的特定回调方法中初始化代码的.activity启动和销毁的时候都用很多回调方法. 这里将要 ...
- C#多线程编程实例 螺纹与窗口交互
C#多线程编程实例 螺纹与窗口交互 代码: public partial class Form1 : Form { //声明线程数组 Thread[] workThreads = new Thread ...
- 【Java收集的源代码分析】Hashtable源代码分析
Hashtable简单介绍 Hashtable相同是基于哈希表实现的,相同每一个元素是一个key-value对,其内部也是通过单链表解决冲突问题,容量不足(超过了阀值)时.相同会自己主动增长. Has ...
- centos 彻底删除nodejs默认的安装文件
1> yum remove nodejs npm -y 2> cd /usr/local/lib 移除所有 node 和 node_modules目录 cd /usr/local/in ...