TCP套接字编程

TCP套接字编程

socket函数

为了执行网络I/O,一个进程必须做的第一件事情就是调用socket函数,指定期望的通信协议类型。

#include <sys/socket.h>

int socket(int family, int type, int protocol);

返回:若成功则为非负描述符,若出错则为-1

参数:

  • family参数指明协议族
family说明
AF_INETIPv4协议
AF_INET6IPv6协议
AF_LOCALUnix域协议
AF_ROUTE路由套接字
AF_KEY密钥套接字

- type参数指明套接字类型

type说明
SOCK_STREAM字节流套接字
SOCK_DGRAM数据报套接字
SOCK_SEQPACKET有序分组套接字
SOCK_RAW原始套接字
  • protocol参数为下图所示的某个协议类型常值,或者设为0,以选择所给定family和type组合的系统默认值。
protocol说明
IPPROTO_TCPTCP传输协议
IPPROTO_UDPUDP传输协议
IPPROTO_SCTPSCTP传输协议

socket函数在成功时返回一个小的非负整数值,称为套接字描述符。

connect函数

TCP客户用connect函数来建立与TCP服务器的连接。

#include <sys/socket.h>
int connect(int sockfd, const struct sockaddr *servaddr, socklen_t addrlen);

返回:若成功则为0,若出错则为-1

参数:

  • sockfd是由socket函数返回的套接字描述符。
  • servaddr参数是指向套接字地址结构的指针。
  • addrlen为sockaddr结构的大小。
bind函数

bind函数把一个本地协议地址赋予一个套接字。

#include <sys/socket.h>
int bind(int sockfd, const struct sockaddr *myaddr, socklen_t addrlen);

返回:若成功则为0,若出错则为-1
  • sockfd是由socket函数返回的套接字描述符。
  • myaddr参数是指向特定于协议的地址结构的指针。
  • addrlen是该地址结构的长度。
listen函数

listen函数仅由TCP服务器调用,它做两件事情:

  • 当socket函数创建一个套接字时,它被假设为一个主动套接字,即它是一个将调用connect发起连接的客户套接字。listen函数把一个未连接的套接字转换成一个被动套接字,指示内核应接受指向该套接字的连接请求。
  • 本函数的第二个参数规定了内核应该为相应套接字排队的最大连接个数。
#include <sys/socket.h>
int listen(int sockfd, int backlog);

返回:若成功则为0,若出错则为-1
  • sockfd是由socket函数返回的套接字描述符。
  • backlog参数规定了内核应该为相应套接字排队的最大连接个数。
accept函数

accept函数由TCP服务器调用,用于从已完成连接队列队头返回下一个已完成连接。如果已完成连接队列唯恐,那么进程将被投入睡眠(假定套接字为默认的阻塞方式)。

#include <sys/socket.h>
int accept(int sockfd, struct sockaddr *cliaddr, socklen_t *addrlen);

返回:若成功则为非负描述符,若出错则为-1
  • sockfd是由socket函数返回的套接字描述符。
  • 参数cliaddr和addrlen用来返回已连接的对端进程(客户)的协议地址。
  • addrlen是值-结果参数:调用前,我们将由*addrlen所引用的整数值置为由cliaddr所指向的套接字地址结构的长度,返回时,该整数值即为由内核存放在该套接字地址结构内的确切字节数。
TCP客户/服务器程序套接字流程

TCP客户/服务器程序套接字流程

服务器/客户端程序实例

tcpserv.c

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

int main()
{
    int listenfd, connfd;
    struct sockaddr_in servaddr, cliaddr;
    char buf[1024];
    int n;

    listenfd = socket(AF_INET, SOCK_STREAM, 0);
    if (listenfd < 0)
        perror("socket error");

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(9877);

    bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    listen(listenfd, 5);

    socklen_t len = sizeof(cliaddr);
    for ( ; ; )
    {
        connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &len);

        while ((n = read(connfd, buf, 1024)) > 0)
        {
            fputs(buf, stdout);
            write(connfd, buf, n);
        }
        close(connfd);
    }

    close(listenfd);
    exit(0);
}

tcpcli.c

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

int main(int argc, char *argv[])
{
    int sockfd, n;
    struct sockaddr_in servaddr;
    char sendline[1024], recvline[1024];

    if (argc != 2)
    {
        printf("usage: tcpcli <IPaddress>\n");
        exit(1);
    }

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
        perror("socket error");

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(9877);
    inet_pton(AF_INET, argv[1], &servaddr.sin_addr);

    connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    while (fgets(sendline, 1024, stdin) != NULL)
    {
        printf(sendline);
        write(sockfd, sendline, strlen(sendline));

        if (read(sockfd, recvline, 1024) == 0)
        {
            printf("str_cli: server terminated prematurely\n");
            exit(1);
        }

        fputs(recvline, stdout);
    }

    exit(0);
}
参考文献

[1] W.Richard Stevens. Unix网络编程卷一:套接字联网API. 人民邮电出版社.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值