- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个带有 2 个 NIC 的 Linux 系统,其中一个 (eth0
) 是默认网关。另一个 (eth1
) 不是默认网关。两者都具有与互联网连接的网络连接 - 因此链接状态已启动。
我想定期检查 eth1
以查看是否可以通过它路由 Internet 流量。在检查 eth1
时,我需要保持通过 eth0
的 Internet 流量路由不间断。如果我可以通过 eth1
建立连接,我将切换我的默认网关(因为这是我更便宜的连接)。
我试图通过打开一个套接字来检查连接,将它绑定(bind)到 eth1
并连接到一个端口。我在示例中使用了 8.8.8.8:443
。
bool canConnect(const string interface)
{
bool result = false;
sockaddr_in host;
timeval timeout;
int sock = ::socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);
if (sock == -1)
return result;
struct ifreq ifr;
::strncpy(ifr.ifr_name, interface.c_str(), IFNAMSIZ);
int res = setsockopt(_sock, SOL_SOCKET, SO_BINDTODEVICE, &ifr, sizeof(ifr));
if (res < 0)
{
::close(sock);
return result;
}
int reuse = 1;
if (::setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(int)) < 0)
{
::close(sock);
return result;
}
::inet_pton(AF_INET, "8.8.8.8", host.sin_addr);
host.sin_family = AF_INET;
host.sin_port = htons(443);
if (::connect(sock, (struct sockaddr *)&host , sizeof(host)) < 0)
{
int error = errno;
if (error != EINPROGRESS) // socket is non-blocking
{
::close(sock);
return result;
}
}
fd_set writeSet;
fd_set readSet;
FD_ZERO(&readSet);
FD_ZERO(&writeSet);
FD_SET(sock, &readSet);
FD_SET(sock, &writeSet);
timeout.tv_sec = 5;
timeout.tv_usec = 0;
int res = ::select(sock + 1, &readSet, &writeSet, NULL, &timeout);
if (res < 0)
{
::close(sock);
return result;
}
else if (res == 0)
{
// function always exits here
::close(sock);
return result;
}
else if (FD_ISSET(sock, &writeSet) || FD_ISSET(sock, &readSet))
{
int val;
socklen_t len = sizeof(val);
::getsockopt(sock, SOL_SOCKET, SO_ERROR, &val, &len);
if (val == 0)
result = true;
}
::close(sock);
return result;
}
当我运行代码时,select()
总是超时。我从来没有设法连接到 8.8.8.8:443
,即使 eth1
具有 Internet 连接(我已经通过将默认网关切换到 eth1< 证明了这一点
暂时。
我是否遗漏了一些代码,或者我是否需要内核选项才能使其正常工作?
我打开了 IP 转发。
最佳答案
如果connect()
成功并等待,您应该检查 select()
仅可写,根本不可读。这在 connect()
mangage 中说明:
EINPROGRESS
The socket is nonblocking and the connection cannot be completed immediately. It is possible toselect(2)
orpoll(2)
for completion by selecting the socket for writing. Afterselect(2)
indicates writability, usegetsockopt(2)
to read theSO_ERROR
option at levelSOL_SOCKET
to determine whetherconnect()
completed successfully (SO_ERROR
is zero) or unsuccessfully (SO_ERROR
is one of the usual error codes listed here, explaining the reason for the failure).
此外,套接字只有在有入站数据要读取时才可读,但您连接到 HTTPS 端口,因此在您的示例中永远不会有任何数据要读取,因为您没有发送 HTTPS 握手请求.
另外,你不应该调用 select()
除非connect()
实际上失败了 EINPROGRESS
.如果connect()
返回 0,连接成功,没有阻塞。所以,你应该移动你的 select()
调用你的if (connect < 0)
检查后阻止 errno
.
此外,您没有将正确的套接字传递给 setsockopt(SO_BINDTODEVICE)
.
此外,您没有将正确的值传递给 dst
inet_pton()
的参数.你传递给它一个 in_addr
, 但它期望一个 in_addr*
相反。
此外,无需使用 SO_REUSEADDR
在这个例子中,因为你不是 bind()
在调用 connect()
之前连接到特定的本地 IP/端口对.
尝试更像这样的东西:
bool canConnect(const string &interface)
{
bool result = false;
int sock = ::socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);
if (sock == -1)
return result;
struct ifreq ifr;
::strncpy(ifr.ifr_name, interface.c_str(), IFNAMSIZ);
int res = setsockopt(sock, SOL_SOCKET, SO_BINDTODEVICE, &ifr, sizeof(ifr));
if (res < 0)
{
::close(sock);
return result;
}
sockaddr_in host;
::inet_pton(AF_INET, "8.8.8.8", &(host.sin_addr));
host.sin_family = AF_INET;
host.sin_port = htons(443);
result = (::connect(sock, (struct sockaddr *)&host, sizeof(host)) == 0);
if (!result)
{
if (errno != EINPROGRESS) // socket is non-blocking
{
::close(sock);
return result;
}
fd_set writeSet;
timeval timeout;
FD_ZERO(&writeSet);
FD_SET(sock, &writeSet);
timeout.tv_sec = 5;
timeout.tv_usec = 0;
res = ::select(sock + 1, NULL, &writeSet, NULL, &timeout);
if (res <= 0)
{
::close(sock);
return result;
}
int val;
socklen_t len = sizeof(val);
::getsockopt(sock, SOL_SOCKET, SO_ERROR, &val, &len);
result = (val == 0);
}
::close(sock);
return result;
}
也就是说,8.8.8.8
是 DNS 服务器,那么您为什么会认为它也在运行 HTTPS 服务器?确保您连接到实际的 HTTPS 服务器(为什么是 HTTPS 而不是 HTTP?)。您可以考虑使用 getaddrinfo()
而不是获取 www.google.com
的 IP 地址(或任何其他常用的网络服务器)而不是硬编码 8.8.8.8
.
关于c++ - 如何在非网关接口(interface) linux 上检测互联网连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47286590/
编写一个仅用于集中其他接口(interface)的接口(interface)是好的做法还是坏的做法? interface InterfaceA : InterfaceB, InterfaceC { }
有没有一种方法可以确定具体类型从任意接口(interface)列表?我知道类型转换,但我想知道所有满意的接口(interface)。 例如,给定: type Mover interface { Mo
我正在尝试制作斐波那契堆。 (在我正在上的算法课中多次提到它们,我想检查一下。)我希望堆使用任何类型的节点,所以我定义了一个 Node 接口(interface): package node type
这是我的代码: type IA interface { FB() IB } type IB interface { Bar() string } type A struct {
示例 A: // pseudo code interface IFoo { void bar(); } class FooPlatformA : IFoo { void bar() {
合并它编译的 leppies 反馈 - 但 IMO 有一些缺点,我希望编译器强制每个子类定义它们自己的 Uri 属性。现在的代码: [] type UriUserControl() = inh
我正在构建一个项目,该项目从用户那里获取一个术语,然后执行谷歌搜索并返回一个 json 格式的标题列表。 我正在使用 serpwow API 来执行谷歌搜索并试图解析响应。 但是我收到的错误是: pa
我只想在其他接口(interface)中实现某些接口(interface),我不希望它们能够被类直接继承。 提前致谢! 最佳答案 您不能在 C# 中执行此操作 - 任何类都可以实现它有权访问的任何接口
我是 Go 的新手,还有一些我还没有掌握的技巧 例如,我有一个可以这样调用的函数: myVar.InitOperation("foo",Operator.EQUAL,"bar") myVar.Init
我有一个通用接口(interface)来描述对输出流的访问,如下所示: interface IOutput { function writeInteger(aValue:Int):Void;
我正在做一个项目,我想通过某种接口(interface)(最好是 USB)将光电探测器电路安装到计算机上。但是,由于我是新手,所以我不知道应该朝哪个方向处理这个问题。假设我有一个带有 USB 连接的光
背景 我正在尝试创建一个简单的应用程序,以真正理解DDD + TDD + etc的整个堆栈。我的目标是在运行时动态注入DAL存储库类。这让我 域和应用程序服务层可测试。我打算用“穷人的DI”来完成 现
在 Java 中,接口(interface)扩展接口(interface)是完全合法的。 UML 中的这种关系看起来像“扩展”关系(实线、闭合、未填充的箭头)还是“实现”关系(虚线、闭合、未填充的箭头
我想创建一个具有相等和比较函数默认实现的接口(interface)。 如果我从类型 IKeyable 中删除所有内容除了Key成员,只要我不添加默认实现,它就是一个有效的接口(interface)。从
COM 中的双接口(interface)是能够通过 DispInterface 或 VTable 方法访问的接口(interface)。 现在有人可以告诉我这两种方法之间到底有什么区别吗? 我认为 V
我有一个类方法,它返回一个可以迭代的员工列表。返回列表的最佳方式是什么?通常我只返回一个 ArrayList。然而,据我了解,界面更适合这种类型的操作。哪个是最好使用的界面?另外,为什么返回接口(in
我想从包装类外部实例化一个内部非静态接口(interface)。 这可能吗? 考虑以下代码: shared class AOuterClass() { Integer val = 3; shared
我为一个类编写了一个接口(interface),如下所示: public interface IGenericMultipleRepository { Lazy> addresses { ge
我是 UML 的初学者,现在我正在创建一个序列图,问题是我想根据用户输入实现 DAO 接口(interface)。如何在时序图中正确绘制以实现接口(interface)。 最佳答案 您不会在 SD 上
要使用 jsr 303 验证创建有条件验证的组,请将接口(interface)类传递给注释,如下所示: @NotEmpty (groups={UpdateValue.class}) 我有很多不同的接口
我是一名优秀的程序员,十分优秀!