1.主机字节序列和网络字节序列
主机字节序列分为大端字节序和小端字节序,不同的主机采用的字节序列可能不同。大端字节序是指一个整数的高位字节存储在内存的低地址处,低位字节存储在内存的高地址处。小端字节序则是指整数的高位字节存储在内存的高地址处,而低位字节则存储在内存的低地址处。在两台使用不同字节序的主机之间传递数据时,可能会出现冲突。所以,在将数据发送到网络时规定整形数据使用大端字节序,所以也把大端字节序成为网络字节序列。对方接收到数据后,可以根据自己的字节序进行转换
Linux 系统提供如下 4 个函数来完成主机字节序和网络字节序之间的转换:
#include <netinet/in.h>
uint32_t htonl(uint32_t hostlong); // 长整型的主机字节序转网络字节序
uint32_t ntohl(uint32_t netlong); // 长整型的网络字节序转主机字节序
uint16_t htons(uint16_t hostshort); // 短整形的主机字节序转网络字节序
uint16_t ntohs(uint16_t netshort); // 短整型的网络字节序转主机字节序
2.双方要通信得有一个套接字,套接字中有ip地址(哪一台主机)和port(主机上的哪个进程)信息,两个进程就有两对ip地址和两对port,一组ip+port称之为套接字的地址
ipv4和ipv6的地址是不一样的
3.套接字地址结构
①通用 socket 地址结构 socket 网络编程接口中表示 socket 地址的是结构体 sockaddr,其定义如下:
#include <bits/socket.h>
struct sockaddr
{
sa_family_t sa_family;
char sa_data[14];
};
sa_family 成员是地址族类型(sa_family_t)的变量。地址族类型通常与协议族类型对应。常见的协议族和对应的地址族如下图所示: ②专用 socket 地址结构
TCP/IP 协议族有 sockaddr_in 和 sockaddr_in6 两个专用 socket 地址结构体,它们分别用于 IPV4 和 IPV6:
/*sin_family: 地址族 AF_INET
sin_port: 端口号,需要用网络字节序表示
sin_addr: IPV4 地址结构:s_addr 以网络字节序表示 IPV4 地址*/
struct in_addr
{
u_int32_t s_addr;
};
struct sockaddr_in
{
sa_family_t sin_family;
u_int16_t sin_port;
struct in_addr sin_addr;
};
struct in6_addr
{
unsigned char sa_addr[16]; // IPV6 地址,要用网络字节序表示
};
struct sockaddr_in6
{
sa_family_t sin6_family; // 地址族:AF_INET6
u_inet16_t sin6_port; // 端口号:用网络字节序表示
u_int32_t sin6_flowinfo; // 流信息,应设置为 0
struct in6_addr sin6_addr; // IPV6 地址结构体
u_int32_t sin6_scope_id; // scope ID,尚处于试验阶段
};
③IP 地址转换函数
通常,人们习惯用点分十进制字符串表示 IPV4 地址,但编程中我们需要先把它们转化为整数方能使用,下面函数可用于点分十进制字符串表示的 IPV4 地址和网络字节序整数表示的 IPV4 地址之间的转换:
#include <arpa/inet.h>
in_addr_t inet_addr( const char *cp); //字符串表示的 IPV4 地址转化为网络字节序
char* inet_ntoa( struct in_addr in); // IPV4 地址的网络字节序转化为字符串表示
4.网络编程接口
#include <sys/types.h>
#include <sys/socket.h>
int socket( int domain, int type, int protocol);
int bind( int sockfd, const struct sockaddr *addr, socklen_t addrlen);
int listen( int sockfd, int backlog);
int accept( int sockfd, struct sockaddr *addr, socklen_t *addrlen);
int connect( int sockfd, const struct sockaddr *serv_addr, socklen_t addrlen);
int close( int sockfd);
ssize_t recv( int sockfd, void *buff, size_t len, int flags);
ssize_t send( int sockfd, const void *buff, size_t len, int flags);
ssize_t recvfrom( int sockfd, void *buff, size_t len, int flags,
struct sockaddr* src_addr, socklen_t *addrlen);
ssize_t sendto( int sockfd, void *buff, size_t len, int flags,
struct sockaddr* dest_addr, socklen_t addrlen);
5.TCP 编程流程
TCP 提供的是面向连接的、可靠的、字节流服务。TCP 的服务器端和客户端编程流程如下: 服务器端: socket();//创建套接字 bind();//指定ip,端口号 listen();//创建监听队列 accept();//接受连接 recv();//接受客户端发来的数据 send();//向客户端发送的数据 close();//断开连接 客户端: socket();//创建套接字 connect();//向服务器端发起连接,有一个三次握手的问题 send();//向服务器端发送的数据 recv();//接受服务器端发来的数据 close();//断开连接,有一个四次挥手的问题
6.Linux可以用ifconfig来查看ipv4,ipv6地址,Windows可以用ifconfig来查看ipv4,ipv6地址
7.TCP服务器端代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main()
{
int sockfd = socket(AF_INET,SOCK_STREAM,0);
assert(sockfd != -1);
struct sockaddr_in saddr,caddr;
memset(&saddr,0,sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
assert(res != -1);
res = listen(sockfd,5);
assert(res != -1);
while(1)
{
int len = sizeof(caddr);
int c = accept(sockfd,(struct sockaddr*)&caddr,&len);
if(c<0)
{
continue;
}
printf("c=%d\n",c);
char buff[128] = {0};
int num = recv(c,buff,127,0);
printf("buff=%s\n",buff);
send(c,"ok",2,0);
close(c);
}
}
8.TCP客户端代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main()
{
int sockfd = socket(AF_INET,SOCK_STREAM,0);
assert(sockfd != -1);
struct sockaddr_in saddr;
memset(&saddr,0,sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
int res = connect(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
assert(res != -1);
printf("input\n");
char buff[128] = {0};
fgets(buff,128,stdin);
send(sockfd,buff,strlen(buff)-1,0);
memset(buff,0,128);
recv(sockfd,buff,127,0);
printf("buff = %s\n",buff);
close(sockfd);
}
|