如果你使用过P2P终结者或者类似的软件,你会发觉只要一打开就可以看到局域网内部的所有机器,而有时候我们正好有这样的需求,那我们应该怎么样用去获得局域网的所有机器呢?如果你到百度或者GOOGLE上面进行搜索你会发现,网上大致都是采用轮询的方法,让你把所有机器都扫描一遍,如果有反应则表示主机存在,但是这种办法并不可取,不仅耗资源,而且耗时间,即使你单独开一个线程去跑,估计半小时都没有任何结果。网上有人提出更加可笑的办法,说开多一些线程去检测。要知道,线程可不是省油的灯,再说,采用轮询的办法每遇到一台主机不存在就会抛出一个异常,而且该类异常一般都是超时无响应才抛出的,使用异常处理的方式来处理问题将会严重影响应用程序的性能。
这里将介绍如何利用巧妙的方式来获取局域网内所有机器:
// 1.先调用系统API判断网络是否处于连接状态
[DllImport("wininet.dll")]
private static extern bool InternetGetConnectedState(out int connectionDescription, int reservedValue);
public static bool IsLocalConnection()
{
int connectionDescription = 0;
return InternetGetConnectedState(out connectionDescription, 0);
}
//2.再调用底层硬件获取本地网关地址信息
static string GetGateWayAddress()
{
ManagementObjectCollection moc = new ManagementClass("Win32_NetworkAdapterConfiguration").GetInstances();
foreach (ManagementObject mo in moc)
{
foreach (PropertyData p in mo.Properties)
{
if (p.Name.Equals("DefaultIPGateway") && (p.Value != null))
{
string[] strs = p.Value as string[];
string[] CS$6$0004 = strs;
int CS$7$0005 = 0;
while (CS$7$0005 < CS$6$0004.Length)
{
return CS$6$0004[CS$7$0005];
}
}
}
}
return "";
}
//3.分别向本地网关内机器发送ICMP数据包
bool Pinging(string addr, int id, uint taskid)
{
try
{
this.m_id = id;
this.m_taskid = taskid;
byte[] byReq = this.FillEchoReq();
IPEndPoint lep = new IPEndPoint(IPAddress.Parse(addr), 0);
this.socket.SendTo(byReq, lep);
}
catch (Exception e)
{
Console.WriteLine("Send error:" + e.ToString());
return false;
}
return true;
}
//4.定义本地机器节点信息类
public class LocalMachine
{
// Fields
private string machineIP;
private string machineMAC;
private string machineName;
// Methods
public LocalMachine();
// Properties
public string MachineIP { get; set; }
public string MachineMAC { get; set; }
public string MachineName { get; set; }
}
//5.根据arp原理,最后通过以下方式读取arp列表节点信息,其实这里还可以IMCP包响应来获取主机响应,
//不过我个人认为用直接读取列表的方式更加快速有效。
static ArrayList GetAllLocalMachines()
{
Process p = new Process();
p.StartInfo.FileName = "cmd.exe";
p.StartInfo.UseShellExecute = false;
p.StartInfo.RedirectStandardInput = true;
p.StartInfo.RedirectStandardOutput = true;
p.StartInfo.RedirectStandardError = true;
p.StartInfo.CreateNoWindow = true;
p.Start();
p.StandardInput.WriteLine("arp -a");
p.StandardInput.WriteLine("exit");
ArrayList list = new ArrayList();
StreamReader reader = p.StandardOutput;
string IPHead = Dns.GetHostByName(Dns.GetHostName()).AddressList[0].ToString().Substring(0, 3);
for (string line = reader.ReadLine(); line != null; line = reader.ReadLine())
{
line = line.Trim();
if (line.StartsWith(IPHead) && (line.IndexOf("dynamic") != -1))
{
string IP = line.Substring(0, 15).Trim();
string Mac = line.Substring(line.IndexOf("-") - 2, 0x11).Trim();
LocalMachine localMachine = new LocalMachine();
localMachine.MachineIP = IP;
localMachine.MachineMAC = Mac;
localMachine.MachineName = "";
list.Add(localMachine);
}
}
return list;
}
//6.最后,你还可以通过以下方法来获取网卡的MAC地址信息
[DllImport("NETAPI32.DLL")]
public static extern char Netbios(ref MACAddress.NCB ncb);
public string GetMacAddress()
{
string addr = "";
try
{
NCB Ncb = new NCB();
Ncb.ncb_command = 0x37;
int cb = Marshal.SizeOf(typeof(LANA_ENUM));
Ncb.ncb_buffer = Marshal.AllocHGlobal(cb);
Ncb.ncb_length = (ushort) cb;
char uRetCode = Win32API.Netbios(ref Ncb);
LANA_ENUM lenum = (LANA_ENUM) Marshal.PtrToStructure(Ncb.ncb_buffer, typeof(LANA_ENUM));
Marshal.FreeHGlobal(Ncb.ncb_buffer);
if (uRetCode != '1')
{}
}
catch
{}
}
}