- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章在.NET中扫描局域网服务的实现方法由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
在最近负责的项目中,需要实现这样一个需求:在客户端程序中,扫描当前机器所在网段中的所有机器上是否有某服务启动,并把所有已经启动服务的机器列出来,供用户选择,连接哪个服务。注意:这里所说的服务事实上就是在一个固定的端口监听基于 TCP 协议的请求的程序或者服务(如 WCF 服务).
要实现这样的功能,核心的一点就是在得到当前机器同网段的所有机器的 IP 后,对每一 IP 发生 TCP 连接请求,如果请求超时或者出现其它异常,则认为没有服务,反之,如果能够正常连接,则认为服务正常.
经过基本功能的实现以及后续的重构之后,就有了本文以下的代码:一个接口和具体实现的类。需要说明的是:在下面的代码中,先提到接口,再提到具体类;而在开发过程中,则是首先创建了类,然后才提取了接口。之所以要提取接口,原因有二:一是可以支持 IoC控制反转;二是将来如果其它的同类需求,可以其于此接口实现新功能.
1、接口定义 。
先看来一下接口:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
/// <
summary
>
/// 扫描服务
/// </
summary
>
public interface IServerScanner
{
/// <
summary
>
/// 扫描完成
/// </
summary
>
event EventHandler<
List
<ConnectionResult>> OnScanComplete;
/// <
summary
>
/// 报告扫描进度
/// </
summary
>
event EventHandler<
ScanProgressEventArgs
> OnScanProgressChanged;
/// <
summary
>
/// 扫描端口
/// </
summary
>
int ScanPort { get; set; }
/// <
summary
>
/// 单次连接超时时长
/// </
summary
>
TimeSpan Timeout { get; set; }
/// <
summary
>
/// 返回指定的IP与端口是否能够连接上
/// </
summary
>
/// <
param
name
=
"ipAddress"
></
param
>
/// <
param
name
=
"port"
></
param
>
/// <
returns
></
returns
>
bool IsConnected(IPAddress ipAddress, int port);
/// <
summary
>
/// 返回指定的IP与端口是否能够连接上
/// </
summary
>
/// <
param
name
=
"ip"
></
param
>
/// <
param
name
=
"port"
></
param
>
/// <
returns
></
returns
>
bool IsConnected(string ip, int port);
/// <
summary
>
/// 开始扫描
/// </
summary
>
void StartScan();
}
|
其中 Timeout 属性是控制每次连接请求超时的时长.
2、具体实现 。
再来看一下具体实现类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
|
/// <
summary
>
/// 扫描结果
/// </
summary
>
public class ConnectionResult
{
/// <
summary
>
/// IPAddress 地址
/// </
summary
>
public IPAddress Address { get; set; }
/// <
summary
>
/// 是否可连接上
/// </
summary
>
public bool CanConnected { get; set; }
}
/// <
summary
>
/// 扫描完成事件参数
/// </
summary
>
public class ScanCompleteEventArgs
{
/// <
summary
>
/// 结果集合
/// </
summary
>
public List<
ConnectionResult
> Reslut { get; set; }
}
/// <
summary
>
/// 扫描进度事件参数
/// </
summary
>
public class ScanProgressEventArgs
{
/// <
summary
>
/// 进度百分比
/// </
summary
>
public int Percent { get; set; }
}
/// <
summary
>
/// 扫描局域网中的服务
/// </
summary
>
public class ServerScanner : IServerScanner
{
/// <
summary
>
/// 同一网段内 IP 地址的数量
/// </
summary
>
private const int SegmentIpMaxCount = 255;
private DateTimeOffset _endTime;
private object _locker = new object();
private SynchronizationContext _originalContext = SynchronizationContext.Current;
private List<
ConnectionResult
> _resultList = new List<
ConnectionResult
>();
private DateTimeOffset _startTime;
/// <
summary
>
/// 记录调用/完成委托的数量
/// </
summary
>
private int _totalCount = 0;
public ServerScanner()
{
Timeout = TimeSpan.FromSeconds(2);
}
/// <
summary
>
/// 当扫描完成时,触发此事件
/// </
summary
>
public event EventHandler<
List
<ConnectionResult>> OnScanComplete;
/// <
summary
>
/// 当扫描进度发生更改时,触发此事件
/// </
summary
>
public event EventHandler<
ScanProgressEventArgs
> OnScanProgressChanged;
/// <
summary
>
/// 扫描端口
/// </
summary
>
public int ScanPort { get; set; }
/// <
summary
>
/// 单次请求的超时时长,默认为2秒
/// </
summary
>
public TimeSpan Timeout { get; set; }
/// <
summary
>
/// 使用 TcpClient 测试是否可以连上指定的 IP 与 Port
/// </
summary
>
/// <
param
name
=
"ipAddress"
></
param
>
/// <
param
name
=
"port"
></
param
>
/// <
returns
></
returns
>
public bool IsConnected(IPAddress ipAddress, int port)
{
var result = TestConnection(ipAddress, port);
return result.CanConnected;
}
/// <
summary
>
/// 使用 TcpClient 测试是否可以连上指定的 IP 与 Port
/// </
summary
>
/// <
param
name
=
"ip"
></
param
>
/// <
param
name
=
"port"
></
param
>
/// <
returns
></
returns
>
public bool IsConnected(string ip, int port)
{
IPAddress ipAddress;
if (IPAddress.TryParse(ip, out ipAddress))
{
return IsConnected(ipAddress, port);
}
else
{
throw new ArgumentException("IP 地址格式不正确");
}
}
/// <
summary
>
/// 开始扫描当前网段
/// </
summary
>
public void StartScan()
{
if (ScanPort == 0)
{
throw new InvalidOperationException("必须指定扫描的端口 ScanPort");
}
// 清除可能存在的数据
_resultList.Clear();
_totalCount = 0;
_startTime = DateTimeOffset.Now;
// 得到本网段的 IP
var ipList = GetAllRemoteIPList();
// 生成委托列表
List<
Func
<IPAddress, int, ConnectionResult>> funcs = new List<
Func
<IPAddress, int, ConnectionResult>>();
for (int i = 0; i <
SegmentIpMaxCount
; i++)
{
var
tmpF
=
new
Func<IPAddress, int, ConnectionResult>(TestConnection);
funcs.Add(tmpF);
}
// 异步调用每个委托
for (int i = 0; i <
SegmentIpMaxCount
; i++)
{
funcs[i].BeginInvoke(ipList[i], ScanPort, OnComplete, funcs[i]);
_totalCount += 1;
}
}
/// <summary>
/// 得到本网段的所有 IP
/// </
summary
>
/// <
returns
></
returns
>
private List<
IPAddress
> GetAllRemoteIPList()
{
var localName = Dns.GetHostName();
var localIPEntry = Dns.GetHostEntry(localName);
List<
IPAddress
> ipList = new List<
IPAddress
>();
IPAddress localInterIP = localIPEntry.AddressList.FirstOrDefault(m => m.AddressFamily == AddressFamily.InterNetwork);
if (localInterIP == null)
{
throw new InvalidOperationException("当前计算机不存在内网 IP");
}
var localInterIPBytes = localInterIP.GetAddressBytes();
for (int i = 1; i <= SegmentIpMaxCount; i++)
{
// 对末位进行替换
localInterIPBytes[3] = (byte)i;
ipList.Add(new IPAddress(localInterIPBytes));
}
return ipList;
}
private void OnComplete(IAsyncResult ar)
{
var state = ar.AsyncState as Func<
IPAddress
, int, ConnectionResult>;
var result = state.EndInvoke(ar);
lock (_locker)
{
// 添加到结果中
_resultList.Add(result);
// 报告进度
_totalCount -= 1;
var percent = (SegmentIpMaxCount - _totalCount) * 100 / SegmentIpMaxCount;
if (SynchronizationContext.Current == _originalContext)
{
OnScanProgressChanged?.Invoke(this, new ScanProgressEventArgs { Percent = percent });
}
else
{
_originalContext.Post(conState =>
{
OnScanProgressChanged?.Invoke(this, new ScanProgressEventArgs { Percent = percent });
}, null);
}
if (_totalCount == 0)
{
// 通过事件抛出结果
if (SynchronizationContext.Current == _originalContext)
{
OnScanComplete?.Invoke(this, _resultList);
}
else
{
_originalContext.Post(conState =>
{
OnScanComplete?.Invoke(this, _resultList);
}, null);
}
// 计算耗时
Debug.WriteLine("Compete");
_endTime = DateTimeOffset.Now;
Debug.WriteLine($"Duration: {_endTime - _startTime}");
}
}
}
/// <
summary
>
/// 测试是否可以连接到
/// </
summary
>
/// <
param
name
=
"address"
></
param
>
/// <
param
name
=
"port"
></
param
>
/// <
returns
></
returns
>
private ConnectionResult TestConnection(IPAddress address, int port)
{
TcpClient c = new TcpClient();
ConnectionResult result = new ConnectionResult();
result.Address = address;
using (TcpClient tcp = new TcpClient())
{
IAsyncResult ar = tcp.BeginConnect(address, port, null, null);
WaitHandle wh = ar.AsyncWaitHandle;
try
{
if (!ar.AsyncWaitHandle.WaitOne(Timeout, false))
{
tcp.Close();
}
else
{
tcp.EndConnect(ar);
result.CanConnected = true;
}
}
catch
{
}
finally
{
wh.Close();
}
}
return result;
}
}
ServerScanner
|
以上代码中注释基本上已经比较详细,这里再简单提几个点:
TestConnection 函数实了现核心功能,即请求给定的 IP 和端口,并返回结果;其中通过调用 IAsyncResult.AsyncWaitHandle 属性的 WaitOne 方法来实现对超时的控制; 。
StartScan 方法中,在得到 IP 列表后,通过生成委托列表并异步调用这些委托来实现整个方法是异步的,不会阻塞 UI,而这些委托指向的方法就是 TestConnection 函数; 。
使用同步上下文 SynchronizationContext,可以保证调用方在原来的线程(通常是 UI 线程)上处理进度更新事件或扫描完成事件; 。
对于每个委托异步完成后,会执行回调方法 OnComplete,在它里面,对全局变量的操作需要加锁,以保证线程安全.
3、如何使用 。
最后来看一下如何使用,非常简单:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
private void View_Loaded()
{
// 在界面 Load 事件中添加以下代码
ServerScanner.OnScanComplete += ServerScanner_OnScanComplete;
ServerScanner.OnScanProgressChanged += ServerScanner_OnScanProgressChanged;
// 扫描的端口号
ServerScanner.ScanPort = 7890;
}
private void StartScan()
{
// 开始扫描
ServerScanner.StartScan();
}
private void ServerScanner_OnScanComplete(object sender, List<
ConnectionResult
> e)
{
...
}
private void ServerScanner_OnScanProgressChanged(object sender, ScanProgressEventArgs e)
{
...
}
|
如果你有更好的建议或意见,请留言互相交流.
以上这篇在.NET中扫描局域网服务的实现方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我.
原文链接:https://www.cnblogs.com/wpinfo/p/serverscan.html 。
最后此篇关于在.NET中扫描局域网服务的实现方法的文章就讲到这里了,如果你想了解更多关于在.NET中扫描局域网服务的实现方法的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
创建使用.NET框架的asp.net页面时,访问该页面的客户端是否需要在其计算机上安装.NET框架? IE。用户访问www.fakesite.com/default.aspx,如果他们没有安装框架,他
我阅读了很多不同的博客和 StackOverflow 问题,试图找到我的问题的答案,但最后我找不到任何东西,所以我想自己问这个问题。 我正在构建一个应用程序,其中有一个长时间运行的工作线程,它执行一些
已锁定。这个问题及其答案是locked因为这个问题是题外话,但却具有历史意义。目前不接受新的答案或互动。 我一直想知道为什么微软为这样一个伟大的平台选择了一个如此奇怪的、对搜索引擎不友好的名称。他们就
.Net Framework .Net .NET Standard的区别 1、.NET Framework 在未来.NET Framework或许成为过去时,目前还是有很多地方在使用的。这一套
如果有选择的话,您会走哪条路? ASP.NET Webforms + ASP.NET AJAX 或 ASP.NET MVC + JavaScript Framework of your Choice
我有一个 Web 服务,它通过专用连接通过 https 使用第三方 Web 服务,我应用了 ServicePointManager.ServerCertificateValidationCallbac
为什么我应该选择ASP.NET Web Application (.NET Framework)而不是ASP.NET Core Web Application (.NET Framework)? 我在
我在网络上没有找到任何关于包含 .NET Standard、.NET Core 和 .NET Framework 项目的 .NET 解决方案的公认命名约定。 就我而言,我们在 .NET 框架项目中有以
.NET Compact 是 .NET 的完美子集吗? 假设我考虑了屏幕大小和其他限制并避免了 .NET Compact 不支持的类和方法,或者 .NET Compact 是一个不同且不兼容的 GUI
我已经阅读了所有我能找到的关于 connectionManagement 中的 maxconnection 设置的文章:即 http://support.microsoft.com/kb/821268
我现在正在使用asp.net mvc,想知道使用内置的Json或 Json.Net哪个是更好的选择,但我不确定一个人是否比另一个人有优势。 另外,如果我确实选择沿用Json.Net的路线,那么我应该选
在 Visual Studio 中,您至少可以创建三种不同类型的类库: 类库(.NET Framework) 类库(.NET 标准) 类库(.NET Core) 虽然第一个是我们多年来一直使用的,但我
.NET 和 ASP.NET 之间有什么区别?它们有什么关系? 最佳答案 ASP.Net 基于 .Net 框架构建,提供有关 Web 开发的附加功能。 你可以去看看wikipedia article
在安装更高版本(3.0)之前,我需要安装.net框架1.1和2.0吗?或者单独安装 3.0 框架就足够了,并为在早期框架版本上编写的软件提供支持?谢谢 ,丽然 最佳答案 不,您不必安装以前的框架。 我
我正在开发一个项目,人们可以“更新”类别,例如更改类别的名称。我收到以下消息 This is called after clicking update 按钮 with the SQL statemen
.NET 类 System.Net.CookieContainer 线程安全吗? --更新:交 key 答复-- 是否有任何方法可以确保异步请求期间修改的变量(即 HttpWebRequest.Coo
我正在使用 JScript.NET 在我编写的 C# WinForms 应用程序中编写脚本。它工作得很好,但我只是尝试在脚本中放置一些异常处理,但我无法弄清楚如何判断我的 C# 代码抛出了哪种类型的异
我需要你的帮助, 比如我有一个小数类型的变量,我想这样取整。 例如 3.0 = 3 3.1 = 4 3.2 = 4 3.3 = 4 3.4 = 4 3.5 = 4 3.6 = 4 3.7 = 4 3.
我使用过这样的代码:http://msdn.microsoft.com/en-us/library/dw70f090.aspx在 ASP.NET 中工作之前访问数据库(2-3 年前)。我没有意识到我正
自 ConfigurationManager .NET Standard 中不存在,检索正在执行的程序集的应用程序设置的最佳方法是什么,无论是 web.config或 appSettings.{env
我是一名优秀的程序员,十分优秀!