C# 如何通过.NET访问ARP协议信息?

C# 如何通过.NET访问ARP协议信息?,c#,.net,ip,lan,arp,C#,.net,Ip,Lan,Arp,我试图找出局域网中哪些设备在线,哪些设备离线 我见过许多程序做一种图形化的网络概览,显示局域网IP和MAC地址 我想知道这些(ARP?)信息是否以及如何从C#/.NET?中提取,如果您知道哪些设备可以使用。这将允许您至少填写ARP表。如果有必要,您可以始终执行ARP-a并解析输出。这里还有一个链接,展示了如何使用pinvoke呼叫。我在下面列举了Ping类的例子,以及如何使用GetIpNetTable访问ARP表 using System; using System.Runtime.Intero

我试图找出局域网中哪些设备在线,哪些设备离线
我见过许多程序做一种图形化的网络概览,显示局域网IP和MAC地址


我想知道这些(ARP?)信息是否以及如何从
C#/.NET

中提取,如果您知道哪些设备可以使用。这将允许您至少填写ARP表。如果有必要,您可以始终执行ARP-a并解析输出。这里还有一个链接,展示了如何使用pinvoke呼叫。我在下面列举了Ping类的例子,以及如何使用GetIpNetTable访问ARP表

using System;
using System.Runtime.InteropServices;
using System.ComponentModel; 
using System.Net;

namespace GetIpNetTable
{
   class Program
   {
      // The max number of physical addresses.
      const int MAXLEN_PHYSADDR = 8;

      // Define the MIB_IPNETROW structure.
      [StructLayout(LayoutKind.Sequential)]
      struct MIB_IPNETROW
      {
         [MarshalAs(UnmanagedType.U4)]
         public int dwIndex;
         [MarshalAs(UnmanagedType.U4)]
         public int dwPhysAddrLen;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac0;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac1;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac2;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac3;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac4;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac5;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac6;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac7;
         [MarshalAs(UnmanagedType.U4)]
         public int dwAddr;
         [MarshalAs(UnmanagedType.U4)]
         public int dwType;
      }

      // Declare the GetIpNetTable function.
      [DllImport("IpHlpApi.dll")]
      [return: MarshalAs(UnmanagedType.U4)]
      static extern int GetIpNetTable(
         IntPtr pIpNetTable,
         [MarshalAs(UnmanagedType.U4)]
         ref int pdwSize,
         bool bOrder);

      [DllImport("IpHlpApi.dll", SetLastError = true, CharSet = CharSet.Auto)]
      internal static extern int FreeMibTable(IntPtr plpNetTable);

      // The insufficient buffer error.
      const int ERROR_INSUFFICIENT_BUFFER = 122;

      static void Main(string[] args)
      {
         // The number of bytes needed.
         int bytesNeeded = 0;

         // The result from the API call.
         int result = GetIpNetTable(IntPtr.Zero, ref bytesNeeded, false);

         // Call the function, expecting an insufficient buffer.
         if (result != ERROR_INSUFFICIENT_BUFFER)
         {
            // Throw an exception.
            throw new Win32Exception(result);
         }

         // Allocate the memory, do it in a try/finally block, to ensure
         // that it is released.
         IntPtr buffer = IntPtr.Zero;

         // Try/finally.
         try
         {
            // Allocate the memory.
            buffer = Marshal.AllocCoTaskMem(bytesNeeded);

            // Make the call again. If it did not succeed, then
            // raise an error.
            result = GetIpNetTable(buffer, ref bytesNeeded, false);

            // If the result is not 0 (no error), then throw an exception.
            if (result != 0)
            {
               // Throw an exception.
               throw new Win32Exception(result);
            }

            // Now we have the buffer, we have to marshal it. We can read
            // the first 4 bytes to get the length of the buffer.
            int entries = Marshal.ReadInt32(buffer);

            // Increment the memory pointer by the size of the int.
            IntPtr currentBuffer = new IntPtr(buffer.ToInt64() +
               Marshal.SizeOf(typeof(int)));

            // Allocate an array of entries.
            MIB_IPNETROW[] table = new MIB_IPNETROW[entries];

            // Cycle through the entries.
            for (int index = 0; index < entries; index++)
            {
               // Call PtrToStructure, getting the structure information.
               table[index] = (MIB_IPNETROW) Marshal.PtrToStructure(new
                  IntPtr(currentBuffer.ToInt64() + (index *
                  Marshal.SizeOf(typeof(MIB_IPNETROW)))), typeof(MIB_IPNETROW));
            }

            for (int index = 0; index < entries; index++)
            {
               MIB_IPNETROW row = table[index];
               IPAddress ip=new IPAddress(BitConverter.GetBytes(row.dwAddr));
               Console.Write("IP:"+ip.ToString()+"\t\tMAC:");

               Console.Write( row.mac0.ToString("X2") + '-');
               Console.Write( row.mac1.ToString("X2") + '-');
               Console.Write( row.mac2.ToString("X2") + '-');
               Console.Write( row.mac3.ToString("X2") + '-');
               Console.Write( row.mac4.ToString("X2") + '-');
               Console.WriteLine( row.mac5.ToString("X2"));

            }
         }
         finally
         {
            // Release the memory.
            FreeMibTable(buffer);
         }
      }
   }
}
这是Ping类的一个示例

using System;
using System.Net;
using System.Net.NetworkInformation;
using System.Text;

namespace Examples.System.Net.NetworkInformation.PingTest
{
    public class PingExample
    {
        // args[0] can be an IPaddress or host name.
        public static void Main (string[] args)
        {
            Ping pingSender = new Ping ();
            PingOptions options = new PingOptions ();

            // Use the default Ttl value which is 128,
            // but change the fragmentation behavior.
            options.DontFragment = true;

            // Create a buffer of 32 bytes of data to be transmitted.
            string data = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa";
            byte[] buffer = Encoding.ASCII.GetBytes (data);
            int timeout = 120;
            PingReply reply = pingSender.Send (args[0], timeout, buffer, options);
            if (reply.Status == IPStatus.Success)
            {
                Console.WriteLine ("Address: {0}", reply.Address.ToString ());
                Console.WriteLine ("RoundTrip time: {0}", reply.RoundtripTime);
                Console.WriteLine ("Time to live: {0}", reply.Options.Ttl);
                Console.WriteLine ("Don't fragment: {0}", reply.Options.DontFragment);
                Console.WriteLine ("Buffer size: {0}", reply.Buffer.Length);
            }
        }
    }
}
这是GetIpNetTable的一个示例

using System;
using System.Runtime.InteropServices;
using System.ComponentModel; 
using System.Net;

namespace GetIpNetTable
{
   class Program
   {
      // The max number of physical addresses.
      const int MAXLEN_PHYSADDR = 8;

      // Define the MIB_IPNETROW structure.
      [StructLayout(LayoutKind.Sequential)]
      struct MIB_IPNETROW
      {
         [MarshalAs(UnmanagedType.U4)]
         public int dwIndex;
         [MarshalAs(UnmanagedType.U4)]
         public int dwPhysAddrLen;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac0;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac1;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac2;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac3;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac4;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac5;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac6;
         [MarshalAs(UnmanagedType.U1)]
         public byte mac7;
         [MarshalAs(UnmanagedType.U4)]
         public int dwAddr;
         [MarshalAs(UnmanagedType.U4)]
         public int dwType;
      }

      // Declare the GetIpNetTable function.
      [DllImport("IpHlpApi.dll")]
      [return: MarshalAs(UnmanagedType.U4)]
      static extern int GetIpNetTable(
         IntPtr pIpNetTable,
         [MarshalAs(UnmanagedType.U4)]
         ref int pdwSize,
         bool bOrder);

      [DllImport("IpHlpApi.dll", SetLastError = true, CharSet = CharSet.Auto)]
      internal static extern int FreeMibTable(IntPtr plpNetTable);

      // The insufficient buffer error.
      const int ERROR_INSUFFICIENT_BUFFER = 122;

      static void Main(string[] args)
      {
         // The number of bytes needed.
         int bytesNeeded = 0;

         // The result from the API call.
         int result = GetIpNetTable(IntPtr.Zero, ref bytesNeeded, false);

         // Call the function, expecting an insufficient buffer.
         if (result != ERROR_INSUFFICIENT_BUFFER)
         {
            // Throw an exception.
            throw new Win32Exception(result);
         }

         // Allocate the memory, do it in a try/finally block, to ensure
         // that it is released.
         IntPtr buffer = IntPtr.Zero;

         // Try/finally.
         try
         {
            // Allocate the memory.
            buffer = Marshal.AllocCoTaskMem(bytesNeeded);

            // Make the call again. If it did not succeed, then
            // raise an error.
            result = GetIpNetTable(buffer, ref bytesNeeded, false);

            // If the result is not 0 (no error), then throw an exception.
            if (result != 0)
            {
               // Throw an exception.
               throw new Win32Exception(result);
            }

            // Now we have the buffer, we have to marshal it. We can read
            // the first 4 bytes to get the length of the buffer.
            int entries = Marshal.ReadInt32(buffer);

            // Increment the memory pointer by the size of the int.
            IntPtr currentBuffer = new IntPtr(buffer.ToInt64() +
               Marshal.SizeOf(typeof(int)));

            // Allocate an array of entries.
            MIB_IPNETROW[] table = new MIB_IPNETROW[entries];

            // Cycle through the entries.
            for (int index = 0; index < entries; index++)
            {
               // Call PtrToStructure, getting the structure information.
               table[index] = (MIB_IPNETROW) Marshal.PtrToStructure(new
                  IntPtr(currentBuffer.ToInt64() + (index *
                  Marshal.SizeOf(typeof(MIB_IPNETROW)))), typeof(MIB_IPNETROW));
            }

            for (int index = 0; index < entries; index++)
            {
               MIB_IPNETROW row = table[index];
               IPAddress ip=new IPAddress(BitConverter.GetBytes(row.dwAddr));
               Console.Write("IP:"+ip.ToString()+"\t\tMAC:");

               Console.Write( row.mac0.ToString("X2") + '-');
               Console.Write( row.mac1.ToString("X2") + '-');
               Console.Write( row.mac2.ToString("X2") + '-');
               Console.Write( row.mac3.ToString("X2") + '-');
               Console.Write( row.mac4.ToString("X2") + '-');
               Console.WriteLine( row.mac5.ToString("X2"));

            }
         }
         finally
         {
            // Release the memory.
            FreeMibTable(buffer);
         }
      }
   }
}
使用系统;
使用System.Runtime.InteropServices;
使用系统组件模型;
Net系统;
命名空间GetIpNetTable
{
班级计划
{
//物理地址的最大数目。
常量int MAXLEN_PHYSADDR=8;
//定义MIB_IPNETROW结构。
[StructLayout(LayoutKind.Sequential)]
结构MIB_IPNETROW
{
[Marshallas(UnmanagedType.U4)]
公共指数;
[Marshallas(UnmanagedType.U4)]
公共交通部;
[Marshallas(UnmanagedType.U1)]
公共字节mac0;
[Marshallas(UnmanagedType.U1)]
公共字节mac1;
[Marshallas(UnmanagedType.U1)]
公共字节mac2;
[Marshallas(UnmanagedType.U1)]
公共字节mac3;
[Marshallas(UnmanagedType.U1)]
公共字节mac4;
[Marshallas(UnmanagedType.U1)]
公共字节mac5;
[Marshallas(UnmanagedType.U1)]
公共字节mac6;
[Marshallas(UnmanagedType.U1)]
公共字节mac7;
[Marshallas(UnmanagedType.U4)]
公共int dwAddr;
[Marshallas(UnmanagedType.U4)]
公共int-dwType;
}
//声明GetIpNetTable函数。
[DllImport(“iphlapi.dll”)]
[返回:Marshallas(UnmanagedType.U4)]
静态外部int GetIpNetTable(
IntPtr pIpNetTable,
[Marshallas(UnmanagedType.U4)]
ref int pdwSize,
布尔边界);
[DllImport(“iphlapi.dll”,SetLastError=true,CharSet=CharSet.Auto)]
内部静态外部int FreeMibTable(IntPtr plpNetTable);
//缓冲区不足错误。
常量int错误\u缓冲区不足=122;
静态void Main(字符串[]参数)
{
//所需的字节数。
int字节=0;
//API调用的结果。
int result=GetIpNetTable(IntPtr.Zero,ref bytesreded,false);
//调用函数,预期缓冲区不足。
如果(结果!=错误\u缓冲区不足)
{
//抛出异常。
抛出新的Win32Exception(结果);
}
//分配内存,在try/finally块中执行,以确保
//它被释放了。
IntPtr buffer=IntPtr.Zero;
//试试/最后。
尝试
{
//分配内存。
buffer=Marshal.allocTaskMem(需要字节);
//再打一次电话。如果没有成功,那么
//提出错误。
结果=GetIpNetTable(缓冲区,需要引用字节,false);
//如果结果不是0(无错误),则抛出异常。
如果(结果!=0)
{
//抛出异常。
抛出新的Win32Exception(结果);
}
//现在我们有了缓冲区,我们必须封送它。我们可以读取
//获取缓冲区长度的前4个字节。
int entries=Marshal.ReadInt32(缓冲区);
//按int的大小递增内存指针。
IntPtr currentBuffer=新的IntPtr(buffer.ToInt64()+
元帅SizeOf(typeof(int));
//分配一个条目数组。
MIB_IPNETROW[]表=新的MIB_IPNETROW[条目];
//循环浏览条目。
for(int index=0;index
希望您尝试从IP地址获取MAC地址,而不是相反

下面是一个男孩例子的链接:


我没有试过,让我们知道它是如何工作的。

在我的例子中,我想查看我网络上的所有ARP广播流量,以检测在我网络上广播冲突IP和MAC地址的设备。我发现“arp-a”轮询实现会导致过时的信息,这使得检测IP地址冲突特别具有挑战性。例如,有两个设备响应ARP请求,但由于一个响应总是在稍后到达,它会在“ARP-a”表中隐藏较早的响应

我曾经用cap创建一个捕获服务