tcpip c语言程序设计,TCP 服务端和客户端程序设计(C)
流程图
1 预备知识
1.1 socket函数
为了执行网络输入输出,一个进程必须做的第一件事就是调用socket函数获得一个文件描述符
#include
int socket(int family,int type,int protocol);
返回:非负描述字---成功 -1---失败
第一个参数指明了协议簇,目前支持5种协议簇,最常用的有AF_INET(IPv4协议)和AF_INET6(IPv6协议);
第二个参数指明套接口类型,有三种类型可选:SOCK_STREAM(字节流套接口)、SOCK_DGRAM(数据报套接口)和SOCK_RAW(原始套接口);
如果套接口类型不是原始套接口,那么第三个参数就为0
1.2 connect函数
当用socket建立了套接口后,可以调用connect为这个套接字指明远程端的地址;如果是字节流套接口,connect就使用三次握手建立一个连接;如果是数据报套接口,connect仅指明远程端地址,而不向它发送任何数据
#include
int connect(int sockfd, const struct sockaddr * addr, socklen_t addrlen);
返回:0---成功 -1---失败
第一个参数是socket函数返回的套接口描述字;第二和第三个参数分别是一个指向套接口地址结构的指针和该结构的大小。
这些地址结构的名字均已“sockaddr_”开头,并以对应每个协议族的唯一后缀结束。以IPv4套接口地址结构为例,它以“sockaddr_in”命名,定义在头文件;以下是结构体的内容:
struct in_addr {
in_addr_t s_addr; /* IPv4地址 */
};
struct sockaddr_in {
uint8_t sin_len; /* 无符号的8位整数 */
sa_family_t sin_family;
/* 套接口地址结构的地址簇,这里为AF_INET */
in_port_t sin_port; /* TCP或UDP端口 */
struct in_addr sin_addr;
char sin_zero[8];
};
1.3 bind函数
为套接口分配一个本地IP和协议端口,对于网际协议,协议地址是32位IPv4地址或128位IPv6地址与16位的TCP或UDP端口号的组合;如指定端口为0,调用bind时内核将选择一个临时端口,如果指定一个通配IP地址,则要等到建立连接后内核才选择一个本地IP地址。
#include
int bind(int sockfd, const struct sockaddr * server, socklen_t addrlen);
返回:0---成功 -1---失败
第一个参数是socket函数返回的套接口描述字;
第二和第三个参数分别是一个指向特定于协议的地址结构的指针和该地址结构的长度。
1.4 listen函数
listen函数仅被TCP服务器调用,它的作用是将用sock创建的主动套接口转换成被动套接口,并等待来自客户端的连接请求。
#include
int listen(int sockfd,int backlog);
返回:0---成功 -1---失败
第一个参数是socket函数返回的套接口描述字;
第二个参数规定了内核为此套接口排队的最大连接个数。
由于listen函数第二个参数的原因,内核要维护两个队列:以完成连接队列和未完成连接队列。未完成队列中存放的是TCP连接的三路握手未完成的连接,accept函数是从已连接队列中取连接返回给进程;当以连接队列为空时,进程将进入睡眠状态。
1.5 accept函数
accept函数由TCP服务器调用,从已完成连接队列头返回一个已完成连接,如果完成连接队列为空,则进程进入睡眠状态。
#include
int accept(int listenfd, struct sockaddr *client, socklen_t * addrlen);
回:非负描述字---成功 -1---失败
第一个参数是socket函数返回的套接口描述字;
第二个和第三个参数分别是一个指向连接方的套接口地址结构和该地址结构的长度;该函数返回的是一个全新的套接口描述字;如果对客户段的信息不感兴趣,可以将第二和第三个参数置为空。
1.6 write和read函数
当服务器和客户端的连接建立起来后,就可以进行数据传输了,服务器和客户端用各自的套接字描述符进行读/写操作。因为套接字描述符也是一种文件描述符,所以可以用文件读/写函数write()和read()进行接收和发送操作。
(1)write()函数用于数据的发送。
#include
int write(int sockfd, char *buf, int len);
回:非负---成功 -1---失败
参数sockfd是套接字描述符,对于服务器是accept()函数返回的已连接套接字描述符,对于客户端是调用socket()函数返回的套接字描述符;
参数buf是指向一个用于发送信息的数据缓冲区;
len指明传送数据缓冲区的大小。
(2)read()函数用于数据的接收。
#include
int read(int sockfd, char *buf, intlen);
回:非负---成功 -1---失败
参数sockfd是套接字描述符,对于服务器是accept()函数返回的已连接套接字描述符,对于客户端是调用socket()函数返回的套接字描述符;
参数buf是指向一个用于接收信息的数据缓冲区;
len指明接收数据缓冲区的大小。
1.7 send和recv函数
TCP套接字提供了send()和recv()函数,用来发送和接收操作。这两个函数与write()和read()函数很相似,只是多了一个附加的参数。
(1)send()函数用于数据的发送。
#include
#include < sys/socket.h >
ssize_t send(int sockfd, const void *buf, size_t len, int flags);
回:返回写出的字节数---成功 -1---失败
前3个参数与write()相同,参数flags是传输控制标志。
(2)recv()函数用于数据的发送。
#include
#include < sys/socket.h >
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
回:返回读入的字节数---成功 -1---失败
前3个参数与read()相同,参数flags是传输控制标志。
2 服务器端
2.1 源码
#include
#include
#include
#include
#include
#include
#include
#include
#define PORT 1234
#define BACKLOG 1
int main()
{
int listenfd, connectfd;
struct sockaddr_in server;
struct sockaddr_in client;
socklen_t addrlen;
if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Creating socket failed.");
exit(1);
}
int opt =SO_REUSEADDR;
setsockopt(listenfd,SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
bzero(&server,sizeof(server));
server.sin_family=AF_INET;
server.sin_port=htons(PORT);
server.sin_addr.s_addr= htonl (INADDR_ANY);
if(bind(listenfd, (struct sockaddr *)&server, sizeof(server)) == -1) {
perror("Binderror.");
exit(1);
}
if(listen(listenfd,BACKLOG)== -1){ /* calls listen() */
perror("listen()error\n");
exit(1);
}
addrlen =sizeof(client);
if((connectfd = accept(listenfd,(struct sockaddr*)&client,&addrlen))==-1) {
perror("accept()error\n");
exit(1);
}
printf("Yougot a connection from cient's ip is %s, prot is %d\n",inet_ntoa(client.sin_addr),htons(client.sin_port));
send(connectfd,"Welcometo my server.\n",22,0);
close(connectfd);
close(listenfd);
return 0;
}
2.2 编译运行
mkdir tcp_learn
cd tcp_learn/
vim tcpserver.c
粘贴入源码,然后:
gcc -o tcpserver tcpserver.c //编译
./tcpserver //运行
暂时还没现象,客户端还没连接进来。
3 客户端
3.1 源码
#include
#include
#include
#include
#include
#include
#include
#include
#define PORT 1234
#define MAXDATASIZE 100
int main(int argc, char *argv[])
{
int sockfd, num;
char buf[MAXDATASIZE];
struct hostent *he;
struct sockaddr_in server;
if (argc!=2) {
printf("Usage:%s \n",argv[0]);
exit(1);
}
if((he=gethostbyname(argv[1]))==NULL){
printf("gethostbyname()error\n");
exit(1);
}
if((sockfd=socket(AF_INET, SOCK_STREAM, 0))==-1){
printf("socket()error\n");
exit(1);
}
bzero(&server,sizeof(server));
server.sin_family= AF_INET;
server.sin_port = htons(PORT);
server.sin_addr =*((struct in_addr *)he->h_addr);
if(connect(sockfd,(struct sockaddr *)&server,sizeof(server))==-1){
printf("connect()error\n");
exit(1);
}
if((num=recv(sockfd,buf,MAXDATASIZE,0)) == -1){
printf("recv() error\n");
exit(1);
}
buf[num-1]='\0';
printf("Server Message: %s\n",buf);
close(sockfd);
return 0;
}
3.2 编译运行
新打开一个命令窗口:
cd tcp_learn/
vim tcpclient.c
粘贴源码,然后运行:
gcc -o tcpclient tcpclient.c
./tcpclient 127.0.0.1
4 结果
TCP
tcpip c语言程序设计,TCP 服务端和客户端程序设计(C)相关推荐
- Go语言实现TCP服务端和客户端
Go语言实现TCP服务端和客户端 Go语言实现TCP通信 TCP协议 TCP服务端 TCP客户端 本文转载自Go语言实现TCP通信 Go语言实现TCP通信 TCP协议 TCP/IP(Transmiss ...
- Qt:Qt实现Winsock网络编程—TCP服务端和客户端通信(多线程)
Qt实现Winsock网络编程-TCP服务端和客户端通信(多线程) 前言 感觉Winsock网络编程的api其实和Linux下网络编程的api非常像,其实和其他编程语言的网络编程都差不太多.博主用Qt ...
- 2-3 建立简易TCP服务端、客户端【socket server/client】【socket、bind、listen、accept、send、closesocket】【conect、recv】
2-3 建立简易TCP服务端.客户端 文章目录 2-3 建立简易TCP服务端.客户端 0-前言 1-服务端简易功能 2-客户端简易功能 3-代码逻辑 4-服务端 4-1 建立socket 4-2 绑定 ...
- windows Socket编程之TCP服务端与客户端
在前面的文章中有一篇讲到了命名管道通信,它是创建一根管道来进行进程之间或网络之间通信的.但是它有些缺陷,比如说效率较低等.而从这篇文章开始将介绍socket编程.socket是通过TCP,UDP,IP ...
- C语言实现TCP服务端
一 前言 上次实现了客户端的TCP通信代码,这次来实现TCP服务端. 二 代码实现 前期文章 C语言TCP客户端获取时间 Unix网络编程卷一第三章笔记 2.代码 #include <stdio ...
- java BIO tcp服务端向客户端消息群发代码教程实战
前言 项目需要和第三方厂商的服务需要用TCP协议通讯,考虑到彼此双方可能都会有断网重连.宕机重启的情况,需要保证 发生上述情况后,服务之间能够自动实现重新通信.研究测试之后整理如下代码实现.因为发现客 ...
- .NET TCP服务端和客户端
客户端 public class TcpClientHelper{/// <summary>/// 是否连接上/// </summary>public bool Connect ...
- JAVA NIO 异步TCP服务端向客户端消息群发代码教程实战
前言 项目需要和第三方厂商的服务需要用TCP协议通讯,考虑到彼此双方可能都会有断网重连.宕机重启的情况,需要保证 发生上述情况后,服务之间能够自动实现重新通信.研究测试之后整理如下代码实现.因为发现客 ...
- python学习随堂笔记—TCP服务端与客户端
思想: 1. 客户端 向服务端 发起连接 2. 服务端 接到请求,双方建立连接 3. 客户端 向 服务端发消息 4. 服务端 应答客户端 5. 服务端 与客户端循环读写操作 6. 操作完成之后客户端发 ...
最新文章
- python 虚拟环境 virtualenv virtualenvwrapper的使用方法、命令
- 贝叶斯优化神经网络参数_贝叶斯超参数优化:神经网络,TensorFlow,相预测示例
- 使用bcftools提取指定样本的vcf文件(extract specified samples in vcf format)
- avue中怎样隐藏新增和编辑的按钮
- qemu搭建arm运行linux内核,centos使用qemu搭建ARM64运行环境
- oc基础-OC中对象方法的使用
- 使用bcp进行大数据量导出导入
- 你知道Java里有多少种锁吗?15种锁最全总结!
- spring boot 中文文档_已献出膝盖!GitHub上的宝藏级SpringBoot核心文档,讲得太清晰了...
- 架构师之路:星环大数据架构师的培训心得
- 通信模块整理(一)JDY-31
- 执着女股民:20年炒一只股票 翻几十倍很轻松
- 硬盘的免费分区工具:简单、易用
- Lamber表达式 List,Map,Set 互相转换
- 私网地址与公网地址是如何转换的?
- 评价的等级优良差_小学生期末评语-等级优良合格(最佳版本)
- Peak prominences(峰日珥)
- netcore 使用NHibernate
- 软件外包开发流程及管理
- Linux系统中与网络配置相关的工具