首先启动服务端,客户端通过TCP的三次握手与服务端建立连接;

然后,客户端发送一段字符串,服务端收到字符串后,原封不动的发回给客户端。

ECHO 程序是网络编程通信交互的一个经典案例,称为回应程序,即客户端输入哪些内容,服务端会在这些内容前加上“ECHO”并将信息发回给客户端。

EchoServer.java

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.ServerSocket;
import java.net.Socket; public class EchoServer {
private ServerSocket server; public EchoServer() throws IOException {
// 创建服务端套接字对象
this.server = new ServerSocket(8888);
System.out.println("服务器启动" + "在" + 8888 + "端口监听连接请求");
} // 服务器主要的处理逻辑
public void service() {
// 是否关闭服务端连接
boolean flag = true;
while (flag) {
Socket socket = null;
try {
// 获取连接的客户端套接字对象
socket = server.accept();
// 获取socket相关的输入流和输出流
BufferedReader reader = getReader(socket);
BufferedWriter writer = getWriter(socket);
// 保存客户端发送的数据
String data = null;
// 读取客户端发送的数据
while ((data = reader.readLine()) != null) {
// 当获取的信息是“bye”时,关闭流
if ("bye".equals(data)) {
flag = false;
if (reader != null) {
reader.close();
}
if (writer != null) {
writer.close();
}
break;
} else {
System.out.println("来自客户端的数据:" + data);
// 回显给客户端的数据
writer.write("echo:" + data);
// 插入一个行分割符,readLine()方法用来判定字符串有没结束
writer.newLine();
// 刷新输出缓冲区
writer.flush();
}
}
} catch (IOException e) {
// 设置结束循环
flag = false;
} finally {
try {
if (server != null && (!server.isClosed())) {
server.close();
System.out.println("服务端关闭");
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
} // 获取socket相关的输入流对象
private BufferedReader getReader(Socket socket) throws IOException {
InputStream in = socket.getInputStream();
return new BufferedReader(new InputStreamReader(in));
} // 获取socket相关的输出流对象
private BufferedWriter getWriter(Socket socket) throws IOException {
OutputStream out = socket.getOutputStream();
return new BufferedWriter(new OutputStreamWriter(out));
} public static void main(String arg[]) throws IOException {
new EchoServer().service();
}
}

EchoClient.java

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.Socket; public class EchoClient {
private Socket socket; public EchoClient() {
try {
// 创建客户端套接字对象
this.socket = new Socket("127.0.0.1", 8888);
} catch (IOException e) {
e.printStackTrace();
}
} public void talkToServer() {
System.out.println("请输入要发送给服务端的数据:");
BufferedReader localReader = null;
String data = null;
// 获取客户端套接字相关的输入流和输出流
BufferedReader reader = getReader(socket);
BufferedWriter writer = getWriter(socket);
try {
// 创建读取用户输入的读取流
localReader = new BufferedReader(new InputStreamReader(System.in));
while ((data = localReader.readLine()) != null) {
// 如果客户端输入的数据是"bye",则关闭I/O流
if ("bye".equals(data)) {
System.out.println("客服端关闭!");
if (reader != null) {
reader.close();
}
if (writer != null) {
writer.close();
}
break;
} else {
// readLine方法必须读取到行分割符才返回读取。所以传递给输入流的字符串必须包含行分割符
System.out.println("客户端输出的数据--->\t" + data);
writer.write(data);
// 插入一个行分隔符,作为内容结束的标识
writer.newLine();
// 非常重要的是必须显式的将数据推送到服务器哪里去
writer.flush();
// 读取服务端返回的数据
System.out.println("服务器响应的数据--->\t " + reader.readLine());
}
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 关闭客户端套接字连接
try {
if (socket != null) {
socket.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
} // 获取客户端套接字相关的输出流
private BufferedWriter getWriter(Socket socket) {
try {
OutputStream out = null;
BufferedWriter writer = null;
out = socket.getOutputStream();
writer = new BufferedWriter(new OutputStreamWriter(out));
return writer;
} catch (Exception e) {
e.printStackTrace();
}
return null;
} // 获取客户端套接字相关的输入流
private BufferedReader getReader(Socket socket) {
try {
InputStream in = null;
BufferedReader reader = null;
in = socket.getInputStream();
reader = new BufferedReader(new InputStreamReader(in));
return reader;
} catch (Exception e) {
e.printStackTrace();
}
return null;
} public static void main(String[] args) {
new EchoClient().talkToServer();
}
}

编写Java程序,实现一个简单的echo程序(网络编程TCP实践练习)的更多相关文章

  1. UNP学习笔记2——从一个简单的ECHO程序分析TCP客户/服务器之间的通信

    1 概述 编写一个简单的ECHO(回复)程序来分析TCP客户和服务器之间的通信流程,要求如下: 客户从标准输入读入一行文本,并发送给服务器 服务器从网络输入读取这个文本,并回复给客户 客户从网络输入读 ...

  2. 编写一个简单的C++程序

    编写一个简单的C++程序 每个C++程序都包含一个或多个函数(function),其中一个必须命名为main.操作系统通过调用main来运行C++程序.下面是一个非常简单的main函数,它什么也不干, ...

  3. 用C语言编写一个简单的词法分析程序

    问题描述: 用C或C++语言编写一个简单的词法分析程序,扫描C语言小子集的源程序,根据给定的词法规则,识别单词,填写相应的表.如果产生词法错误,则显示错误信息.位置,并试图从错误中恢复.简单的恢复方法 ...

  4. 利用JSP编程技术实现一个简单的购物车程序

    实验二   JSP编程 一.实验目的1. 掌握JSP指令的使用方法:2. 掌握JSP动作的使用方法:3. 掌握JSP内置对象的使用方法:4. 掌握JavaBean的编程技术及使用方法:5. 掌握JSP ...

  5. JAVA课程设计——一个简单的教务人事管理系统

    大三上学期期末总结,没错,上学期,写在下学期新学期开始,哈哈哈. 上学期学习了面向对象程序设计,课程设计的题目使用JAVA语言完成一个简单的教务人事管理系统,能够实现访问数据库的登录验证,分别按部门和 ...

  6. netty入门实现简单的echo程序

    最近看以往在程序中编写的代码,发现有一个功能是使用socket通讯来实现的,而那个时候使用的是基于bio的阻塞io来实现的,最近在看netty,发现可以使用netty来使用nio的方式来实现,此博客记 ...

  7. 如何创建一个简单的struts2程序

    如何创建一个简单的Struts2程序 “计应134(实验班) 凌豪” 1.创建一个新的Web项目test(File->new->Web Project) 2.Struts2框架的核心配置文 ...

  8. Linux系统学习笔记之 1 一个简单的shell程序

    不看笔记,长时间不用自己都忘了,还是得经常看看笔记啊. 一个简单的shell程序 shell结构 1.#!指定执行脚本的shell 2.#注释行 3.命令和控制结构 创建shell程序的步骤 第一步: ...

  9. 第一讲 一个简单的Qt程序分析

    本文概要:通过一个简单的Qt程序来介绍Qt程序编写的基本框架与一些Qt程序中常见的概念 #include <QApplication> #include <QPushButton&g ...

随机推荐

  1. NERD_commenter快捷键

    快捷键有点多,记不过来,做个备份 1. \cc 注释当前行和选中行 2. \cn 没有发现和\cc有区别 3. \c<空格> 如果被选区域有部分被注释,则对被选区域执行取消注释操作,其它情 ...

  2. web端 - 返回上一步,点击返回,跳转上个页面 JS

    1.方法一: <script language="javascript" type="text/javascript"> window.locati ...

  3. binlog浅析

    binlog浅析 一.基础知识 什么是binlog? (图一) 全称:Binary Log (二进制日志),包含描述数据库更改的" 事件 ",例如表创建操作或对表数据的更改.二进制 ...

  4. 【C/C++】散列/算法笔记4.2

    先说一下我自己的理解. 我先给你N组数据,这个N组里可能有重复的! 然后我们先统计好了N组里面的独立的每个对应的出现了几次(相当于map,然后每项属性有出现了多少次的),用的是数组下标对应 现在我们给 ...

  5. 为什么企业全面云化需要IT战略支撑和驱动?

    引子:为什么传统企业全面云化一直磨磨唧唧举步维艰? 笔者将企业上云大体上分为几个阶段: 第一个阶段是基础设施虚拟化.即将应用从物理机搬到(lift and shift migration)虚拟机上.基 ...

  6. [源码解析] PyTorch 分布式(15) --- 使用分布式 RPC 框架实现参数服务器

    [源码解析] PyTorch 分布式(15) --- 使用分布式 RPC 框架实现参数服务器 目录 [源码解析] PyTorch 分布式(15) --- 使用分布式 RPC 框架实现参数服务器 0x0 ...

  7. hitcontraining_magicheap

    拿到题目例行检查 程序是64位的程序 保护几乎全开,试运行一下程序 十分明显的堆溢出的界面,将程序放入ida中,shift+f12发现了后门程序 进入main主函数进行查看 可以看到当,v3==486 ...

  8. YC-Framework版本更新:V1.0.3

    分布式微服务框架:YC-Framework版本更新V1.0.3!!! 本次版本V1.0.3更新 集成分布式事务Seata: 集成分布式事务Tx-LCN: 集成Kafka: 集成RocketMQ: 集成 ...

  9. SpringCloud Alibaba实战(12:引入Dubbo实现RPC调用)

    源码地址:https://gitee.com/fighter3/eshop-project.git 持续更新中-- 大家好,我是老三,断更了半年,我又滚回来继续写这个系列了,还有人看吗-- 在前面的章 ...

  10. DevOps实战(Docker+Jenkins+Git)

    基于Docker+Jenkins+Git的CI/CD实战 与上一篇随笔:基于 Jenkins+Docker+Git 的CI流程初探 有所不同,该内容更偏向于实际业务的基础需求. 有几点需要注意: 该实 ...