|
– – 本来一直在做交通智能仿真产品,结果中途来了一个插曲,目前需求就是这样的,由于现场网络环境局限,是公安网,有很多台设备只能将模式设置为服务端,那么我要编写一个TCP客户端程序,去接收每一台设备的数据,然后解析处理存入数据库,我用TCP异步的方式实现了该功能,但是发现异步也有异步的缺点,所以我想多线程+同步接收的方式来实现以后做一个对比,稳定性或者效率等等。下面是我连接一台设备服务器的代码,我想知道,如果是多台,我该如何修改以下代码,以实现接受处理多台服务器的数据?谢谢!
/// <summary>
/// 程序开始
/// </summary>
public static void StartMain()
{
try
{
ByteQueue queue = new ByteQueue();
int port = 4001;
string host = "192.168.0.81";
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);
Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
socket.Connect(ipe);
while (true)
{
//建立缓冲区
byte[] recByte = new byte[4096];
int bytes = socket.Receive(recByte, recByte.Length, 0);
//分多次接收
byte[] reallData = new byte[bytes];
Array.Copy(recByte, reallData, reallData.Length);
queue.Enqueue(reallData);
//while可处理同时接收到多个包 防止一帧数据不是完整的包
while (queue.Find())
{
byte[] readBuffer = queue.Dequeue();
//解析雷达数据 包含平均速度/占有率/车流量
string data = BitConverter.ToString(readBuffer);
if (data.StartsWith("FF-FF-FF-FF-CA-CB-CC-CD") && data.EndsWith("EA-EB-EC-ED"))
{
string repStr = data.Replace("-", " ");
Regex regex = new Regex("07 81 08.{24}");
MatchCollection matches = regex.Matches(repStr, 0);
//判断是否是统计数据
if (matches.Count > 0)
{
List<T_RadarStatistics> statisticsList = HandleStatisticsData(repStr, host);
foreach (var item in statisticsList)
{
StringBuilder builder = new StringBuilder();
//builder.AppendFormat("\n 地址:{0}:{1} ", ip, port);
builder.Append("\n 时间: " + DateTime.Now);
builder.Append("\n 测量线: " + item.Measureline);
builder.Append("\n 车道: " + item.StatisLlane);
builder.Append("\n 平均车速: " + item.SumAvspeed);
builder.Append("\n 时间占有率: " + item.SumOccupancy);
builder.Append("\n 车流量: " + item.SumVolume);
builder.Append("\n 车头实距: " + item.HeadWay);
builder.Append("\n----------------------------------------------------------");
Console.WriteLine(builder.ToString());
}
}
}
}
}
}
catch (ArgumentNullException e)
{
//MessageBox.Show("ArgumentNullException: {0}", e.Message);
}
catch (SocketException e)
{
//MessageBox.Show("SocketException: {0}", e.Message);
}
}
|
|
|
服务端不会主动地向客户端发送数据,因为他并不知道客户端是否存在
所以通讯过程是客户端向服务端发送请求后,才能接收到服务端发回的数据 你使用异步方式是很科学的,因为等待数据回传是不占用系统资源的 你若改成同步方式,那么势必需要构建一个持久的循环,不断地检查数据传输是否结束 |
|
|
首先谢谢您的回答,但是目前的情况就是,一旦设备设置成Server模式,那么只要我这边客户端连接成功,就可以获取数据了。这个不用考虑。现在就是想用多线程+同步的方式实现看看。 |
|
|
把你的方法 StartMain() 修改为 public static void StartMain(string host, int port)
{
.....
同时检查一遍、方法里不应该乱用其它“共享变量”。 var clientHostList = new string[ ]{"192.168.0.81"};
var port = 12345;
foreach(string host in clientHostList)
{
var h = host;
ThreadPool.QueueUserWorkItem(h=> StartMain(h, port));
}
|
|
|
嗯,变量命名有重复,应该编译不过去。把变量命名改一下,例如
foreach(string host in clientHostList)
{
var svr = host;
ThreadPool.QueueUserWorkItem(h=> StartMain(svr, port));
}
|
|
|
我擦又是你?
|
|
|
下位机客户端做模拟就好做了..
byte[] buffer = new byte[1024]; Socket socket; 连接很简单 socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); socket.Connect(txt_ip.Text, int.Parse(txt_port.Text)); 然后丢一个这个就好了 socket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(ReceiveMessage), socket); 当然实现的方式是这样的.
void ReceiveMessage(IAsyncResult ar)
{
var socket = ar.AsyncState as Socket;
var length = socket.EndReceive(ar);
byte[] reallData = new byte[length];
Array.Copy(buffer, reallData, length);
var abc=string.Join("-", reallData.Select(d => d.ToString("X2")).ToArray());
//处理
socket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(ReceiveMessage), socket);
}
|
|
|
楼上的- – 我异步的方法做好了 你这个是异步的方式 我知道 我现在想要利用多线程+同步的方式
|
|
|
我刚才试了一下 这样子 我调试的时候就没法获得主线程的数据过来了 调试到以下这一步 就无法进行调试了 在main函数中调用的 ByteQueue queue = new ByteQueue(); IPAddress ip = IPAddress.Parse(host); IPEndPoint ipe = new IPEndPoint(ip, port); |
|
|
客户端,特别是短连接的客户端,用不着异步处理。
你的方法要能够“只写一份”而启动上百个线程调用它,所需要变化的参数设备的host和port。除了参数,其它部分应该保持你的 StartMain 方法的基本不变。 |
|
|
现在改成这样,但是CMD窗口 直接任何反应都没有了 调试也没有调试进去,这个应该是多线程问题,这个我该怎么获取数据了- –
static void Main(string[] args)
{
var clientHostList = new string[] { "192.168.0.81", "192.168.0.87" };
var port = 4001;
foreach (string host in clientHostList)
{
var svr = host;
ThreadPool.QueueUserWorkItem(h => StartMain(svr, port));
}
}
/// <summary>
/// 程序开始
/// </summary>
public static void StartMain(string host, int port)
{
try
{
ByteQueue queue = new ByteQueue();
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);
Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
socket.Connect(ipe);
if (socket.Connected)
{
Console.WriteLine(ip + " Connected......");
}
while (true)
{
//建立缓冲区
byte[] recByte = new byte[4096];
int bytes = socket.Receive(recByte, recByte.Length, 0);
//分多次接收
byte[] reallData = new byte[bytes];
Array.Copy(recByte, reallData, reallData.Length);
queue.Enqueue(reallData);
//while可处理同时接收到多个包 防止一帧数据不是完整的包
while (queue.Find())
{
byte[] readBuffer = queue.Dequeue();
//解析雷达数据 包含平均速度/占有率/车流量
string data = BitConverter.ToString(readBuffer);
if (data.StartsWith("FF-FF-FF-FF-CA-CB-CC-CD") && data.EndsWith("EA-EB-EC-ED"))
{
string repStr = data.Replace("-", " ");
Regex regex = new Regex("07 81 08.{24}");
MatchCollection matches = regex.Matches(repStr, 0);
//判断是否是统计数据
if (matches.Count > 0)
{
List<T_RadarStatistics> statisticsList = HandleStatisticsData(repStr, host);
foreach (var item in statisticsList)
{
StringBuilder builder = new StringBuilder();
builder.AppendFormat("\n 地址:{0}:{1} ", ip, port);
builder.Append("\n 时间: " + DateTime.Now);
builder.Append("\n 测量线: " + item.Measureline);
builder.Append("\n 车道: " + item.StatisLlane);
builder.Append("\n 平均车速: " + item.SumAvspeed);
builder.Append("\n 时间占有率: " + item.SumOccupancy);
builder.Append("\n 车流量: " + item.SumVolume);
builder.Append("\n 车头实距: " + item.HeadWay);
builder.Append("\n----------------------------------------------------------");
Console.WriteLine(builder.ToString());
}
}
}
}
}
}
catch (ArgumentNullException e)
{
//MessageBox.Show("ArgumentNullException: {0}", e.Message);
}
catch (SocketException e)
{
//MessageBox.Show("SocketException: {0}", e.Message);
}
}
|
|
|
使用 HttpListener 可以同时监听多个端口,很方便。
ThreadPool.SetMaxThreads(100,100); |
|
|
sp大神送佛送到西- – 哈哈 |
|
| 100分 |
CMD窗口?难道你贴出顶楼问题时其实是没有跑起来的?你在顶楼的程序怎样跑起来的,现在不用做任何改动!只不过把原来调用 StartMain() 的那一行代码修改一下啊。 唉!如果你一定要使用 CMD 窗口老来测试的话,好歹要在最后写一行 Console.ReadLine() 之类的话啊。不然 CMD 窗口进程直接就结束了嘛。 而这种程序我相信通常是跑在你的桌面(例如winform)程序中的,要在一个窗体中去实时显示每一个设备的通讯状态(至少你要给人家显示几排黄绿灯图标)啊。 |