TCP IP网络编程(四) 基于TCP的服务器端、客户端

这篇具有很好参考价值的文章主要介绍了TCP IP网络编程(四) 基于TCP的服务器端、客户端。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

理解TCP、UDP

TCP/IP协议栈

TCP/IP协议栈

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

TCP/IP协议栈共分为4层,可以理解为数据收发分成了4个层次化过程。

TCP协议栈

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

UDP协议栈

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

链路层

链路层是物理连接领域标准化的结果,也是最基本的领域,专门定义LAN、WAN、MAN等网络标准。两台主机通过网络进行数据交换,这需要像下图所示的物理连接,链路层就负责这些标准。

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

IP层

IP协议是面向消息的,不可靠的协议。每次传输数据时会帮我们选择路径,但并不一致。如果传输中发生路径错误,则选择其他路径。但如果发生数据丢失或错误,则无法解决。IP协议无法应对数据错误。

TCP/UDP层

IP层解决数据传输中路径选择问,只需要按照此路径传输数据即可。TCP和UDP层以IP层提供的路径信息为基础完成实际的数据传输,故该层又称为传输层。TCP可以保证可靠的数据传输,但它发送数据时以IP层为基础。

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

TCP和UDP存在于IP层之上,决定主机之间的数据传输方式,TCP协议确认后向不可靠的IP协议赋予可靠性。

应用层

以上类容是套接字通信过程中自动处理的。选择数据传输路径、数据确认过程都被隐藏到套接字内部。但掌握了这些理论,才能编写出符合需求的网络程序。

向大家提供的工具就是套接字,大家只需要利用套接字编写出程序即可。编写软件的过程中,需要根据程序特点决定服务器端和客户端之间的数据传输规则,这便是应用层协议。网络编程的大部分内容就是设计并实现应用层协议。

实现基于TCP的服务器端、客户端

TCP服务器端的默认函数调用顺序

1、socket()    		创建套接字
2、bind()			分配套接字地址
3、listen()			等待连接请求状态
4、accept()			允许连接
5、read()/write()	数据交换
6、close()			断开连接

进入等待连接请求状态

我们已经调用bind函数给套接字分配了地址,接下来通过调用listen函数进入等待连接请求状态,只有调用了listen函数,客户端才能进入可发出连接请求的状态,这时客户端才可以调用connect函数。

#include<sys/socket.h>

int listen(int sock, int backlog);
	成功返回0,失败返回-1
    sock	为希望进入等待连接请求状态的文件描述符,传递的描述符套接字参数为服务器端套接字
    backlog 连接请求等待队列的长度,若为5,则队列长度为5,表示最多使5个连接请求进入队列

等待连接请求状态是指客户端请求连接时,受理连接前一直使连接处于等待状态,客户端连接请求本身也是网络中收到的一种数据,而想要接受就需要套接字。

受理客户端连接请求

调用listen函数后,有新的连接请求,则应按序受理。受理请求意味着进入可接受数据的状态,此时就需要套接字来接受数据,但服务器端的套接字在做门卫,不能再充当接受数据的角色。因此需要另外一个套接字,该套接字不需要亲自创建,accept函数将会创建套接字并连接到发起请求的客户端。

#include <sys/socket.h>

int accept(int sock, struct sockaddr* addr, socklen_t* addrlen);
	成功返回创建的套接字文件描述符,失败返回-1
    sock  	服务器套接字的文件描述符
	addr  	保存发起连接请求的客户端地址信息的变量的地址,调用函数后会向该变量填充客户端地址信息
	addrlen	第二个参数addr结构体的长度,调用函数后会向该变量填充客户端地址长度

accept函数受理连接请求等待队列中待处理中的客户端连接请求。函数调用成功时,accept函数内部将产生用于数据I/O的套接字,并返回文件描述符,套接字使自动创建的,并且自动与发起连接请求的客户端建立连接。

TCP客户端的默认函数调用顺序

TCP客户端函数的调用顺序

1、socket()			创建套接字
2、connect()			请求连接
3、read()/write()	交换数据
4、closr()			断开连接

与服务器端相比,区别就在于请求连接,它使创建客户端套接字后向服务器端发起的连接请求,服务器端调用listen函数后创建请求等待队列,之后客户端即可请求连接。

#include<sys/socket.h>

int connect(int sock, struct sockaddr* servaddr, socklen_t addrlen);
	成功返回0,失败返回-1
     sock  		客户端套接字文件描述符
	 servaddr	保存目标服务器地址信息的变量地址值
	 addrlen	以字节为单位,传递第二个参数的地址变量的长度

客户端调用connect函数后,发生以下情况才会返回

  • 服务器端接受连接请求
  • 发生断网等异常情况而中断连接请求

接受连接请求并不是服务器端调用accept函数,其实是服务器端把连接请求信息记录到等待队列中,因此connect函数返回后并不立即进行数据交换

基于TCP的服务器端、客户端函数调用关系

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

总体流程如下:

​ 服务器端创建套接字后连续调用bind、listen函数进入等待状态,客户端通过调用connect函数发起连接请求,客户端侄女等到服务器端调用listen函数之后才能调用connect发起连接请求,也要主义客户端调用connect函数前,服务器端可能率先调用accept函数,此时服务器端调用accept函数进入阻塞状态,知道客户端调用connect函数为止。

实现迭代服务器端、客户端

实现迭代服务器端

tcp客户端,TCP/IP网络编程,网络,tcp/ip,网络协议,linux,服务器

实现迭代服务器端最简单的办法就是插入循环语句反复调用accept函数。循环最后的close(client)关闭的调用accept函数创建的套接字,意味着结束了针对某一客户端的服务,此时如果还想服务于其他客户端,就要重新调用accept函数。目前同一时刻只能服务于一个客户端,学完进程和线程后,就可以编写同时服务于多个客户端的服务器端。

迭代回声服务器端、客户端

回声服务器端以及配套的回声客户端的程度的基本运行方式:

  • 服务器端在同一时刻只与一个客户端相连,并提供回声服务。
  • 服务器端依次向5个客户端提供服务并退出。
  • 客户端接受用户的输入字符串并发送到服务器端。
  • 服务器端将收到的字符串数据传回客户端,即“回声”。
  • 两端之间的字符串回声一直执行到客户端输入Q为止。

首先介绍满足以上要求的回声服务器端

echo_server.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 1024
void error_handling(char *message);

int main(int argc, char *argv[])
{
	int serv_sock, clnt_sock;
	char message[BUF_SIZE];
	int str_len, i;
	
	struct sockaddr_in serv_adr;
	struct sockaddr_in clnt_adr;
	socklen_t clnt_adr_sz;
	
	if(argc!=2) {
		printf("Usage : %s <port>\n", argv[0]);
		exit(1);
	}
	
	serv_sock=socket(PF_INET, SOCK_STREAM, 0);   
	if(serv_sock==-1)
		error_handling("socket() error");
	
	memset(&serv_adr, 0, sizeof(serv_adr));
	serv_adr.sin_family=AF_INET;
	serv_adr.sin_addr.s_addr=htonl(INADDR_ANY);
	serv_adr.sin_port=htons(atoi(argv[1]));

	if(bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr))==-1)
		error_handling("bind() error");
	
	if(listen(serv_sock, 5)==-1)
		error_handling("listen() error");
	
	clnt_adr_sz=sizeof(clnt_adr);

	for(i=0; i<5; i++)
	{
		clnt_sock=accept(serv_sock, (struct sockaddr*)&clnt_adr, &clnt_adr_sz);
		if(clnt_sock==-1)
			error_handling("accept() error");
		else
			printf("Connected client %d \n", i+1);
	
		while((str_len=read(clnt_sock, message, BUF_SIZE))!=0)
			write(clnt_sock, message, str_len);

		close(clnt_sock);
	}

	close(serv_sock);
	return 0;
}

void error_handling(char *message)
{
	fputs(message, stderr);
	fputc('\n', stderr);
	exit(1);
}

运行结果

gcc echo_server.c -o eserver
./eserver 9190
输出:
Connecten client 1
Connecten client 2
Connecten client 3

回声客户端代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 1024
void error_handling(char *message);

int main(int argc, char *argv[])
{
	int sock;
	char message[BUF_SIZE];
	int str_len;
	struct sockaddr_in serv_adr;

	if(argc!=3) {
		printf("Usage : %s <IP> <port>\n", argv[0]);
		exit(1);
	}
	
	sock=socket(PF_INET, SOCK_STREAM, 0);   
	if(sock==-1)
		error_handling("socket() error");
	
	memset(&serv_adr, 0, sizeof(serv_adr));
	serv_adr.sin_family=AF_INET;
	serv_adr.sin_addr.s_addr=inet_addr(argv[1]);
	serv_adr.sin_port=htons(atoi(argv[2]));
	
	if(connect(sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr))==-1)
		error_handling("connect() error!");
	else
		puts("Connected...........");
	
	while(1) 
	{
		fputs("Input message(Q to quit): ", stdout); 	//如果输入Q说明结束while循环
		fgets(message, BUF_SIZE, stdin);
		
		if(!strcmp(message,"q\n") || !strcmp(message,"Q\n"))  //检验message是否为Q/q
			break;

		write(sock, message, strlen(message));
		str_len=read(sock, message, BUF_SIZE-1);
		message[str_len]=0;
		printf("Message from server: %s", message);
	}
	
	close(sock);
	return 0;
}

void error_handling(char *message)
{
	fputs(message, stderr);
	fputc('\n', stderr);
	exit(1);
}

运行结果

gcc echo_client.c -o eclient
./eclient 192.168.233.20 9190
输出:
Connected ....
Input message: hello
Message from server: hello
Input message : Q

这是《TCP/IP网络编程》专栏的第四篇文章,欢迎各位读者订阅!

更多资料点击 GitHub 欢迎各位读者去Star

⭐学术交流群Q 754410389 持续更新中~~~文章来源地址https://www.toymoban.com/news/detail-794298.html

到了这里,关于TCP IP网络编程(四) 基于TCP的服务器端、客户端的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Socket网络编程(TCP/IP)实现服务器/客户端通信。

    一.前言 回顾之前进程间通信(无名管道,有名管道,消息队列,共享内存,信号,信号量),都是在同一主机由内核来完成的通信。 那不同主机间该怎么通信呢? 可以使用Socket编程来实现。 Socket编程可以通过网络来实现实现不同主机之间的通讯。 二.Socket编程的网络模型如

    2024年02月08日
    浏览(89)
  • Linux网络编程之TCP/IP实现高并发网络服务器设计指南

    目录 引言: 多进程服务器 例程分享: 多线程服务器  例程分享: I/O多路复用服务器 select 例程分享: poll 例程分享: epoll 例程分享: 总结建议         随着互联网的迅猛发展,服务器面临着越来越多的并发请求。如何设计一个能够高效处理大量并发请求的服务器成为

    2024年02月20日
    浏览(53)
  • 【网络编程】——基于TCP协议实现回显服务器及客户端

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【网络编程】【Java系列】 本专栏旨在分享学习网络编程的一点学习心得,欢迎大家在评论区交流讨论💌 TCP提供的API主要有两个类 Socket ( 既会给服务器使用也会给客

    2024年02月03日
    浏览(65)
  • 【Linux网络编程】网络编程套接字(TCP服务器)

    作者:爱写代码的刚子 时间:2024.4.4 前言:本篇博客主要介绍TCP及其服务器编码 只介绍基于IPv4的socket网络编程,sockaddr_in中的成员struct in_addr sin_addr表示32位 的IP地址 但是我们通常用点分十进制的字符串表示IP地址,以下函数可以在字符串表示和in_addr表示之间转换 字符串转in

    2024年04月14日
    浏览(79)
  • 网络编程(一)TCP单进程服务器编程详解

    想要学习socket网络编程的读者一定要首先学好计算机网络的理论知识,包括 1)osi网络七层模型与ip四层模型 2)套接字含义 3)局域网通信过程 4)广域网通信过程 5)tcp,udp通信协议,在这两个协议中的连接建立,数据封装,传输过程,传输中可能遇到的问题的处理(差错控

    2024年02月15日
    浏览(47)
  • 【网络编程】demo版TCP网络服务器实现

    UDP和TCP的区别: 对于TCP协议有几个特点: 1️⃣ 传输层协议 2️⃣ 有连接(正式通信前要先建立连接) 3️⃣ 可靠传输(在内部帮我们做可靠传输工作) 4️⃣ 面向字节流 对于UDP协议有几个特点: 1️⃣ 传输层协议 2️⃣ 无连接 3️⃣ 不可靠传输 4️⃣ 面向数据报 可以看到

    2024年02月06日
    浏览(55)
  • 【网络编程】TCP流套接字编程(TCP实现回显服务器)

    Socket(既能给客户端使用,也能给服务器使用) 构造方法 基本方法: ServerSocket(只能给服务器使用) 构造方法: 基本方法: 客户端代码示例: 服务器代码示例: 运行结果: 代码执行流程: 服务器启动,阻塞在accept,等待客户端建立连接. 客户端启动.这里的new操作会触发和服务器之间建立连

    2024年04月25日
    浏览(66)
  • 网络编程(8.14)TCP并发服务器模型

    作业: 1. 多线程中的newfd,能否修改成全局,不行,为什么? 2. 多线程中分支线程的newfd能否不另存,直接用指针间接访问主线程中的newfd,不行,为什么? 多线程并发服务器模型原代码: 1.将newfd改成全局变量效果:  答:不行,因为newfd是全局变量的话,客户端连接后生成

    2024年02月13日
    浏览(49)
  • 【网络编程】实现UDP/TCP客户端、服务器

    需要云服务器等云产品来学习Linux的同学可以移步/--腾讯云--/--阿里云--/--华为云--/官网,轻量型云服务器低至112元/年,新用户首次下单享超低折扣。   目录 一、UDP 1、Linux客户端、服务器 1.1udpServer.hpp 1.2udpServer.cc 1.3udpClient.hpp 1.4udpClient.cc 1.5onlineUser.hpp 2、Windows客户端 二、T

    2024年02月06日
    浏览(60)
  • 【网络编程】实现一个简单多线程版本TCP服务器(附源码)

    accept 函数是在服务器端用于接受客户端连接请求的函数,它在监听套接字上等待客户端的连接,并在有新的连接请求到来时创建一个新的套接字用于与该客户端通信。 下面是 accept 函数的详细介绍以及各个参数的意义: sockfd: 是服务器监听套接字的文件描述符,通常是使用

    2024年02月13日
    浏览(53)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包