如何高效实现扫描局域网IP、主机名、MAC和端口
近几年工作经常使用RFID识读器,智能家居网关,温湿度传感器、串口服务器、视频编码器等,一般是有串口和网口,由于现场原因一般较少使用串口,大多使用网口。连接方法是IP地址和端口,有的设备带搜索软件,有的就不带。经常测试环境和现场来回用,难免记混。同时使用网上的一些端口扫描工具,发现有一个问题,就是在进行扫描的时候,不怎么好用,比如,扫描器一般都是带端口扫描的,但是,我仅仅只用扫描设备在不在线,不用扫描端口,但是有一些设备是只要扫描端口,这个怎么更方便更好用呢?
同时在扫描成功之后,需要通过TCP连接到设备,发数据,看数据能不能正常相应,或者设备状态对不对,这是不是又要单独开另外一个软件?使用起来,比较繁琐,因此就根据自己的需要制作了一个小工具,用来扫描IP和端口。
扫描速度一定要快,成功和不成功区分要明显,好了,先上图:
一、主要IP扫描方法:
(1)、IP地址匹配,看IP设置是否正确
//匹配正确的IP地址
Regex rgx = new Regex(@"^([1-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])(\.([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])){3}$"); if (rgx.IsMatch(IPStarttextBox.Text) && rgx.IsMatch(IPStoptextBox.Text))//匹配正确IP
{
if (scan_type == 1)
{
int startIp = Int32.Parse(IPStarttextBox.Text.Split('.')[3]);
int endIp = Int32.Parse(IPStoptextBox.Text.Split('.')[3]);
progressBar.Minimum = startIp;
progressBar.Maximum = endIp;
}
else
{
if (PortStarttextBox.Text == "")
{
MessageBox.Show("请输入端口号!");
}
else
{
portStart = Int32.Parse(PortStarttextBox.Text);
portEnd = Int32.Parse(PortEndtextBox.Text);
progressBar.Minimum = portStart;
progressBar.Maximum = portEnd;
} if (portEnd < portStart)
{
MessageBox.Show("请填写正确端口范围");
return;
}
} }
else
{
MessageBox.Show("请填写正确IP");
return;
}
(2)、获取主机名
public static string GetHostEntry(object ip)
{
string hostname = "-";
try
{
IPHostEntry host = Dns.GetHostEntry(ip.ToString());
hostname = host.HostName;
}
catch(Exception ex)
{
Console.WriteLine(ex.ToString());
}
return hostname;
}
(3)、获取MAC地址
[DllImport("ws2_32.dll")]
private static extern int inet_addr(string cp);
[DllImport("IPHLPAPI.dll")]
private static extern int SendARP(Int32 DestIP, Int32 SrcIP, ref Int64 pMacAddr, ref Int32 PhyAddrLen); private string GetMacAddress(string hostip)//获取远程IP(不能跨网段)的MAC地址
{
string Mac = "";
try
{
Int32 ldest = inet_addr(hostip); //将IP地址从 点数格式转换成无符号长整型
Int64 macinfo = new Int64();
Int32 len = ;
SendARP(ldest, , ref macinfo, ref len);
string TmpMac = Convert.ToString(macinfo, ).PadLeft(, '');//转换成16进制 注意有些没有十二位
Mac = TmpMac.Substring(, ).ToUpper();//
for (int i = ; i < TmpMac.Length; i = i + )
{
Mac = TmpMac.Substring(i, ).ToUpper() + "-" + Mac;
}
}
catch (Exception Mye)
{
Mac = "获取远程主机的MAC错误:" + Mye.Message;
}
return Mac;
}
(3)、主要准备工作做好了,接下来,进入主题,如何通过多线程判断IP是否在线。
Thread waitT = new Thread(new ThreadStart(ipwait));
waitT.Start();//等待所有线程执行完毕在写入textbox中 public void ipwait()
{
int startIp = Int32.Parse(IPStarttextBox.Text.Split('.')[]);
int endIp = Int32.Parse(IPStoptextBox.Text.Split('.')[]);
ListViewItem lvi; string ip = IPStarttextBox.Text.Split('.')[] + "." + IPStarttextBox.Text.Split('.')[] + "." + IPStarttextBox.Text.Split('.')[] + ".";
String ipaddr = "";
int timeout = int.Parse(TimeOuttextBox.Text); string mac = ""; progressBar.Minimum = startIp;
progressBar.Maximum = endIp;
try
{ for (int q = startIp; q <= endIp && isrun == true; q++)
{ //---------------------ping
ipaddr = ip + q;
lvi = new ListViewItem();
Ping ping = new Ping();
PingReply reply = ping.Send(IPAddress.Parse(ipaddr), overTime); if (reply.Status == IPStatus.Success)
{ ResuleTextBox.Text += ipaddr + " Ping时间 " + reply.RoundtripTime + " 毫秒\n";
mac= GetMacAddress(ipaddr);
// IPHostEntry host = Dns.GetHostEntry(ipaddr);
// IPHostEntry host = Dns.GetHostByAddress(ipaddr);
// ResuleTextBox.Text += "主机名为 " + host.HostName + "\n";
// ScanResultlistBox.Items.Add(ipaddr + " " + mac + " 在线 ");
lvi.BackColor = Color.LightSeaGreen;
lvi.Text = ipaddr;
Task<string> t1 = new Task<string>(GetHostEntry,ipaddr);
t1.Start();
t1.Wait(timeout); //设置获取主机名超时时间
if(t1.IsCompleted)
{
lvi.SubItems.Add(t1.Result);
}
else
{
lvi.SubItems.Add("-");
}
lvi.SubItems.Add(mac);
lvi.SubItems.Add("在线"); }
else
{
lvi.BackColor = Color.Pink;
lvi.Text = ipaddr;
lvi.SubItems.Add("-");
lvi.SubItems.Add("-");
lvi.SubItems.Add("不在线");
} progressBar.Value = q;
progressBar.Update();
ListViewUpdate(lvi);
ResuleTextBox.SelectionStart = ResuleTextBox.Text.Length -;
ResuleTextBox.ScrollToCaret();
}
ScanResultlistBox.Items[ScanResultlistBox.Items.Count - ].EnsureVisible(); }
catch(Exception ex)
{
Console.WriteLine(ex.ToString());
isrun = false;
progressBar.Value = progressBar.Minimum;
startScanbutton.Enabled = true;
stopScanbutton.Enabled = false;
ResuleTextBox.Text += ex.ToString();
}
startScanbutton.Enabled = true;
stopScanbutton.Enabled = false;
isrun = false;
progressBar.Value = progressBar.Maximum; }
说明一下:在我获取主机名的过程中,发现如果不对获取主机名设置一个超时时间,那么将会出现一个问题,程序将会卡在获取主机名的这个阶段,无法向下继续执行,因为,为了高效的执行,给获取主机名设置一个超时时间,将会大大提高程序的执行效率。
二、主要端口状态获取方法
Thread waitT = new Thread(new ThreadStart(portwait));
waitT.Start();//等待所有线程执行完毕在写入textbox中
public void portwait()
{
int startIp = Int32.Parse(IPStarttextBox.Text.Split('.')[]);
int endIp = Int32.Parse(IPStoptextBox.Text.Split('.')[]); string ip = IPStarttextBox.Text.Split('.')[] + "." + IPStarttextBox.Text.Split('.')[] + "." + IPStarttextBox.Text.Split('.')[] + ".";
String mac = "" , ipaddr ="";
ListViewItem lvi;
String host;
int timeout = int.Parse(TimeOuttextBox.Text);
for (int q = startIp; q <= endIp && isrun == true; q++)
{
ResuleTextBox.Text += (ip + q) + " IP地址扫描中---\n";
//---------------------ping
Ping ping = new Ping(); PingReply reply = ping.Send(IPAddress.Parse(ip + q), overTime);
ipaddr = ip + q;
if (reply.Status == IPStatus.Success)
{
ResuleTextBox.Text +=ipaddr + " Ping时间" + reply.RoundtripTime + "毫秒\n";
mac = GetMacAddress(ipaddr);
Task<string> t1 = new Task<string>(GetHostEntry, ipaddr);
t1.Start();
t1.Wait(timeout);
if (t1.IsCompleted)
{
host = t1.Result.ToString();
}
else
{
host = "-";
}
ResuleTextBox.Text += "MAC地址为 " + mac + "\n";
}
else
{
// ScanResultlistBox.Items.Add(ipaddr + " " + "-" + " " + "-" + " " + " - 不可达\n");
lvi = new ListViewItem();
lvi.BackColor = Color.Pink;
lvi.Text = ipaddr;
lvi.SubItems.Add("-");
lvi.SubItems.Add("-");
lvi.SubItems.Add("不可达");
ResuleTextBox.Text += ipaddr + "不可达\n";
ListViewUpdate(lvi);
continue;
} //---------------------end
Thread[] tharr;
if (numThread < (portEnd - portStart + ))
{
tharr = new Thread[portEnd - portStart + ];
}
else
{
tharr = new Thread[numThread];
}
str = new List<string>();
for (int i = portStart; (i <= portEnd) && (isrun ==true); i++)
{
Thread thread = new Thread(new ParameterizedThreadStart(Scan));
thread.Start(new IPEndPoint(IPAddress.Parse(ip + q), i));//每扫描一个端口创建一个线程 ResuleTextBox.Text += i + " 端口扫描中---\n"; tharr[i - portStart] = thread;
progressBar.Value = i; string s = State(i); lvi = new ListViewItem(); if (s == "Open")
{
lvi.BackColor = Color.LightSeaGreen;
}
else
{
lvi.BackColor = Color.Silver;
} lvi.Text = ipaddr + ":" + i.ToString();
lvi.SubItems.Add(host);
lvi.SubItems.Add(mac); if (checkBox1.Checked)
{
if (s == "Open")
{ // ScanResultlistBox.Items.Add(ip + q + " " + mac +" "+ i + " " + s + " " + " " + Service(i)); lvi.SubItems.Add(i.ToString());
lvi.SubItems.Add(s);
lvi.SubItems.Add(Service(i));
}
else
{
lvi = null;
}
}
else
{
if (s == "Open")
{ // ScanResultlistBox.Items.Add(ip + q + " " + mac + " " + i + " " + s + " " + " " + Service(i)); lvi.SubItems.Add(i.ToString());
lvi.SubItems.Add(s);
lvi.SubItems.Add(Service(i));
}
else
{
// ScanResultlistBox.Items.Add(ip + q + " " + mac + " " + i + " " + s + " " + " "); lvi.SubItems.Add(i.ToString());
lvi.SubItems.Add(s);
lvi.SubItems.Add("");
}
}
if (lvi != null)
{
ListViewUpdate(lvi);
}
} bool iscon = true;//第一个线程等待时间
for (int i = ; i < tharr.Length; i++)
{
if (tharr[i] == null)
continue;
while (tharr[i].IsAlive && iscon)//端口超时设置时间(目前200毫秒),一直等待此ip所有线程执行完毕才扫描下个ip
{
Thread.Sleep();
iscon = false;//第一个线程给200ms等待时间,其他线程由于同步执行的,所以没等待时间了,如果线程还没执行完,说明此端口不可达。
}
}
str.Sort();
ResuleTextBox.Text += "开放端口: ";
for (int k = ; k < str.Count; k++)
ResuleTextBox.Text += str[k] + " ";
ResuleTextBox.Text += "\n";
}
ScanResultlistBox.Items[ScanResultlistBox.Items.Count - ].EnsureVisible();
if (isrun == true)
{
MessageBox.Show("扫描完成");
progressBar.Value = progressBar.Minimum;
startScanbutton.Enabled = true;
stopScanbutton.Enabled = false;
isrun = false;
}
else
{
MessageBox.Show("扫描终止");
startScanbutton.Enabled = true;
stopScanbutton.Enabled = false;
progressBar.Value = progressBar.Minimum;
isrun = false;
} }
public string State(int i)
{
str.Sort();
for (int k = ; k < str.Count; k++)
{
string s = str[k];
if (Convert.ToString(i) == s)
return "Open";
}
return "Closed";
}
public string Service(int i)
{
switch (i)
{ case :
return "HTTP协议代理服务";
case :
return "DCE endpoint resolutionnetbios-ns";
case :
return "安全服务";
case :
return "NetSpy.698(YAI)"; case :
return "HTTP协议代理服务"; case :
return "HTTP协议代理服务"; case :
return "HTTP协议代理服务"; case :
return "HTTP协议代理服务"; case :
return "SOCKS代理协议服务"; case :
return "FTP(文件传输)协议代理服务"; case :
return "Telnet(远程登录)协议代理服务"; case :
return "HTTPS协议代理服务"; case :
return "TFTP协议代理服务"; case :
return "SSH、SCP、端口重定向协议代理服务"; case :
return "SMTP协议代理服务"; case :
return "POP3协议代理服务";
default:
return "Unknow Servies"; }
} public void Scan(object Point)
{
IPEndPoint IPPoint = (IPEndPoint)Point;
try
{
TcpClient tcp = new TcpClient();
tcp.Connect(IPPoint);
if (tcp.Connected)
str.Add(Convert.ToString(IPPoint.Port));
}
catch (Exception ex)
{
Console.WriteLine(ex.ToString());
}
}
好了,大家最后,可能都会说,我不想设计,也不想开发,我想要现成的怎么办?当然了,最后附上程序,大家可以实际进行测试使用。
工具链接:http://download.csdn.net/detail/dwx1005526886/9808540
如何高效实现扫描局域网IP、主机名、MAC和端口的更多相关文章
- python获取主机IP,主机名
获取主机内网,外网IP,主机名 代码如下: #!/usr/bin/env python #-*- coding:utf-8 -*- # author:Zeng Xianhe import socket ...
- 调用python-nmap实现扫描局域网存活主机
使用环境:Raspberry 3b+ +netifaces+python-nmap+nmap 调用netifaces自动获取ip地址: def get_gateways(): return netif ...
- linux系统的初化始配置 IP 主机名 防火墙 selinux
本次内容包括Linux:ip.主机名.关闭firewalld与selinux 开机临时生效和永久生效 ------------------------------------------------- ...
- java 代码,练习ip,主机名的获取方法。InetAddress类
package clientFrame; import java.io.IOException; import java.net.*; public class tai { public static ...
- 用arp-scan扫描局域网IP地址
1,在安装之前需要安装yum install -y libpcap libpcap-devel如果没有安装yum工具需要用rpm安装如下软件包[root@oradba arp-scan-1.8]# y ...
- c# 扫描局域网IP列表的几种方法
很多软件都有获知局域网在线计算机IP的功能,但是在.net怎么实现呢,有好多方法, 下面我给大家介绍几种,供大家参考. 1.微软社区上介绍了使用Active Directory 来遍历局域网 利用Di ...
- 扫描局域网ip的shell
# vim /mysh/ipscan.sh #!/bin/bash # scan the local alive ipaddress # -- if [ -f $filename ];then ech ...
- 扫描局域网ip存活
#!/bin/bash # #******************************************************************** #encoding -*-utf ...
- Linux学习(一):linux更改ip地址命令_更改DNS_更改默认网关_更改子网掩码_主机名
如何使用命令来更改linux的IP .DNS .子网掩码,在虚拟机(vitrualBox)上添加一个Linux的虚拟机; 安装方法:http://pan.baidu.com/s/1sjJPhP7 安装 ...
随机推荐
- iOS 创建一个在退出登录时可以销毁的单例
一.单例简介 单例模式是在软件开发中经常用的一种模式.单例模式通俗的理解是,在整个软件生命周期内,一个类只能有一个实例对象存在. 二.遇到的问题 在平时开发使用单例的过程中,有时候会有这样的需求,在用 ...
- 每天一个Linux命令(13)--less命令
less 工具也是对文件或其它输出进行分页显示的工具,应该说是Linux正统馋看文件内容的工具,功能极其强大.less 的用法比起 more 更加有弹性. 在 more 的时候,我们没有办法向前面 ...
- 转 jquery怎么在header中设置请求信息
jquery是js的类库,js本身不能操作header,因为js是在浏览器加载页面过程中才开始执行的 header需要服务器端执行操作 如果是ajax,是可以设置header $.ajax({ url ...
- 每一个程序员都应该知道的高并发处理技巧、创业公司如何解决高并发问题、互联网高并发问题解决思路、caoz大神多年经验总结分享
本文来源于caoz梦呓公众号高并发专辑,以图形化.松耦合的方式,对互联网高并发问题做了详细解读与分析,"技术在短期内被高估,而在长期中又被低估",而不同的场景和人员成本又导致了巨头 ...
- SpringCache缓存初探
body,table tr { background-color: #fff } table tr td,table tr th { border: 1px solid #ccc; text-alig ...
- ThinkPHP框架知识的注意点
ThinkPHP框架 访问入口文件后在application文件夹中会出现一些文件夹,其中的home文件夹是前端模块,也可以在application文件夹中新建文件夹.home文件夹模块中Conf文件 ...
- jquery事件与绑定事件
1.首先,我们来看一下经常使用的添加事件的方式: <input type="button" id="btn" value="click me!& ...
- KoaHub.js -- 基于 Koa.js 平台的 Node.js web 快速开发框架之koahub-yilianyun
koahub-yilianyun 微信易联云打印机接口 koahub-yilianyun易联云打印机node接口 Installation $ npm install koahub-yilianyun ...
- 1002: [FJOI2007]轮状病毒
1002: [FJOI2007]轮状病毒 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 2609 Solved: 1450[Submit][Statu ...
- ios录音Demo
<AudioToolbox/AudioToolbox.h> :这个库是C的接口,偏向于底层,主要用于在线流媒体的播放 <AVFoundation/AVFoundation.h> ...