Socket应用编程实验(二)

本文详细介绍了Socket应用编程中的客户端和服务器端的连接过程,包括服务器的socket(), bind(), listen(), accept()和客户端的socket(), connect()函数的使用。强调了connect()函数在客户端的重要性,并解析了数据传输的send()和recv()函数。同时,提到了网络字节序与本地字节序的转换在跨平台通信中的作用。
摘要由CSDN通过智能技术生成

Socket应用编程实验(2)

五、申请建立连接

在服务器端的函数调用之后,来到客户端编程(read/write),客户端编程相对服务器端简单。(只需要再学一个函数)

  • 回顾流程
服务器端

socket() --> bind() --> listen() --> accept() --> read()/write() --> close()
创建套接字、分配套接字地址、等待连接请求状态、允许连接、进行数据交换、断开连接  

客户端
socket() --> connect() --> read()/write() --> close()
创建套接字、请求连接、进行数据交换、断开连接  
 // 掌握 connect()函数,就可以自己写一个服务器端和客户端来进行实时通信
  • connect
# include<sys/types.h>
# include<sys/socket.h>
int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
int sockfd: socket文件描述符
const struct sockaddr *addr: 传入参数,指定服务器端地址信息,含IP地址和端口号
socklen_t addrlen:传入参数,传入 sizeof(addr)大小
返回值: 成功: 0 失败: -1,设置 errno

当客户端调用 connect() 函数之后,发生下一情况之一才会返回(完成函数调用)

  • 服务端接收连接请求
  • 发生断网的异常情况而中断连接请求

所谓的“接收连接"并不意味着服务器调用 accept() 函数。其实是服务器端把连接请求信息记录到等待队列。因此 connect() 函数返回后并不进行数据交换,而是要等服务器端 accept() 之后才能进行数据交换(read、write)。

客户端需要调用connect()连接服务器,connect和bind 的参数形式一致,区别在于 bind 的参数是自己的地址,而 connect 的参数是对方的地址。

  • 连接的另一方需要主动建立连接
int connect(int sockfd, const struct sockaddr* addr, socklen_t addrlen);
  • sockfd: 之前建立的 socket 文件描述符
  • addr : 对端(服务器端)的网络地址(包括IP地址和端口号)
  • addrlen: addr的大小(不同协议的addr大小不同)
  • 返回值指示连接是否建立成功
struct sockaddr_in dst;
dst.sin_family = AF_INET;
dst.sin_addr.s_addr = inet_addr("10.0.0.1"); 
// 本地连接的IP地址是 10.0.0.1
dst.sin_port = htons(80);

connect(sock,(struct sockaddr *)&dst,sizeof(dst));

六、数据传输

  • 数据发送方

    int send(int sockfd, const void *buf,size_t len,int flags);
    int sockfd: 发送端套接字描述符
    const void *buf: 指明一个存放应用程序要发送数据的缓冲区
    size_t len: 指明实际要发送的数据的字节数;
    int flags: 一般置 0
    
  • 数据接收方

    int recv(int sockfd, void *buf,size_t len,int flags);
    
  • 连接任意一端都可以发送或者接收数据

  • 对于一个阻塞式(blocking)socket

    • send直到所有数据被拷贝到协议栈缓存中才返回,或产生错误返回
    • recv直到接收到数据(不一定等于len),或对端连接关闭(返回0),或产生错误才返回。

七、网络字节序与本地字节序

  • 网络协议使用网络字节序(即大端字节序)传输数据,地址低位存储值的高位。
  • Intel的 x86 平台使用的是小端字节序,地址低位存储值的低位

为了保证数据在不同主机之间传输时能够被正确解释,主机在发送和接收数据时,需要进行网络字节序与主机字节序之间的相互转换:

htons() // 主机到网络,整型数据
htonl() // 主机到网络,长整型
htonll()
ntohs() / ntohl() / ntohll()
// 只有在收发整数数据时才需要转换字节序,收发字符串/字节流时不需要关心
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值