Listen套接字只能在没有-std=c++的情况下在c++中工作
Listen socket works in g++ only without -std=c++11
我正在尝试编写一个基本的套接字应用程序,它将接受来自客户端的字符串,并对我在内存中的数据结构做一些工作。我试图利用c++ 11,因为我想有多线程的语言支持,我发现了一个奇怪的问题。我的代码在端口1234上创建一个监听套接字,并接受一个连接,打印出一个虚拟字符串并关闭它。如果我用:
编译它 g++ -o socket_test socket_test.cpp
它工作得很好,但是如果我用:
编译它 g++ -std=c++11 -o socket_test socket_test.cpp
可以编译并运行良好,但是如果我尝试连接到端口,它会返回并拒绝连接。测试代码在这里:
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
using namespace std;
int main()
{
cout << "Hello world, I'm going to open a socket on port 1234 now!" << endl;
int m_socket = 0;
int m_client = 0;
int port = 1234;
sockaddr_in serv_addr, client_addr;
socklen_t client_len;
m_socket = socket(AF_INET, SOCK_STREAM, 0);
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(port);
bind(m_socket, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
cout << "Listen returns: " << listen(m_socket, SOMAXCONN) << endl;
m_client = accept(m_socket, (struct sockaddr *)&client_addr, &client_len);
write(m_client, "Numa numan", 10);
close(m_client);
close(m_socket);
return 0;
}
我的构建环境是OSX 10.9.2,使用g++如下:
battra:socket_test matt$ g++ --version
Configured with: --prefix=/Applications/Xcode.app/Contents/Developer/usr
--with-gxx-include-dir=/Applications/Xcode.app/Contents/Developer/Platforms/
MacOSX.platform/Developer/SDKs/MacOSX10.9.sdk/usr/include/c++/4.2.1
Apple LLVM version 5.1 (clang-503.0.38) (based on LLVM 3.4svn)
Target: x86_64-apple-darwin13.2.0
Thread model: posix
battra:socket_test matt$
这个问题是一个典型的例子,说明为什么导入整个命名空间不是一个好主意。最终,导入的命名空间的内容将会改变——在本例中,在c++ 11中添加了std::bind
——并且在无声地改变含义之前,非限定名工作得很好。
您是否检查了bind
的返回值:
cout << "Bind returns: "
<< bind(m_socket, (struct sockaddr *)&serv_addr, sizeof(serv_addr))
<< endl;
所产生的编译错误会使问题更明显:
main.cpp:24:8: error:二进制表达式(
'basic_ostream<char, std::__1::char_traits<char> >'
和'__bind<int &, sockaddr *, unsigned long>'
)无效<< bind(m_socket, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) ^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
简单的解决方案是使用限定名::bind
来明确表示来自全局名称空间的bind
。更好的解决方案是从程序中完全删除using namespace std
,以避免将来出现类似的问题。
我怀疑这个问题与您正在使用-std=c++11
的事实无关,而更多的是与您第二运行该版本的事实有关。
如果您尝试运行第一个版本两次,您可能会在第二次运行时注意到相同的错误,因为端口尚未从第一次运行中释放。
如果您更改端口号并再次运行,它应该再次工作。
要解决这个问题,你应该看看这里的讨论。
我修改了你的程序使用SO_REUSEADDR
,它似乎工作一致,没有c++11
选项。请注意,我在Ubuntu上运行,所以你的OS X
可能会有不同的行为。
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
using namespace std;
int main()
{
cout << "Hello world, I'm going to open a socket on port 1234 now!" << endl;
int m_socket = 0;
int m_client = 0;
int port = 1236;
sockaddr_in serv_addr, client_addr;
socklen_t client_len;
m_socket = socket(AF_INET, SOCK_STREAM, 0);
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(port);
// Set to reuse
int so_reuseaddr = 1;
setsockopt(m_socket,SOL_SOCKET,SO_REUSEADDR,
&so_reuseaddr,
sizeof so_reuseaddr);
bind(m_socket, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
cout << "Listen returns: " << listen(m_socket, SOMAXCONN) << endl;
m_client = accept(m_socket, (struct sockaddr *)&client_addr, &client_len);
write(m_client, "Numa numan", 10);
close(m_client);
close(m_socket);
return 0;
}
- 在没有太多条件句的情况下,我如何避免被零除
- 为什么在没有显式默认构造函数的情况下,将另一个结构封装在联合中作为成员的结构不能编译
- 在未初始化映射的情况下,将值插入到映射的映射中
- 是默认情况下分配给char数组常量的值
- 为什么我不能在不创建字符串变量的情况下使用函数的字符串输出
- 如何在不产生任何垃圾的情况下获得C中的像素
- 在已经使用Git的情况下减少编译时间
- 为什么在Windows上的VS 2019和Clang 9中"size_t"在没有标题的情况下工作
- 如何在没有信号的情况下从C++执行QML插槽
- 如何在不知道向量大小的情况下输入向量内部的向量?
- 为什么在某些情况下不写入此文件?
- 为什么Mat类的两个对象可以在不重载运算符+的情况下添加
- 在没有Xcode的情况下在Mac捆绑包中嵌入框架
- UE4-如何在给定4个屏幕坐标的情况下缩放纹理或材质
- 为什么需要复制构造函数,在哪些情况下它们非常有用
- 在C++中如何在没有pow的情况下进行基础计算
- 松弛原子与无同步情况下的记忆连贯性
- 在 Windows 上,是否可以让 dll 在不使用 PATH 环境变量的情况下在另一个文件夹中查找依赖项?
- 我是c ++的新手,你能解释一下在这种情况下的指针吗
- 在没有参数列表的情况下使用模板名称"Event"无效,模板问题