我正在创建一个小型userControl库,用于在特定端口 [LAN] 上搜索活动服务器。
我正在使用[.Net 4.0] TCP套接字(IPV4)
所以首先我从我的电脑上获得IP地址:
string sHostName = Dns.GetHostName();
IPHostEntry ipE = Dns.GetHostByName(sHostName);
IPAddress[] IpA = ipE.AddressList;
然后我取IP地址的前3个部分,因此如果输入IPAddress =“192.168.1.X”,则返回“192.168.1”。 使用:
string scanIP = IpA[i].ToString().Substring(0, IpA[i].ToString().LastIndexOf(".")) + ".";
然后我搜索(例如:192.168.1。 1 - 至192.168.1。 254 )
我需要检查是否重复 scanIP 以避免多次扫描254个IP地址..因为阵列 IpA 可能会返回许多IP地址(无线网络, wirenetwork,bluetooth network..etc) 例如:我的无线IP是:192.168.1.25 我的以太网IP是:192.168.1.5
这是我的代码:
public ServerDiscovery()
{
InitializeComponent();
timer.Elapsed += timer_tick;
}
System.Timers.Timer timer = new System.Timers.Timer(5000);
List<SocketAsyncEventArgs> list = new List<SocketAsyncEventArgs>();
private void btnRefresh_Click(object sender, EventArgs e)
{
listServer.Items.Clear();
timer.Start();
//getting array of IP Addresses.
IPAddress[] IpA = Dns.GetHostByName(Dns.GetHostName()).AddressList;
for(int j = 0; j < IpA.Length; j++)
{
if (IpA[j].AddressFamily == AddressFamily.InterNetwork) //checking if IP is version 4
{
string scanIP = IpA[j].ToString().Substring(0, IpA[j].ToString().LastIndexOf(".")) + ".";
if (some condition)// i need a condition to ensure that dublicated scanIP doesn't scan again
{
Parallel.For(1, 255, (i, loopState) =>
{
Scan(scanIP + i);
});
}
}
}
}
private void Scan(string ipAdd)
{
Socket s = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
SocketAsyncEventArgs e = new SocketAsyncEventArgs();
e.RemoteEndPoint = new IPEndPoint(IPAddress.Parse(ipAdd), 9990);
e.UserToken = s;
e.Completed += new EventHandler<SocketAsyncEventArgs>(e_Completed);
list.Add(e);
s.ConnectAsync(e);
}
private void e_Completed(object sender, SocketAsyncEventArgs e)
{
if (e.ConnectSocket != null)
{
StreamReader sr = new StreamReader(new NetworkStream(e.ConnectSocket));
ListViewItem item = new ListViewItem();
string[] cmd = sr.ReadLine().Split('<');
item.Text = cmd[0];
item.SubItems.Add(((IPEndPoint)e.RemoteEndPoint).Address.ToString());
item.SubItems.Add(cmd[1]);
AddServer(item);
}
}
delegate void AddItem(ListViewItem item);
private void AddServer(ListViewItem item)
{
if (InvokeRequired)
{
Invoke(new AddItem(AddServer), item);
return;
}
listServer.Items.Add(item);
}
private void timer_tick(object sender, EventArgs e)
{
timer.Stop();
foreach (var s in list)
{
((Socket)s.UserToken).Dispose();
}
}
答案 0 :(得分:3)
最简单的方法是使用Distinct()
LINQ扩展方法:
IpA = IpA.Distinct().ToArray();
for(int j = 0; j < IpA.Length; j++)
{
...
您还可以使用HashSet检查重复项。我认为这是一个更快的解决方案:
HashSet<IPAddress> set = new HashSet<IPAddress>(IpA);
for(int j = 0; j < IpA.Length; j++)
{
if( set.Contains(IpA[j]) )
continue;
set.Add(IpA[j]);
if (IpA[j].AddressFamily == AddressFamily.InterNetwork) //checking if IP is version 4
{
string scanIP = IpA[j].ToString().Substring(0, IpA[j].ToString().LastIndexOf(".")) + ".";
Parallel.For(1, 255, (i, loopState) =>
{
Scan(scanIP + i);
});
}
}
答案 1 :(得分:2)
您可以使用HashSet来保留已使用过的地址。
HashSet<string> usedIP = new HashSet<string>();
用
替换SomeConditionif (!usedIp.Contains(scanIP))
{
usedIP.Add(scanIP);
Parallel.For(1, 255, (i, loopState) =>
{
Scan(scanIP + i);
});
}
答案 2 :(得分:0)
将scanIP添加到字典中,并确保它尚未添加。
Dictionary<string, string> scannedIPs = new Dictionary<string, string>();
...
string scanIP = IpA[j].ToString().Substring(0, IpA[j].ToString().LastIndexOf(".")) + ".";
if (!scannedIPs.ContainsKey(scanIP))
{
scannedIPs.Add(scanIP, scanIP);