【问题标题】:UNIX socket error : socket operation on non-socketUNIX 套接字错误:非套接字上的套接字操作
【发布时间】:2017-06-10 11:49:59
【问题描述】:

我正在尝试使用 UNIX 套接字和 STREAM 协议进行客户端-服务器通信。

我的服务器运行良好,但我的客户端不工作。每当我尝试发送或接收数据时,都会出现错误:“非套接字上的套接字操作”。我真的不明白它来自哪里,因为我的服务器非常相似,我没有任何问题。我的服务器在我的本地机器 (127.0.0.1) 和端口 5000 上。它处于打开状态并正在侦听(我使用 netstat 命令进行了检查)。

代码在那里:

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

#define CHECK(ret, mes) if ((ret) == -1) {perror(mes); exit(-1);}

#define STRING_LENGTH 250

int createSocketINETClient(short mode, char *addr, short port)
{
    int s;
    struct sockaddr_in moi, server;
    int moi_len, server_len;

    moi.sin_family = AF_INET;
    moi.sin_port = htons(port);
    moi.sin_addr.s_addr = inet_addr(addr);

    memset(moi.sin_zero, 0, 8);

    s = socket(AF_INET, mode, 0);

    CHECK(s, "socket");

    moi_len = sizeof(moi);

    CHECK(bind(s, (struct sockaddr*) &moi, moi_len), "bind");

    return s;
}

void infoSocket (int s)
{
    struct sockaddr_in sock_addr;
    socklen_t len = sizeof(sock_addr);

    getsockname(s, (struct sockaddr*) &sock_addr, &len);

    printf("Onfo of socket %d\n", s);
    printf("\t IP : %s\n", inet_ntoa(sock_addr.sin_addr));
    printf("\t port : %d\n\n", ntohs(sock_addr.sin_port));
}

int main ()
{
    int bytes;
    int sock = createSocketINETClient(SOCK_STREAM, "0.0.0.0", 0);
    struct sockaddr_in serveurSock;
    int client = 0, clientSockLen = 0;
    char message[] = "I am a message that is supposed to WORK !!!!\n";
    char fromServer[STRING_LENGTH] = "";
    infoSocket(sock);


    serveurSock.sin_family = AF_INET;
    serveurSock.sin_port = htons(5000);
    serveurSock.sin_addr.s_addr = inet_addr("127.0.0.1");

    memset(serveurSock.sin_zero, 0, 8);

    CHECK(connect(sock, (struct sockaddr*) &serveurSock, sizeof(serveurSock)), "connect");

    usleep(1000000);

    CHECK((bytes = send(client, message, sizeof(message), 0)), "send");
    printf("Message sent to server : %d bytes, \"%s\"\n", bytes, message);

    CHECK((bytes = recv(client, fromServer, sizeof(fromServer), 0)), "recv");
    printf("Message received from server : %d bytes, \"%s\"\n", bytes, fromServer);

    close(client);
    printf("Client released !\n\n");

    return 0;
}

我做错了什么?

编辑:错误来自这一行:

CHECK((bytes = send(client, message, sizeof(message), 0)), "send");

【问题讨论】:

  • 来自哪个函数?
  • 我编辑了帖子以包含错误所在的行,对此感到抱歉
  • 你使用的是client,而你应该使用sock
  • 这是 Berkeley 套接字,而不是 UNIX 套接字。 UNIX 套接字有所不同,它们使用文件名作为地址而不是 IP 号和端口。 en.wikipedia.org/wiki/Berkeley_sockets

标签: c sockets unix tcp


【解决方案1】:

在发布的代码中,您初始化了int client = 0,然后我看不到您对其进行了更改。因此,当您致电 send(0, ...) 时,您显然会收到该错误。

正如@EJP 在对您的问题的评论中已经提到的那样,这可能是一个错字,因为看起来您确实打算使用sock(当您连接它时:connect(sock, ...) 而不是client 调用@ 987654326@.

【讨论】:

    【解决方案2】:

    您正在使用“client”变量作为 sendto() 的套接字参数,而实际上您应该使用“sock”变量。

    如果你 strace 程序,你可以看到以下内容:

    connect(3, {sa_family=AF_INET, sin_port=htons(5000), sin_addr=inet_addr("127.0.0.1")}, 16) = 0
    nanosleep({tv_sec=1, tv_nsec=0}, NULL)  = 0
    sendto(0, "I am a message that is supposed "..., 46, 0, NULL, 0) = -1 ENOTSOCK (Socket operation on non-socket)
    

    注意第一个 sendto() 参数是 0(默认情况下这是标准输入文件描述符),而实际上它应该是 3(参见 connect(...) 行)

    作为旁注,您不需要 bind() 客户端套接字,除非您有明确的理由这样做(使用默认路由以外的其他路由,在某处绕过防火墙规则等)。默认情况下,操作系统将分配默认路由经过的网络接口的 ip 和一个随机空闲端口。

    【讨论】:

    • 谢谢!我看到这是一个愚蠢的错误......我打算使用客户端作为套接字的变量,但我在某个地方搞砸了。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-01-21
    • 2014-09-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多