每个人都知道一个监听套接字连接的进程的经典模型,并分派一个新进程来处理每个新的连接.通常的做法是让父进程立即在新创建的套接字上调用close,减少句柄数,以便只有子对一个新的套接字有一个句柄.
我已经看到,Linux中进程和线程的唯一区别是线程共享相同的内存.在这种情况下,我假设产生一个新的线程来处理新的连接也会重复文件描述符,并且还需要“父”线程关闭它的套接字的副本?
解决方法
线程共享相同的内存,所以它们共享相同的变量.如果在父线程中关闭套接字,它也将在子线程中关闭.
编辑:
man fork:小孩继承父系列的打开文件描述符的副本.
> man pthreads:线程共享一系列其他属性(即这些属性是进程范围而不是每个线程):[…]打开文件描述符
和一些代码:
#include <cstring> #include <iostream> using namespace std; #include <errno.h> #include <fcntl.h> #include <pthread.h> #include <unistd.h> // global variable int fd = -1; void * threadProc(void * param) { cout << "thread: begin" << endl; sleep(2); int rc = close(fd); if (rc == -1) { int errsv = errno; cout << "thread: close() Failed: " << strerror(errsv) << endl; } else { cout << "thread: file is closed" << endl; } cout << "thread: end" << endl; } int main() { int rc = open("/etc/passwd",O_RDONLY); fd = rc; pthread_t threadId; rc = pthread_create(&threadId,NULL,&threadProc,NULL); sleep(1); rc = close(fd); if (rc == -1) { int errsv = errno; cout << "main: close() Failed: " << strerror(errsv) << endl; return 0; } else { cout << "main: file is closed" << endl; } sleep(2); }
输出为:
thread: begin main: file is closed thread: close() Failed: Bad file descriptor thread: end