Socket编程接口API并实现简单的TCP网络编程

#include <sys/types.h>
#include <sys/socket.h>
socket()创建套接字,成功返回套接字的文件描述符,失败返回-1
domain: 设置套接字的协议簇, AF_UNIX AF_INET AF_INET6
type: 设置套接字的服务类型 SOCK_STREAM SOCK_DGRAM
protocol: 一般设置为 0,表示使用默认协议
int socket(int domain, int type, int protocol);


bind()将 sockfd 与一个 socket 地址绑定,成功返回 0,失败返回-1
sockfd 是网络套接字描述符
addr 是地址结构
addrlen 是 socket 地址的长度
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);


listen()创建一个监听队列以存储待处理的客户连接,成功返回 0,失败返回-1
sockfd 是被监听的 socket 套接字
backlog 表示处于完全连接状态的 socket 的上限
int listen(int sockfd, int backlog);


accept()从 listen 监听队列中接收一个连接,成功返回一个新的连接 socket,
该 socket 唯一地标识了被接收的这个连接,失败返回-1
sockfd 是执行过 listen 系统调用的监听 socket
addr 参数用来获取被接受连接的远端 socket 地址
addrlen 指定该 socket 地址的长度
int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);


connect()客户端需要通过此系统调用来主动与服务器建立连接,成功返回 0,失败返回-1
sockfd 参数是由 socket()返回的一个 socket。
serv_addr 是服务器监听的 socket 地址
addrlen 则指定这个地址的长度
int connect(int sockfd, const struct sockaddr *serv_addr, socklen_t addrlen);


close()关闭一个连接,实际上就是关闭该连接对应的 socket
int close(int sockfd);


TCP 数据读写:
recv()读取 sockfd 上的数据,buff 和 len 参数分别指定读缓冲区的位置和大小
send()往 socket 上写入数据,buff 和 len 参数分别指定写缓冲区的位置和数据长

flags 参数为数据收发提供了额外的控制
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);


UDP 数据读写:
recvfrom()读取 sockfd 上的数据,buff 和 len 参数分别指定读缓冲区的位置和大

src_addr 记录发送端的 socket 地址
addrlen 指定该地址的长度
sendto()往 socket 上写入数据,buff 和 len 参数分别指定写缓冲区的位置和数据长

dest_addr 指定接收数据端的 socket 地址
addrlen 指定该地址的长度
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);

TCP:面向连接的,可靠的字节流服务。

可靠性:应答确认,超时重传,去重,乱序重排,滑动窗口(进行流量控制)

服务端

 

#include<stdlib.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<arpa/inet.h>
int main()
{
    int sockfd = socket(AF_INET,SOCK_STREAM,0);//创建套接字
    if(socfd == -1)
    {
        exit(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");//127.0.0.1相当于自己的主机
    int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));//指定套接字的地址(ip port)
    if(res == -1)
    {
        perror("bind err\n");
    	exit(1);
    }
    listen(sockfd,5);//5的意义:在linux系统上,设置已完成三次握手的监听队列大小  
    while(1)
    {
        struct sockaddr_in  caddr;//accept会初始化caddr,所以传参是以指针方式
        int len = sizeof(caddr);
        int c = accept(sockfd,(struct sockaddr*)&caddr,&len);//客户端的描述符
        if(c<0)
        {
            continue;
        }
        printf("accept c=%d,ip=%s,port=%d\n",c,inet_ntoa(caddr.sin_addr),ntohs(caddr.sin_port));//打印客户端的IP地址和端口
        while(1)
        {
            char buff[128] = {0};
        	int n = recv(c,buff,127,0);//接收数据
            if(n<=0)//客户端关闭的唯一标志返回值等于0
            {
                break;
            }
        	printf("buff=%s\n",buff);
        	send(c,"ok",2,0);//发送数据
        }
        close(c);
    }
    close(sockfd);
    exit(0);
}

客户端

 

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

int main()
{
    int sockfd = socket(AF_INET,SOCK_STREAM,0);
    if(sockfd == -1)
    {
        exit(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));
    if(res == -1)
    {
        perror("connect err\n");
        exit(1);
    }
    while(1)
    {
        printf("input:\n");
        char buff[128]={0};
    	fgets(buff,128,stdin);
        if(strncmp(buff,"end",3)==0)
        {
            break;
        }
    	send(sockfd,buff,strlen(buff),0);
        memset(buff,0,128);
    	int n=recv(sockfd,buff,127,0);//ok
        if(n<=0)
        {
            break;
        }
    	printf("buff=%s\n",buff);
    }
    close(sockfd);
    exit(0);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值