2

我完全没有使用 C(或任何其他非托管语言)编程的背景,但我想在我的 .NET 应用程序中使用 IP Helper API 中的GetBestInterface函数。我试图了解如何使用 P/invoke 调用来调用本机方法,但有很多事情对我来说没有意义(例如托管代码中的类型如何映射到非托管类型)。

System.Net 命名空间中是否有隐藏在某个地方的替代函数可以做大致相同的事情?或者我可以使用现有的基类结合一些魔法来编写自己的替代方案吗?因为在我看来,这基本上就是:魔法。据我所知,该方法如何完成它的工作并没有真正的解释......

编辑

我刚刚在类上发现了 LocalEndPoint 属性System.Net.Sockets.Socket,我认为这在这方面可能非常有用。据我了解,它将代表我进行最佳接口选择,我只需要获取与本地端点对应的 NIC。

4

2 回答 2

3

我还需要GetBestInterface,因为我需要一个解决方案,该解决方案可以在无法访问要探测的远程地址时工作(Steven 的解决方案会遇到问题),并且与我移植到 C# 的本机应用程序 100% 兼容。

幸运的是,您不必移植GetAdaptersInfo和所有相应的IP_ADAPTER_INFO子结构混乱来查找哪个网络接口具有由 .NET 返回的索引GetBestInterface,因为.NET 也可以从接口检索该索引

所以,首先移植GetBestInterface方法如下:

[DllImport("iphlpapi")]
private static extern int GetBestInterface(uint dwDestAddr, ref uint pdwBestIfIndex);

然后,编写一个包装函数,如下所示:

private static NetworkInterface GetBestNetworkInterface(IPAddress remoteAddress)
{
    // Get the index of the interface with the best route to the remote address.
    uint dwDestAddr = BitConverter.ToUInt32(remoteAddress.GetAddressBytes());
    uint dwBestIfIndex = 0;
    uint result = GetBestInterface(dwDestAddr, ref dwBestIfIndex);
    if (result != 0)
        throw new NetworkInformationException((int)result);

    // Find a matching .NET interface object with the given index.
    foreach (NetworkInterface networkInterface in NetworkInterface.GetAllNetworkInterfaces())
        if (networkInterface.GetIPProperties().GetIPv4Properties().Index == dwBestIfIndex)
            return networkInterface;

    throw new InvalidOperationException($"Could not find best interface for {remoteAddress}.");
}

然后,您可以像这样简单地调用该方法 - 不要忘记无法访问的主机现在也可以工作:

NetworkInterface bestNetworkInterface = GetBestNetworkInterface(IPAddress.Parse("8.8.8.8"));
于 2019-06-06T20:32:35.083 回答
1

好吧,我自己整理了一些东西,这些东西通过捎带框架自己进行界面选择的能力来工作。本质上它只是连接一个套接字,然后使用套接字上的 LocalEndPoint 属性来获取它正在使用的 NIC。可能不是最好的方法,但它对我有用。

使用以下导入语句:

Imports System.Net
Imports System.Net.NetworkInformation
Imports System.Net.Sockets

还有我超级棒的方法:

Private Function GetBestInterfaceManaged(ByVal address As IPAddress, ByVal port As Integer) As NetworkInterface

    Dim socket As Socket = New Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.IP)
    Dim outgoingInterface As NetworkInterface = Nothing

    Try
        socket.Connect(New IPEndPoint(address, port))

        If socket.Connected Then ' find the outgoing NIC

            Dim interfaces As List(Of NetworkInterface) = NetworkInterface.GetAllNetworkInterfaces.ToList()

            For Each nic As NetworkInterface In interfaces
                Dim properties As IPInterfaceProperties = nic.GetIPProperties

                For Each unicastAddress In properties.UnicastAddresses
                    If unicastAddress.Address.Equals(DirectCast(socket.LocalEndPoint, IPEndPoint).Address) Then
                        outgoingInterface = nic
                    End If
                Next
            Next

            If outgoingInterface Is Nothing Then
                Console.WriteLine("Darn... it didn't work!")
            Else
                Console.WriteLine("Outgoing interface: {0}", outgoingInterface.Name)
            End If

            Return outgoingInterface
        End If
    Catch ex As SocketException
        Console.WriteLine(ex.Message)
    End Try
    Return Nothing
End Function

这可以解决问题:

Sub Main()
    Dim hostEntry As IPHostEntry = Dns.GetHostEntry("www.stackoverflow.com")
    Dim NIC As NetworkInterface = GetBestInterfaceManaged(hostEntry.AddressList.First, 80)
    ' Other code goes down here
End Sub
于 2012-04-28T10:57:16.813 回答