dup,dup2都是属于创建文件描述符的函数,所以返回值也是一个文件描述符。有时候我们希望把标准输入重定向到一个文件的时候,或者把标准输出重定向到一个网络连接的时候,就可以通过这两个函数实现。简而言之,这两个函数就是复制文件描述符的。dup(oldfd)执行以后会返回一个新的并且是当前系统可用的最小整数值作为新的文件描述符,此时这个新的文件描述符将指向oldfd指向的同一个文件,共享所有的锁定、读写和各项权限和标志位。dup(oldfd,newfd)的意思就是,将文件描述符newfd指向oldfd(或者说oldfd被复制之后新的文件描述符被设定为newfd),dup2多出来的就是第二个参数可以指定新文件描述符的数值,因此dup2的返回值一定是不小于newfd的整数数值的。
标准输入文件的文件描述符: 0
标准输出文件的文件描述符:1
标准错误输出的文件描述符:2
例1:
int fd1=2; int fd2; fd2 = dup(fd1); //此时新生成的文件描述符fd2 就是文件描述符fd1的拷贝。此时已经把标准错误输出重定向到了文件描述符为fd2的文件中了。
例2:
int fd = open("1.txt" , (O_RWDR|O_CREATE) , 0644); dup2(fd , 1); //此时文件描述符1就是文件描述符fd的拷贝。所以此时文件描述符1将指向文件描述符fd所指向的文件,此时标准输出文件已经重定向到1.txt中了。所以写到stdout中的东西会全部写到文件1.txt中。
参考博客:https://blog.csdn.net/nyist327/article/details/41049699
对CGI也没有太多了解,这次的代码只是对dup ,dup2函数的一个理解例子。。。
CGI.h文件
#ifndef _CGI_H
#define _CGI_H
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
const char *ip = "127.0.0.1";
const int port = 4507;
#endif
CGI_server.cpp文件
#include"CGI.h"
class server
{
private:
int sock;
public:
server();
void create_socket();
};
server::server()
{
int sock = 0;
}
void server::create_socket()
{
struct sockaddr_in address;
bzero(&address, sizeof(address));
address.sin_family = AF_INET;
inet_pton(AF_INET, ip, &address.sin_addr);
address.sin_port = htons(port);
sock = socket(PF_INET, SOCK_STREAM, 0);
assert(sock >= 0);
int ret = bind(sock,(struct sockaddr*)&address, sizeof(address));
assert(ret != -1);
ret = listen(sock,5);
assert(ret != -1);
struct sockaddr_in client;
bzero(&client, sizeof(client));
socklen_t client_addrlength = sizeof(client);
int connfd = accept(sock, (struct sockaddr*)&client, &client_addrlength);
if(connfd<0)
{
cout << "errno is :" << errno << endl;
}
else{
close(1);//close(STDOUT_FILENO)
int fd=dup2(connfd,1);
printf("new filled:\n",fd);//fd会将返回到客户端的socket上
close(connfd);
}
close(sock);
}
int main()
{
server CGI;
CGI.create_socket();
return 0;
}
在代码中,我们首先关闭了文件描述符为1的STDOUT_FILENO.然后复制了socket文件描述符connfd.因为dup总是返回系统中最小的可用文件描述符,所以返回值实际上也是1,也就是之前关闭的标准输出文件描述符的值。这样的话,服务器输出到标准输出的内容(在这里是将dup2返回的新的文件描述符发送到了客户端的socket上,也就是printf的内容会到达客户端的接收缓冲区中)就会直接发送到与客户连接的socket上,因此此处的printf调用的输出将被客户端获取而不是显示在服务器程序的终端上。这就是CGI服务器的基本工作原理。(为了测试客户端的socket是否有数据,多此一举写了一个客户端)
CGI_client.cpp
#include"CGI.h"
class client{
private:
int sock;
public:
client();
void create_client();
};
client::client()
{
int sock = 0;
}
void client::create_client()
{
struct sockaddr_in address;
bzero(&address, sizeof(address));
address.sin_family=AF_INET;
inet_pton(AF_INET, ip, &address.sin_addr);
address.sin_port = htons(port);
sock = socket(PF_INET, SOCK_STREAM, 0);
assert(sock >= 0);
if((connect(sock,(struct sockaddr*)&address, sizeof(address))) != -1)
{
char buf[30];
cout << "connect successful!\n";
if(recv(sock,buf,sizeof(buf),0) > 0)//测试socket接收缓存上是否接收到"new filefd:fd"
{
cout << buf << endl;
}
else{
cout << "the errno:" << errno << endl;
}
}
else{
cout << "the errno:" << errno <
测试截图: