UNIX套接字错误:非套接字上的套接字操作

时间:2017-01-25 10:31:07

标签: c sockets unix tcp

我正在尝试使用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");

2 个答案:

答案 0 :(得分:1)

你正在使用“client”变量作为socketto参数来发送到(),实际上你应该使用“sock”变量。

如果您对该程序进行操作,您可以看到以下内容:

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(默认情况下这是stdin文件描述符),实际上它应该是3(参见connect(...)line)

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

答案 1 :(得分:0)

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

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