代码之家  ›  专栏  ›  技术社区  ›  BerggreenDK

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

  •  25
  • BerggreenDK  · 技术社区  · 15 年前

    我正试图弄清楚局域网中哪些设备是在线的和离线的。我已经看到许多程序做了一种图形网络概述,显示局域网IP和MAC地址。我想知道这些(arp)是否以及如何实现?可以从c/.net中提取信息?

    任何示例代码片段/链接都将不胜感激。

    3 回复  |  直到 7 年前
        1
  •  32
  •   DrBB    11 年前

    如果您知道外面有哪些设备,可以使用 Ping Class .这将允许您至少填充ARP表。如果必须的话,可以执行arp-a并解析输出。这里还有一个链接,显示如何调用pinvoke GetIpNetTable . 我已经包括了下面关于ping类的例子,以及如何使用getipnettable访问arp表。

    这是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);
                }
            }
        }
    }
    

    这是getipnetable的一个示例。

    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);
             }
          }
       }
    }
    
        2
  •  1
  •   jonathanpeppers    15 年前

    希望您正在尝试从IP地址中获取MAC地址,而不是反过来。

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

    ARP Resolver

    我没试过,告诉我们它是怎么工作的。

        3
  •  -4
  •   Himanshu    7 年前

    谷歌搜索“fingbox”。好像你在试着检测直觉?

    这是一个完全合法的入侵者检测设备,很高兴知道谁使用您的WiFi在哪些端口上传输。有时它还显示mac地址和can ping。有很多其他功能。