.NET网络编程——TCP通信

一、网络编程的基本概念 :

1. 网络

        就是将不同区域的电脑连接到一起,组成局域网、城域网或广域网。把分部在不同地理区域的计算机于专门的外部设备用通信线路 互联成一个规模大、功能强的网络系统,从而使众多的计算机可以方便地互相传递信息,共享硬件、软件、数据信息等资源。

2. 计算机网络 

        通过传输介质、通信设施和网络通信协议,将地理位置相同的具有独立功能的多台计算机及其外部设备连起来,实现资源共享 和数据传输的系统。

3. 通信协议

计算机网路中实现通信必须有一些通信协议的规定,对传输代码、代码结构、传输控制步骤出错控制等制定标准。

4. 通信接口

为了使两个节点之间能进行对话,必须在他们之间建立通信工具(即接口),使彼此之间能进行信息交换。

5. 网络分层

       1. 由于结点之间联系很复杂,在指定协议时,把复杂成份分解成一些简单的成份,再将他们复合起来。最常用的复合方式是层次 方式,即同层间可以通信、上一层可以调用下一层,而与再下一层不发生关系

        2. TCP/IP是一个协议族,也是按照层次划分,共四层:应用层,传输层,互连网络层,接口层(物理+数据链路层)

        3. OSI网络通信协议模型,是一个参考模型,而TCP/IP协议是事实上的标准。TCP/IP协议参考了OSI模型,但是并没有严格按照OSI规 定的七层标准去划分,而只划分了四层,这样会更简单点,当划分太多层时,你很难区分某个协议是属于哪个层次的

二、网络编程三要素 

1. IP地址

        要想让网络中的计算机能够互相通信,必须为计算机指定一个标识号,通过这个标识号来指定要接受数据的计算机和识别发送 的计算机,而IP地址就是这个标识号,也就是设备的标识。

1.1 IP地址分为两大类

  1. IPv4:是给每个连接在网络上的主机分配一个32bit地址。按照TCP/IP规定,IP地址用二进制来表示,每个IP地址的 长32bit,也就是4个字节。
  2. IPv6:由于互联网的蓬勃发展,IP地址的需求量愈来愈大,但是网络地址资源有限,使得IP的分配越发紧张。为了扩 大地址空间,通过IPv6重新定义地址空间,采用128bit地址长度,每16个字节一组,分成8组十六进制数,这就解决 了网络地址资源数量不够的问题。

1.2 InetAddress类 (在Java中使用InetAddress类代表IP)

        常用的方法 :

String getHostAddress ()返回IP地址字符串(以文本表现形式)
String getHostName ()返回此IP地址的主机名
Byte[] getAddress ()返回此InetAddress对象的原始IP地址

2. 端口

2.1 概念:

        网络的通信,本质上是两个应用程序的通信,每台计算机都有很多的应用程序。IP地址是唯一的标识网络的设备,端口 号就是唯一标识设备中的应用程序,也就是应用程序的标识。

2.2 端口号:

        用两个字节表示的整数,它的取值范围是0~65535。其中,0~1023之间的端口号用于一些知名的网络服务和应用,普通 的应用程序需要使用1024以上的端口号。如果端口号被另外一个服务或应用所占用,会导致当前程序启动失败。

2.3 IntetSocketAddress类

        包含IP和端口信息,常用于Socket通信。此类实现套接字数字地址(IP地址+端口号),不依赖任何协议。

        常用的方法 :

InetAddress getAddress()获得InetAddress
Int getPort ()获取端口号
String getHostName()获取主机名

三、协议

        通过计算机网络可以使多台计算机实现连接,位于同一网络中的计算机进行连接和通信时需要遵守一定的规则,这就好比在道 路中形势的汽车一定要遵守交通规则一样。在计算机网路中,这些连接和通信的规则被称为网络通信协议。它对数据的传输格式、传 输速率、传输步骤做了统一规定,通信双方必须同时遵守才能完成数据交换。

3.1 UDP协议

  1. 用户数据报协议(User Datagram Protocol)
  2. UDP是预无线通信协议,即在数据传输时,数据的发送端和接受端不建立逻辑连接。简单来说,当一台计算机向另外一台计算机 发送数据时,发送端不会确认接受端是否存在,就会发送出数据,同样接受端在收到数据时,夜不会向发送端反馈是否收到数 据
  3. 由于使用UDP协议消耗资源少,通信效率高,所以通常都会用于音频、视频和普通数据的传输
  4. 例如视频会议通常采用UDP协议,因为这种情况即使偶尔丢失一两个数据包,也不会对接受结果产生太大影响。但是在使用UDP 协议传送数据时,由于UDP的面向无连接性,不能保证数据的完整性,因此在传输重要数据时不建议使用UDP协议。

3.2 TCP协议

  • 传输控制协议(Transmission Control Protocol)
  • TCP协议是面向连接的通信协议,即传输数据之前,在发送端和接受端建立逻辑连接,然后再传输数据,它提供了两台计算机之 间可靠无差错的数据连接。在TCP连接中必须明确客户端于服务端,由于客户端向服务端发出连接请求,每次连接的创建都需要 经过“三次握手”;

3.3 三次握手

TCP协议中,在发送数据的准备阶段,客户端与服务器之间的三次交互,以保证连接的可靠性

第一次握手 : 客户端向服务器发送连接请求,等待服务器确认

第二次握手 : 服务器向客户端回送一个响应,通知客户端收到了连接请求

第三次握手 : 客户端再次向服务器发送确认信息,确认连接

        完成三次握手连接建立后,客户端和服务器就可以开始进行数据传输了。由于这种面向连接的特性,TCP协议可以保证传输数据 的安全,所以应用非常广泛。例如上传文件,下载文件、浏览网页。

3.3.1 TCP通信原理 :

        TCP通信协议是一种可靠的网络协议,它在通信的两端各建立一个Socket对象,从而在通信的两端形成网络虚拟链路,一旦建 立了虚拟的网络链路,两端的程序就可以通过虚拟链路进行通信。

  • 使用基于TCP协议的Socket网络编程实现,使用Socket对象来代表两端的通信端口
  • TCP协议基于请求-响应模式,第一次主动发起的程序被客户端(Client)程序
  • 第一次通讯中等待连接的程序被服务器(Serser)程序
  • 利用IO流实现数据的传输

四、TCP实现步骤(聊天案例)

1. 步骤

1、在服务端指定一个端口号来创建ServerSocket,并使用accept方法进行侦听,这将阻塞服务器线程,等待用户请求。

2、在客户端指定服务的主机IP和端口号来创建socket,并连接服务端ServerSocket,此时服务端accept方法被唤醒,同时返回一个 和客户端通信的socket。

3、在客户端和服务端分别使用socket来获取网络通信输入/输出流,并按照一定的通信协议对socket进行读/写操作。

4、通信完成后,在客户端和服务端中分别关闭socket。

2. 服务器端

  1. 创建ServerSocket(int port)对象        
  2. 在Socket上使用accept方法监听客户端的连接请求(阻塞等待连接功能)
  3. 接受并处理请求信息 • 将处理结果返回给客户端
  4. 关闭流和Socket对象

3. 客户端

  1. 创建Socket(Strng host , int port)对象
  2. 向服务器发送连接请求 • 向服务端发送服务请求 • 接受服务结果(服务响应)
  3. 关闭流和Socket对象

4. ServerSocket类

常用构造器 : ServerSocket(int port) 创建绑定到指定端口的服务器套接字

常用方法 :Socket accept() 侦听要连接到此套接字并接受它

Java为客户端提供了Socket类,为服务器端提供了ServerSocket类

构造方法:

Socket(InetAddress address , int port)address(IP地址), port(端口号)

创建流套接字并将其连接到指定IP地址的指定端口号;

常用方法 :

OutoutStream getOutputStream()返回套接字的输出流
InputStream getInputStream ()返回套接字的输入流
Void shutdownOutput()禁用套接字的输出流 

5. 利用实现客户端于服务器无线聊天功能

1. 服务器

package com.net;import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;/*** 服务器* @author 云村小威** @2023年7月19日 下午10:02:11*/
public class Server {public static void main(String[] args) throws Exception {System.out.println("⁎⁎⁎⁎⁎⁎服务端⁎⁎⁎⁎⁎⁎");/* 创建服务器连接对象 */ServerSocket ss = new ServerSocket(6666);System.out.println("服务器已开启,等待连接...");/* 调用accept方法,客服端没有启动连接不能执行下一步(阻塞功能) */Socket server = ss.accept();System.out.println("客户端连接成功");/* 获取输入流,读取客户端发送的数据 */InputStream inputStream = server.getInputStream();BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));/* 获取输出流,向客户端回写数据 */Scanner zw = new Scanner(System.in);OutputStream outputStream = server.getOutputStream();BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(outputStream));while (true) {String readLine = br.readLine();System.out.println("客户端 :" + readLine);if ("拜拜".equals(readLine)) {System.out.println("再见");break;}System.out.println("请输入发送到客户端的内容:");String test = zw.next();/* 向客户端写入内容 */bw.write(test);bw.newLine();bw.flush();if ("拜拜".equals(test)) {System.out.println("再见");break;}}/* 关闭资源 */bw.close();outputStream.close();br.close();inputStream.close();server.close();ss.close();}
}

2. 客户端

package com.net;import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.InetAddress;
import java.net.Socket;
import java.util.Scanner;/*** 客户端* @author 云村小威** @2023年7月19日 下午10:02:00*/
public class Client {public static void main(String[] args) throws Exception {System.out.println("⁕⁕⁕⁕⁕客户端⁕⁕⁕⁕⁕");/* 创建Socket对象,请求连接 */Socket client = new Socket(InetAddress.getByName("127.0.0.1"), 6666);System.out.println("服务器连接成功...");/* 获取输出流对象,向服务器写入数据 */OutputStream outputStream = client.getOutputStream();BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(outputStream));/* 获取输入流对象,读取服务器数据 */InputStream inputStream = client.getInputStream();BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));/* 向服务器发送信息 */Scanner zw = new Scanner(System.in);while (true) {System.out.println("请输入发送到服务器的内容:");String test = zw.next();/* 向服务器写入内容 */bw.write(test);bw.newLine();bw.flush();if ("拜拜".equals(test)) {System.out.println("再见");break;}/* 每次读取一个字节数组 */String content = br.readLine();System.out.println("服务器 : " + content);if ("拜拜".equals(content)) {System.out.println("再见");break;}}/* 关闭资源 */br.close();inputStream.close();bw.close();outputStream.close();client.close();}
}

3. 控制台版 效果

    感谢您的观看,我会在本专栏持续更新,后续更新多线程等知识。接下来就可以利用多线程实现窗体实时无线聊天功能,请敬请期待吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/1678.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

坐标系变换的坑

坐标系变换的坑 坐标系变换本来是很简单的事情,公式也很简单。但是卡了我很多天,原因是:两个坐标系的位姿,虽然都是右手系,但我的在顺时针旋转是yaw角是递增的,同事发给我的却是逆时针递减的。 理论上很简…

Jenkins+Robot 接口自动化测试

目录 前言: 设计目标 项目说明 目录结构 配置 jenkins 1.安装插件 2.配置项目 前言: JenkinsRobot是一种常见的接口自动化测试方案,可以实现自动化的接口测试和持续集成。Jenkins是一个流行的持续集成工具,而Robot Framew…

55 # 实现可写流

先在 LinkedList.js 给链表添加一个移除方法 class Node {constructor(element, next) {this.element element;this.next next;} }class LinkedList {constructor() {this.head null; // 链表的头this.size 0; // 链表长度}// 可以直接在尾部添加内容,或者根据…

聊聊ChatGPT是如何组织对话的

为什么要组织对话? 总所周知,ChatGPT的训练大致可分为下图中展示的几个阶段,其中,在Pretraining阶段,模型的训练数据是纯文本,目标是根据上文预测下一个token,而在后面的几个阶段中&#xff0c…

网络安全能力成熟度模型介绍

一、概述 经过多年网络安全工作,一直缺乏网络安全的整体视角,网络安全的全貌到底是什么,一直挺迷惑的。目前网络安全的分类和厂家非常多,而且每年还会冒出来不少新的产品。但这些产品感觉还是像盲人摸象,只看到网络安…

回归预测 | MATLAB实现WOA-CNN鲸鱼算法优化卷积神经网络的数据多输入单输出回归预测

回归预测 | MATLAB实现WOA-CNN鲸鱼算法优化卷积神经网络的数据多输入单输出回归预测 目录 回归预测 | MATLAB实现WOA-CNN鲸鱼算法优化卷积神经网络的数据多输入单输出回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 回归预测 | MATLAB实现WOA-CNN鲸鱼算法优化卷积…

数据采集专家----4通道AD采集子卡推荐

FMC136是一款4通道250MHz采样率16位AD采集FMC子卡,符合VITA57规范,可以作为一个理想的IO模块耦合至FPGA前端,4通道AD通过高带宽的FMC连接器(HPC)连接至FPGA从而大大降低了系统信号延迟。 该板卡支持板上可编程采样时钟…

Linux进程

Linux进程 对于进程的理解,我们要从计算机的重要的冯诺依曼体系结构讲起,只有知道我们的程序/文件是如何在计算机中被操作运行并输出到显示器中,通过对于操作系统的理解,才能对于进程进行一定的理解。 文章目录 Linux进程冯诺依…

c#示例-json序列化和json树

序列化 由于指针和引用类型的存在,在运行中的程序中,数据不一定是整块的。 可能东一块西一块散落在内存的各个地方。 序列,是指连续且有序的一个整体。序列化就是把数据变为连续有序整体的过程。 经过这样处理后的数据就可以方便的进行传输…

数据结构--时间复杂度与空间复杂度

数据结构–时间复杂度与空间复杂度 文章目录 数据结构--时间复杂度与空间复杂度时间复杂度一、什么是时间复杂度二、具体实例1.大O的渐进表示法2.二分查找的时间复杂度 空间复杂度一、什么是空间复杂度二、具体实例总结 时间复杂度 一、什么是时间复杂度 在计算机科学中&…

云原生微服务应用的平台工程实践

作者:纳海 01 微服务应用云原生化 微服务是一个广泛使用的应用架构,而如何使得微服务应用云原生化却是近些年一直在演进的课题。国内外云厂商对云原生概念的诠释大同小异,基本都会遵循 CNCF 基金会的定义: 云原生技术有利于各组…

【后端面经-Java】JVM垃圾回收机制

【后端面经-Java】JVM垃圾回收机制 1. Where:回收哪里的东西?——JVM内存分配2. Which:内存对象中谁会被回收?——GC分代思想2.1 年轻代/老年代/永久代2.2 内存细分 3. When:什么时候回收垃圾?——GC触发条…

【MySQL】根据MVCC和Read View分析事务的四种隔离级别在读写场景分别是如何体现其隔离性的

目录 一、数据库并发的三种场景 二、读写场景的MVCC 1、3个(4个)记录隐藏列字段 2、undo log(撤销日志) 3、模拟MVCC场景 3.1update场景 3.2delete场景 3.3insert 3.4select场景 4、Read View 5、RR和RC的区别 5.1当…

Windows安装激活注意事项

选择语言、版本(Windows 10指的是专业版本)和体系结构(32位/64位),这里自行根据情况选择(如果机器预装的是Windows 10家庭中文版则选择家庭中文版,如果预装的是专业版则选择Windows 10。这样原先…

Revit 导出明细表的两种方法!

方法一、Revit中怎么灵活运用明细表格式的导出与导入 在做项目的时候,遇到一些项目需要进行工程量统计的时候,经常需要设置明细表里面的格式,例如字体、表格排布样式等,但是项目一旦多起来,这些工作重复性又太高&#…

适合小公司的自动化部署脚本

背景(偷懒) 在小小的公司里面,挖呀挖呀挖。快挖不动了,一件事重复个5次,还在人肉手工,身体和心理就开始不舒服了,并且违背了个人的座右铭:“偷懒”是人类进步的第一推动力。 每次想…

解决MAC IDEA终端每次都要source ~/.zshrc

安装nvm之后,发现每隔一段时间(不清楚是新打开一个终端还是会定时刷新)就要重新执行source ~/zshrc,才能执行nvm命令。找了一圈发现idea默认使用的shell是bash,将默认的shell改成zsh就可以,更改位置&#x…

【运维】shell监控脚本结合钉钉机器人实现服务及服务器监控告警

文章目录 前言一、监控shell脚本和钉钉机器人二、创建钉钉机器人:1.在钉钉群聊里点击设置2.在设置里点击机器人选项3.再点击添加机器人4.再点击选择自定义机器人5.设置机器人名称、是否加密、是否限制ip、以及触发关键字6.获取机器人的Webhook地址 三、编写监控脚本…

[爬虫]解决机票网站文本混淆问题-实战讲解

前言 最近有遇到很多小伙伴私信向我求助,遇到的问题基本上都是关于文本混淆或者是字体反爬的问题。今天给大家带来其中一个小伙伴的实际案例给大家讲讲解决方法 📝个人主页→数据挖掘博主ZTLJQ的主页 ​​ 个人推荐python学习系列: ☄️爬虫J…

架构训练营3:架构设计流程和架构师职责

架构师相关职责: 架构师是业务和技术之间的桥梁,架构师不能只顾技术,不懂业务,架构师很容易两头不讨好 三个核心能力: 判断:1业务理解力2.技术能力3.沟通能力 拆解:1技术深度2.技术宽度3.技术…