– – 本来一直在做交通智能仿真产品,结果中途来了一个插曲,目前需求就是这样的,由于现场网络环境局限,是公安网,有很多台设备只能将模式设置为服务端,那么我要编写一个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)程序中的,要在一个窗体中去实时显示每一个设备的通讯状态(至少你要给人家显示几排黄绿灯图标)啊。 |