C中的网络编程(简单服务器演示代码混淆)

时间:2013-02-14 05:22:45

标签: c networking

我正在从Jon Erickson的Hacking,剥削艺术这本书中学习,我对他提供的一个简单的代码示例感到困惑。代码是设置一个简单的服务器,但是当我编译它(没有错误)并运行代码时,它会挂起

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "hacking.h"

#define PORT 7890   // the port users will be connecting to

int main(void) {
int sockfd, new_sockfd;  // listen on sock_fd, new connection on new_fd
struct sockaddr_in host_addr, client_addr;  // my address information
socklen_t sin_size;
int recv_length=1, yes=1;
char buffer[1024];

if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1)
    fatal("in socket");

if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int)) == -1)
    fatal("setting socket option SO_REUSEADDR");

host_addr.sin_family = AF_INET;      // host byte order
host_addr.sin_port = htons(PORT);    // short, network byte order
host_addr.sin_addr.s_addr = INADDR_ANY; // automatically fill with my IP
memset(&(host_addr.sin_zero), '\0', 8); // zero the rest of the struct

if (bind(sockfd, (struct sockaddr *)&host_addr, sizeof(struct sockaddr)) == -1)
    fatal("binding to socket");

if (listen(sockfd, 5) == -1)
    fatal("listening on socket");

while(1) {    // Accept loop
    sin_size = sizeof(struct sockaddr_in);
    new_sockfd = accept(sockfd, (struct sockaddr *)&client_addr, &sin_size);
    if(new_sockfd == -1)
        fatal("accepting connection");
    printf("server: got connection from %s port %d\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));
    send(new_sockfd, "Hello World!\n", 13, 0);
    recv_length = recv(new_sockfd, &buffer, 1024, 0);
    while(recv_length > 0) {
        printf("RECV: %d bytes\n", recv_length);
        dump(buffer, recv_length);
        recv_length = recv(new_sockfd, &buffer, 1024, 0);
    }
    close(new_sockfd);
}
return 0;
}

我做了一点printf()来找出我挂在哪里,结果就是这一行

sin_size = sizeof(struct sockaddr_in);

我不确定它是否与我的环境有关,或者有些东西我不知道。本书使用的环境无法再更新(Ubuntu的一些旧版本)。所以我目前正在使用最新版本。

有人可以向我解释为什么程序不起作用吗?

如果在学习网络章节之前需要了解一些基本知识,请告诉我们。

3 个答案:

答案 0 :(得分:2)

accept收到来自客户端程序的传入连接之后,该行程将在sizeof之后继续执行。你的printf显示接受被调用但被阻止了。

您需要使用正确的选项(IP /端口)编译和运行客户端以连接到此服务器程序。

<强>更新 如果192.168.42.248来自本书,那么您可能正在尝试连接到错误的IP。试试telnet 127.0.0.1 7890

答案 1 :(得分:2)

它是一个服务器,它将“挂起”,直到你连接到端口7890.这是程序的全部要点(对于更多的细节,它阻止,因为accept()正在等待连接)

假设您正在运行unix,请在运行时尝试在同一台计算机的终端中键入echo "hi there" | nc localhost 7890,然后您将看到它如何“取消阻止”

答案 2 :(得分:0)

告诉你线程中的所有评论,我建议使用以下命令行连接telnet:telnet localhost 7890

telnet将连接到的主机和要连接到此主机的端口作为参数。使用“localhost”类似于使用环回IP 127.0.0.1。

为什么连接服务器会解决“挂起”问题?如果您可以阅读man page或编程环境的任何其他文档,则接受阻止。这意味着在客户端连接之前,函数不会返回。连接函数后,返回为连接客户端创建的套接字的句柄,该套接字可用于通信。