socket单向断开函数shutdown

目录

1、函数声明

2、shutdown()与close()/closesocket()的区别

3、适用情景

4、代码


因为socket是双向,client和server都可以进行input和output,所以有时候我们需要数据在socket上实现单向的传输,即数据往一个方向传输。单向的socket为半开放Socket。

要实现半开放式,需要用到shutdown()函数。

1、函数声明

/* Shut down all or part of the connection open on socket FD.
   HOW determines what to shut down:
     SHUT_RD   = No more receptions;
     SHUT_WR   = No more transmissions;
     SHUT_RDWR = No more receptions or transmissions.
   Returns 0 on success, -1 for errors.  */
extern int shutdown (int __fd, int __how) __THROW;

SHUT_RD:断开输入流。套接字无法接收数据(即使缓冲区收到数据也会被清除),无法调用输入相关函数。

SHUT_WD:断开输出流。套接字无法发送数据,但如果输出缓冲区中还有未传输的数据,则将传递到目标主机。

SHUT_RDWR:同时断开I/O流。相当于分两次调用shutdown(),其中一次以SHUT_RD为参数,另一次以SHUT_WR为参数。

2、shutdown()与close()/closesocket()的区别

在多进程环境中:

close()/closesocket()破坏当前的进程所引用的socket标识符,但是socket还是完好存在的,别的进程还是可以照样使用的(就像shared_ptr一样),当引用数降为0的时候,socket被释放销毁。

shutdown() 用来关闭连接,而不是套接字,不管调用多少次 shutdown(),套接字依然存在,直到调用 close() / closesocket() 将套接字从内存清除。而shutdown()的杀伤面积就比较大了,假设有一对父子进程公用一个socket,如果父进程对此socket对应的标识符进行了shutdown(fd,SHUT_RD)操作,则此socket的所有引用操作符均无法接收数据,只能发送数据了。

调用 close()/closesocket() 关闭套接字时,或调用 shutdown() 关闭输出流时,都会向对方发送 FIN 包。FIN 包表示数据传输完毕,计算机收到 FIN 包就知道不会再有数据传送过来了。

默认情况下,close()/closesocket() 会立即向网络中发送FIN包,不管输出缓冲区中是否还有数据,而shutdown() 会等输出缓冲区中的数据传输完毕再发送FIN包。也就意味着,调用 close()/closesocket() 将丢失输出缓冲区中的数据,而调用 shutdown() 不会。

3、适用情景

因为shutdown可以让socket实现单向传输,而半开放socket使用于以下场合:

  • 当你想要确保所有写好的数据已经发送成功时。如果在发送数据的过程中,网络出现断开或者异常,系统不一定会返回异常,这时候你要怎么确定对端已经接收到数据了呢? 可以使用shutdown()函数来确定数据是否发送成功,因为调用shutdown()函数时只有在缓存中的数据全部发送成功后才会返回。
  • 当不能往有些socket上写数据,或者不能从有些socket上读数据时,可以使用shutdown让其断开,后续对其的写或者读的操作将会返回异常。
  • 当多线程时,想防止其他线程或进程访问到该资源是,有或者你想彻底关闭这个socket,让别的线程或者进程也没法使用则可以调用shutdown()。

4、代码

/*================================================================
 *   Copyright (C) 2021 baichao All rights reserved.
 *
 *   文件名称:shutdownService.cpp
 *   创 建 者:baichao
 *   创建日期:2021年02月08日
 *   描    述:
 *
 ================================================================*/
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>

int main(){
    //创建套接字
    int serv_sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);

    //将套接字和IP、端口绑定
    struct sockaddr_in serv_addr;
    memset(&serv_addr, 0, sizeof(serv_addr));  //每个字节都用0填充
    serv_addr.sin_family = AF_INET;  //使用IPv4地址
    serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");  //具体的IP地址
    serv_addr.sin_port = htons(11230);  //端口
    bind(serv_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr));

    //进入监听状态,等待用户发起请求
    listen(serv_sock, SOMAXCONN);


    //接收客户端请求
    struct sockaddr_in clnt_addr;
    socklen_t clnt_addr_size = sizeof(clnt_addr);

    int clnt_sock = accept(serv_sock, (struct sockaddr*)&clnt_addr, &clnt_addr_size);
    int count = 0;
    int size = 3;
    while(size--)
    {

        //sleep(5);
        char str[100];
        int readNum = read(clnt_sock,str,sizeof(str));
        if(readNum <= 0)
            continue;
        std::cout<<"第"<<++count<<"次读取数据长度为:"<<readNum<<std::endl;
        // write(clnt_sock, , sizeof(str));

    }
    shutdown(clnt_sock,SHUT_RD);

    size = 3;
    while(size--)
    {

        //sleep(5);
        char str[100];
        int readNum = read(clnt_sock,str,sizeof(str));
        if(readNum <= 0)
            continue;
        std::cout<<"第"<<++count<<"次读取数据长度为:"<<readNum<<std::endl;
        // write(clnt_sock, , sizeof(str));

    }
    close(clnt_sock);
    close(serv_sock);
    return 0;
}
/*================================================================
 *   Copyright (C) 2021 baichao All rights reserved.
 *
 *   文件名称:shutdownClient.cpp
 *   创 建 者:baichao
 *   创建日期:2021年02月08日
 *   描    述:
 *
 ================================================================*/

#include<iostream>
#include <cstring>
#include <cstdlib>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

int main(){

    int serv_sock = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in serv_addr;
    memset(&serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    serv_addr.sin_port = htons(11230);

    connect(serv_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr));

    char context[100] = {0};
    for(int i = 0; i < sizeof(context)/sizeof(context[0]); ++i)
        context[i] = '1';
    int size = 3;
    while(size--)
    {
        write(serv_sock,context,sizeof(context)-1);
    }
    sleep(5);
    size = 3;
    while(size--)
    {
        write(serv_sock,context,sizeof(context)-1);
    }
    close(serv_sock);

    return 0;
}

运行结果:

 

 

 

 

 

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在Socket编程中,shutdown()函数通常用于关闭已连接的套接字的一部分或全部通信。shutdown()函数可以用于以下情况: 1. 关闭一个已连接套接字的一侧:当套接字的一侧想要停止发送数据时,可以使用shutdown()函数关闭套接字的一侧。这个时候,另一侧仍然可以发送和接收数据,而关闭的一侧则不能再发送数据。 2. 完全关闭一个已连接套接字:当一个套接字的双方都不需要再进行通信时,可以使用shutdown()函数关闭整个套接字。这个时候,套接字的两侧都不能再发送和接收数据了。 需要注意的是,关闭套接字并不是必需的,因为在套接字被关闭时,操作系统会自动释放套接字所占用的资源。但是,当应用程序需要显式地关闭套接字时,可以使用shutdown()函数。 ### 回答2: shutdown()函数可以用于关闭套接字的一个或多个方向的通信。它适用于以下场景: 1. 安全地关闭套接字:在关闭套接字之前,可以使用shutdown()函数停止对该套接字的读取和写入操作。这样可以确保在关闭套接字之前,所有未发送和接收的数据都已经处理完毕,避免数据丢失或混乱。 2. 断开连接:在网络编程中,当一个套接字与另一个套接字建立连接之后,可以使用shutdown()函数断开与对方的连接。通过关闭套接字的某个方向的通信,可以使得另一个套接字无法再发送或接收数据,从而实现断开连接的效果。 3. 超时处理:使用shutdown()函数可以设置超时,即在一定时间内没有收到对方的数据或没有发送数据,就可以通过关闭套接字来处理超时情况,从而及时释放资源,提高程序的效率和可靠性。 总之,shutdown()函数适用于需要精确控制套接字通信的场景,可以确保数据的安全传输和及时释放资源。在网络编程中,合理地使用shutdown()函数可以提高程序的效率和可靠性。 ### 回答3: socket中的shutdown()函数用于在网络通信中关闭连接的一部分。它适用于以下场景: 1. 半关闭连接:可以通过shutdown()函数关闭socket的读或写操作,从而实现半关闭连接。例如,在一个TCP连接中,一个端口发送完所有的数据后可以调用shutdown(socket.SHUT_WR)来关闭写操作,而接收端仍可以继续读取数据。这种情况下,可以使用shutdown()函数来表明数据的发送已经完成,但是仍然可以接收数据。 2. 优雅关闭连接:在网络通信中,通常需要先发送一个特殊的标志来告知对方关闭连接。在TCP中,如果直接调用close()函数来关闭连接,对方可能会认为是异常关闭而无法正确处理。因此,可以通过调用shutdown()函数来发送这个特殊的标志,然后再调用close()函数来关闭连接。这样可以告知对方连接即将关闭,对方可以做一些必要的处理。 3. 超时关闭连接:在网络通信中,有些情况下需要在一定时间内关闭连接。通过设置超时时间,可以在规定的时间内调用shutdown()函数来关闭连接。如果在超时时间内没有调用close()函数来关闭连接,则系统会自动关闭连接。 综上所述,shutdown()函数适用于需要局部关闭连接、优雅关闭连接或者超时关闭连接的场景。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值