最近开始学习在Linux上创建服务器,这是我的一些学习笔记。
TCP/IP协议层 (标准协议层)
应用层:可以由程序员来制定, 完成数据加密等 代表协议: http、tftp、nfs等...
传输层:实现端对端的数据通信. 代表协议: TCP(传输控制协议)、UDP(用户数据报协议)等
网络层:实现数据的路由. 代表协议:IP,ICMP等.
网络接口层(物理层、数据链路层): 物理层实现底层网络驱动,链路层主要以数据帧方式传输 代
表协议:以太网.
网络套接字编程
套接字
是网络通信过程中最基本的接口,用一个特殊的文件描述符(非负整数)来表示
分类:
流式套接字:数据以字节流方式进行传递,数据大小没有上限. 通常适用于TCP协议.
数据报套接字:数据以报文的方式进行传递,每一个报文大小有限制. 通常适用于UDP协议.
原始套接字:一般用于本地数据传递.
IP地址
作用:标识主机
形式:
点分十进制:例如:"192.168.2.5" 用户能够认识的形式
二进制形式:例如:11000000 10101000 00000010 00000101 计算机使用的形式
端口Port
作用:标识进程
取值:
是一个unsigned short类型 范围:0 ~ 65535
0 ~ 1024 被内核使用,用户可以分配 1025 ~ 65535
TCP服务器搭建流程
1.创建套接字 -------> socket()
AF_INET:使用ipv4协议表
SOCK_STREAM:流式套接字 ----> 与第三个0配合,表示使用TCP协议。
SOCK_DGRAM:数据包套接字 ----> 与第三个0配合,表示使用UDP协议。
1.套接字绑定 -------> bind() 核心:IP 、Port
inet_addr():地址转换函数
字节序
{
大端序:ARM架构、网络字节序。(低地址存放高字节)
小端序:X86架构。(低地址存放低字节)
字节序转换函数:htons、ntohs
htons:host to net short 主机转为网络字节序。小--->大
}
1.监听客户端的链接请求 -->listen()
int listen(int sockfd,int backlog);
{
backlog:同时能够监听最多客户端个数。。(使用队列来存放监听消息)
不表示同时能够连接的客户端个数!
}
1.接受客户端连接请求--->accept()
int accept(int sockfd,struct sockaddr *addr,socklen_t *addrlen)
{
addr:获取并且存放客户端的地址信息 如果不关心客户端信息则传递 NULL
addrlen:客户端地址信息长度 如果不关心客户端信息则传递 NULL
返回值: 正确 返回同于和客户端通信的套接字。 该套接字被称为 通信套接字(连接套接字)
错误 -1
sockfd:被称为 监听套接字
后续都是使用 通信套接字 来完成数据交互。
}
1.数据接收、发送--->read()/write()
2.关闭套接字---->close()
TCP客户端搭建流程
1. 创建套接字 -----> socket( )
设置服务地址信息. struct sockaddr_in
2. 主动请求连接 ----> connect( )
3. 数据接收/发送 ---> read()/write()
4. 关闭套接字 ---> close( )
TCP客户端程序展示
#include <strings.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#define BACKLOG 5
#define PERROR(msg) do{perror(msg);exit(-1);}while(0)
int main(int argc, char *argv[])
{
int sockfd = socket(AF_INET, SOCK_STREAM, 0);
if(sockfd == -1)
PERROR("socket");
struct sockaddr_in ser;
ser.sin_family = AF_INET;
ser.sin_port = htons(9292); // uint16_t
ser.sin_addr.s_addr = inet_addr("192.168.18.141"); //inet_addr() ip_string-->ip_bin
if(-1 == bind(sockfd, (struct sockaddr*)&ser, sizeof(ser)))
PERROR("bind");
if(-1 == listen(sockfd, BACKLOG))
PERROR("listen");
printf("listen---------\n");
int comfd = accept(sockfd, NULL, NULL);
if(comfd == -1)
PERROR("accept");
printf("accept---success\n");
char buf[32] = {0};
/*data read/write*/
while(1)
{
read(comfd, buf, sizeof(buf));
printf("recv:%s\n", buf);
write(comfd, "world", 6);
}
close(comfd);
close(sockfd);
printf("server---over----\n");
return 0;
}
结果展示