为什么我得到一个糟糕的文件描述符错误

Why am I getting a bad file descriptor error?

本文关键字:文件 描述 错误 一个 为什么      更新时间:2023-10-16

我正在尝试编写一个像客户端一样的短程序,如telnet。我从用户那里收到如下输入:www.google.com 80(端口号)和/index.html然而,我得到了一些错误。当我编写一些调试信息时,它说我有一个坏的文件描述符,并且在文件描述符100 messagesize = 0上读取失败。

struct hostent * pHostInfo;
struct sockaddr_in Address;
long nHostAddress;
char pBuffer[BUFFER_SIZE];
unsigned nReadAmount;
int nHostPort = atoi(port);
vector<char *> headerLines;
char buffer[MAX_MSG_SZ];
char contentType[MAX_MSG_SZ];
int hSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (hSocket == SOCKET_ERROR)
{
  cout << "nCould Not Make a Socket!n" << endl;
  return 1;
}
pHostInfo = gethostbyname(strHostName);
memcpy(&nHostAddress,pHostInfo -> h_addr, pHostInfo -> h_length);
Address.sin_addr.s_addr = nHostAddress;
Address.sin_port=htons(nHostPort);
Address.sin_family = AF_INET;
if (connect(hSocket, (struct sockaddr *)&Address, sizeof(Address)) == SOCKET_ERROR)
{
  cout << "Could not connect to the host!" << endl;
  exit(1);
}
char get[] = "GET ";
char http[] = " HTTP/1.0rnHost: ";
char end[] = "rnrn";
strcat(get, URI);
strcat(get, http);
strcat(get, strHostName);
strcat(get, end);
strcpy(pBuffer, get);
int len = strlen(pBuffer);
send(hSocket, pBuffer, len, 0);
nReadAmount = recv(hSocket,pBuffer, BUFFER_SIZE, 0);
//Parsing of data here, then close socket
if (close(hSocket) == SOCKET_ERROR)
{
  cout << "Could not close the socket!" << endl;
  exit(1);
} 

谢谢!

您的代码中有缓冲区溢出。您正在尝试将一堆字符数组分配到一个只有5字节长的缓冲区中:

char get[] = "GET ";  // 'get' is 5 bytes long
char http[] = " HTTP/1.0rnHost: ";
char end[] = "rnrn";
strcat(get, URI);  // BOOM!
strcat(get, http);
strcat(get, strHostName);
strcat(get, end);

这可能会覆盖您的本地变量hSocket,从而导致"错误的文件描述符错误"。

因为你正在使用c++(你有一个vector在你的代码),只是使用std::string代替字符的C数组,这样你就不必担心内存管理

另一个导致'错误的文件描述符'的原因可能是当您试图写入没有写入权限的位置时。例如,当尝试在Windows环境中使用wget下载内容时:
Cannot write to 'ftp.org.com/parent/child/index.html' (Bad file descriptor).

另一种情况可能是目标输出文件名包含文件系统不允许的字符。例如,假设您在一台Linux计算机上运行wget命令,目标路径是一个用NTFS格式化的外部硬盘驱动器。Linux Ext4文件系统将允许诸如'?’,但NTFS文件系统不会。wget的默认行为是将目标文件夹命名为指定的url。如果url包含像":"这样的特殊字符,NTFS将不接受该字符,导致错误"错误的文件描述符"。您可以使用——no-host-directories参数在wget中禁用此默认行为。

'坏文件描述符'的另一个原因可能是意外地试图将()写入只读描述符。我被困了一段时间,因为我花了所有的时间寻找第二个close()或缓冲区溢出。