C# WPF上位机实现和下位机TCP通讯

简介: C# WPF上位机实现和下位机TCP通讯下位机使用北京大华程控电源DH1766-1,上位机使用WPF。实现了电压电流实时采集,曲线显示。上午在公司调试成功,手头没有程控电源,使用TCP服务端模拟。昨天写的TCP服务端正好排上用场。

C# WPF上位机实现和下位机TCP通讯
下位机使用北京大华程控电源DH1766-1,上位机使用WPF。实现了电压电流实时采集,曲线显示。上午在公司调试成功,手头没有程控电源,使用TCP服务端模拟。昨天写的TCP服务端正好排上用场。

界面如下:

服务端

服务端实在上篇基础上实现的。需要做如下更改:

复制代码
while (true)

                      {
                          try
                          {
                              byte[] bufferDate = new byte[1024];
                              int realLen = pSocket.Receive(bufferDate);
                              if (realLen <= 0)
                              {
                                  this.Invoke(addTextDelegate, pSocket.RemoteEndPoint.ToString() + "退出\r\n");
                                  socketList.Remove(pSocket);
                                  //客户端退出的时候会发送一个空字节
                                  pSocket.Shutdown(SocketShutdown.Both);
                                  pSocket.Close();
                                  return;
                              }
                              string receiveStr = Encoding.Default.GetString(bufferDate, 0, realLen);
                              switch (receiveStr)
                              {
                                  case "MEAS:VOLTage:ALL?\n":
                                      proxSocket.Send(Encoding.Default.GetBytes(r.Next(16,25).ToString()+ ","+r.Next(16, 25).ToString()+","+ r.Next(16, 25).ToString()));
                                      break;
                                  case "MEAS:CURR:ALL?\n":
                                      proxSocket.Send(Encoding.Default.GetBytes(r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString()));
                                      break;
                                  default:
                                      break;
                              }
                              this.Invoke(addTextDelegate, receiveStr + "from" + pSocket.RemoteEndPoint.ToString() + "\r\n");
                          }
                          catch (Exception ex)
                          {
                              this.Invoke(addTextDelegate, pSocket.RemoteEndPoint.ToString() + "异常退出\r\n");
                              socketList.Remove(pSocket);
                              pSocket.Shutdown(SocketShutdown.Both);
                              pSocket.Close();
                              return;
                          }
                      }

复制代码
在While循环中加入:

switch (receiveStr)
{
  case "MEAS:VOLTage:ALL?n":
  proxSocket.Send(Encoding.Default.GetBytes(r.Next(16,25).ToString()+ ","+r.Next(16, 25).ToString()+","+ r.Next(16, 25).ToString()));
  break;
  case "MEAS:CURR:ALL?n":
  proxSocket.Send(Encoding.Default.GetBytes(r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString()));
  break;
  default:
  break;

}

模拟电源,当收到电压查询时,发送16~25中随机数,由于电源是三个通道的,因此发送三个随机数,用逗号隔开。同样收到电流查询,发送2~5之间的随机数。

完整的客户端源码:

复制代码
public partial class Form1 : Form

{
    public Form1()
    {
        InitializeComponent();
        addTextDelegate = new AddTextDelegate(AddText);
    }
    private AddTextDelegate addTextDelegate;
    private List<Socket> socketList = new List<Socket>();

    public delegate void AddTextDelegate(string text);
    private void AddText(string text)
    {
        txtLog.Text += text;
    }

    Random r = new Random();

    private void btnStart_Click(object sender, EventArgs e)
    {
        //参数:寻址方式    传输数据方式  通信协议
        Socket socket = new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

        IPAddress iPAddress = IPAddress.Parse(txtIP.Text);

        //创建EndPoint
        IPEndPoint iPEndPoint = new IPEndPoint(iPAddress, int.Parse(txtPort.Text));

        //绑定端口
        socket.Bind(iPEndPoint);

        //开启侦听
        socket.Listen(10);

        txtLog.Text += "服务启动开启侦听……\r\n";

        Thread thread = new Thread((s) =>
          {
              Socket serSocket = (Socket)s;
              while (true)//不断接收客户端连接
              {
                  this.Invoke(addTextDelegate, "服务正在等待客户端连接……\r\n");

                  //开始接收客户端的连接
                  //阻塞当前线程,等待客户端连接
                  //客户端连接上之后,服务端自动生成一个socket和连接的客端通信
                  Socket proxSocket = serSocket.Accept();

                  this.Invoke(addTextDelegate, "客户端连接成功!\r\n" + proxSocket.RemoteEndPoint.ToString());

                  //proxSocket.Send(Encoding.Default.GetBytes("连接成功!"));

                  socketList.Add(proxSocket);//当前通信的socket放到集合中

                  new Thread(p =>
                  {
                      Socket pSocket = (Socket)p;
                      while (true)
                      {
                          try
                          {
                              byte[] bufferDate = new byte[1024];
                              int realLen = pSocket.Receive(bufferDate);

                              if (realLen <= 0)
                              {
                                  this.Invoke(addTextDelegate, pSocket.RemoteEndPoint.ToString() + "退出\r\n");

                                  socketList.Remove(pSocket);
                                  //客户端退出的时候会发送一个空字节
                                  pSocket.Shutdown(SocketShutdown.Both);
                                  pSocket.Close();

                                  return;
                              }
                              string receiveStr = Encoding.Default.GetString(bufferDate, 0, realLen);
                              switch (receiveStr)
                              {
                                  case "MEAS:VOLTage:ALL?\n":
                                      proxSocket.Send(Encoding.Default.GetBytes(r.Next(16,25).ToString()+ ","+r.Next(16, 25).ToString()+","+ r.Next(16, 25).ToString()));
                                      break;
                                  case "MEAS:CURR:ALL?\n":
                                      proxSocket.Send(Encoding.Default.GetBytes(r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString() + "," + r.Next(2, 5).ToString()));
                                      break;
                                  default:
                                      break;
                              }
                              this.Invoke(addTextDelegate, receiveStr + "from" + pSocket.RemoteEndPoint.ToString() + "\r\n");
                          }
                          catch (Exception ex)
                          {
                              this.Invoke(addTextDelegate, pSocket.RemoteEndPoint.ToString() + "异常退出\r\n");

                              socketList.Remove(pSocket);
                              pSocket.Shutdown(SocketShutdown.Both);
                              pSocket.Close();
                              return;
                          }
                      }
                  })
                  { IsBackground = true }.Start(proxSocket);
              }
          });
        thread.IsBackground = true;
        thread.Start(socket);
       
    }

    private void btnSend_Click(object sender, EventArgs e)
    {
        string str = txtSend.Text;
        byte[] data = Encoding.Default.GetBytes(str);
        foreach (var socket in socketList)
        {
            if (socket != null && socket.Connected)
            {
                socket.Send(data);
            }
        }
    }
}

复制代码
上位机实现客户端功能。具体如下:

1、字段和属性

public readonly IPEndPoint TagetIPEP;
public bool IsConnected { get; set; } = false;
private Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp) { /ReceiveTimeout=1000,SendTimeout=1000/};
private Thread recListenThread;
public string ReceiveStr { get; set; }
public byte[] ReceiveByte { get; set; }
TagetIPEP是服务器地址和端口。
IsConnected是连接的状态,这个比较重要,在发送和接收时,都要更加IsConnected进行,并更新IsConnected。
Socket用于和客户端通讯。
recListenThread是监听客户端消息的线程。
ReceiveStr和ReceiveByte用来存储客户端发来的消息。
2、方法函数
连接方法:
复制代码
public bool Connect()

    {
        try
        {
            socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp)
            {
                //ReceiveTimeout = 1000,
                //SendTimeout=1000
            };

            //IAsyncResult connResult = socket.BeginConnect(TagetIPEP.Address, TagetIPEP.Port, null, null);
            //connResult.AsyncWaitHandle.WaitOne(5000, true);
            //if (connResult.IsCompleted)
            //{
            socket.Connect(TagetIPEP.Address, TagetIPEP.Port);
            IsConnected = true;
                //开启接收监听

                recListenThread = new Thread(() =>
                {
                    while (true)
                    {
                        try
                        {
                            ReceiveByte = new byte[1024];
                            int realLen = socket.Receive(ReceiveByte);
                            ReceiveStr = Encoding.Default.GetString(ReceiveByte, 0, realLen);
                            ReceiveEvent();
                            if (realLen <= 0)
                            {
                                if (socket != null && socket.Connected)
                                {
                                    //服务器退出
                                    IsConnected = false;
                                    Log.WriteLog("服务器退出!");
                                    socket.Shutdown(SocketShutdown.Both);
                                    socket.Close();
                                    MessageBox.Show("连接断开!");
                                }
                                return;
                            }
                        }
                        catch (Exception ex)
                        {
                            if (socket != null && socket.Connected)
                            {
                                IsConnected = false;
                                Log.WriteLog("服务器异常退出!", ex);
                                socket.Shutdown(SocketShutdown.Both);
                                socket.Close();
                            }
                            return;
                        }
                    }
                })
                { IsBackground = true };
                recListenThread.Start();
                return true;
            //}

        }
        catch (Exception ex)
        {
            Log.WriteLog(TagetIPEP.Address + "连接失败", ex);
        }
        return false;
    }

复制代码
连接函数返回值为bool类型,根据返回值判断连接是否成功连接。这里每次连接都实例化了一个socket,因为在执行socket.close()后,重新打开会失败,而断线重连会经常用到,没有找到更好的方法,干脆重新实例化socket。连接成功后,开启监听服务端消息的线程。这里使用了一个ReceiveEvent()事件,在接收到消息时会触发这个事件,刷新UI界面。

发送方法:

复制代码
public bool Send(string msg)

    {
        byte[] sendMsg = Encoding.Default.GetBytes(msg);
        if (sendMsg.Length > 0&&IsConnected)
        {
            if (socket != null && socket.Connected)
            {
                try
                {
                    socket.Send(sendMsg);
                    return true;
                }
                catch (Exception ex)
                {
                    IsConnected = false;
                    Log.WriteLog("发送数据失败,目标地址" + TagetIPEP.Address, ex);
                }
            }
        }

        return false;

    }

复制代码
关闭方法:

复制代码

public void Close()
    {
        if (socket != null && socket.Connected)
        {
            IsConnected = false;
            recListenThread.Abort();
            Log.WriteLog("关闭连接!");
            socket.Shutdown(SocketShutdown.Both);
            socket.Close();
        }
    }

复制代码
在出现异常时调用

消息接收事件:

public event Action ReceiveEvent;

每次接收消息时触发,获取属性ReceiveStr和ReceiveByte的值,刷新UI界面。

完整代码:

复制代码
public class TCPClient

{
    public TCPClient(/*IPEndPoint localIPEP,*/IPEndPoint targetIPEP)
    {
        //socket.Bind(localIPEP);
        TagetIPEP = targetIPEP;
       
    }

    public readonly IPEndPoint TagetIPEP;

    public bool IsConnected { get; set; } = false;

    private Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp) { /*ReceiveTimeout=1000,SendTimeout=1000*/};

    public bool Connect()
    {
        try
        {
            socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp)
            {
                //ReceiveTimeout = 1000,
                //SendTimeout=1000
            };

            //IAsyncResult connResult = socket.BeginConnect(TagetIPEP.Address, TagetIPEP.Port, null, null);
            //connResult.AsyncWaitHandle.WaitOne(5000, true);
            //if (connResult.IsCompleted)
            //{
            socket.Connect(TagetIPEP.Address, TagetIPEP.Port);
            IsConnected = true;
                //开启接收监听

                recListenThread = new Thread(() =>
                {
                    while (true)
                    {
                        try
                        {
                            ReceiveByte = new byte[1024];
                            int realLen = socket.Receive(ReceiveByte);
                            ReceiveStr = Encoding.Default.GetString(ReceiveByte, 0, realLen);
                            ReceiveEvent();
                            if (realLen <= 0)
                            {
                                if (socket != null && socket.Connected)
                                {
                                    //服务器退出
                                    IsConnected = false;
                                    Log.WriteLog("服务器退出!");
                                    socket.Shutdown(SocketShutdown.Both);
                                    socket.Close();
                                    MessageBox.Show("连接断开!");
                                }
                                return;
                            }
                        }
                        catch (Exception ex)
                        {
                            if (socket != null && socket.Connected)
                            {
                                IsConnected = false;
                                Log.WriteLog("服务器异常退出!", ex);
                                socket.Shutdown(SocketShutdown.Both);
                                socket.Close();
                            }
                            return;
                        }
                    }
                })
                { IsBackground = true };
                recListenThread.Start();
                return true;
            //}

        }
        catch (Exception ex)
        {
            Log.WriteLog(TagetIPEP.Address + "连接失败", ex);
        }
        return false;
    }

    public bool Send(string msg)
    {
        byte[] sendMsg = Encoding.Default.GetBytes(msg);
        if (sendMsg.Length > 0&&IsConnected)
        {
            if (socket != null && socket.Connected)
            {
                try
                {
                    socket.Send(sendMsg);
                    return true;
                }
                catch (Exception ex)
                {
                    IsConnected = false;
                    Log.WriteLog("发送数据失败,目标地址" + TagetIPEP.Address, ex);
                }
            }
        }

        return false;

    }

    public event Action ReceiveEvent;

    public string ReceiveStr { get; set; }

    public byte[] ReceiveByte { get; set; }

    public void Close()
    {
        if (socket != null && socket.Connected)
        {
            IsConnected = false;
            recListenThread.Abort();
            Log.WriteLog("关闭连接!");
            socket.Shutdown(SocketShutdown.Both);
            socket.Close();
        }
    }

    private Thread recListenThread;

}

复制代码
前台调用,声明Timer定时器,每个一秒触发一次。触发事件如下:

复制代码

    private string flag = "";
    private void QueryTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
    {
        Now = DateTime.Now;

        if (!tcp.Send("MEAS:VOLTage:ALL?\n"))
        {
            queryTimer.Enabled = false;
            StartContent = "开始";
            ConnContent = "连接";
            tcp.IsConnected = false;
            MessageBox.Show("查询失败!");
            return;
        }
        flag = "V";
        Thread.Sleep(50);

        if (!tcp.Send("MEAS:CURR:ALL?\n"))
        {
            queryTimer.Enabled = false;
            StartContent = "开始";
            ConnContent = "连接";
            tcp.IsConnected = false;
            MessageBox.Show("查询失败!");
            return;
        }
        flag = "C";

        #region 测试
        //angle += 18;
        //if (angle > 360)
        //{
        //    angle = 18;
        //}

        #endregion

    }

复制代码
刷新UI界面的事件如下:

复制代码
private void Tcp_ReceiveEvent()

    {
        Task.Run(() =>
        {
            Application.Current.Dispatcher.Invoke(() =>
            {
                RemoteIP = tcp.TagetIPEP.ToString();
                switch (flag)
                {
                    case "V":
                        VoltValue = Math.Round(Convert.ToDouble(tcp.ReceiveStr.Split(',')[0]), 3);
                        break;
                    case "C":
                        CurrentValue = Math.Round(Convert.ToDouble(tcp.ReceiveStr.Split(',')[0]), 3);
                        break;
                    default:
                        break;
                }

                #region 测试
                //VoltValue = Math.Round(Math.Sin((angle) * pi / 180) * 16 + 16, 3);
                //CurrentValue = Math.Round(Math.Sin((angle) * pi / 180) * 2.5 + 2.5, 3);
                #endregion

                VoltValues.Add(VoltValue);
                CurrentValues.Add(CurrentValue);

                if (VoltValues.Count > 30)
                {
                    VoltValues.RemoveAt(0);
                    CurrentValues.RemoveAt(0);
                }
            });
        });
    }

复制代码
原文地址https://www.cnblogs.com/Samberger/p/10561243.html

相关文章
|
8天前
|
监控 网络协议 C#
一款基于C#开发的通讯调试工具(支持Modbus RTU、MQTT调试)
一款基于C#开发的通讯调试工具(支持Modbus RTU、MQTT调试)
|
28天前
|
存储 算法 数据处理
C# | 上位机开发新手指南(十一)压缩算法
流式压缩 流式压缩是一种能够实时处理数据流的压缩方式,例如音频、视频等实时传输的数据。 通过流式压缩算法,我们可以边读取边压缩数据,并能够随时输出已压缩的数据,以确保数据的实时性和减少存储和传输所需的带宽。 块压缩 块压缩则是将数据划分为固定大小的块,在每个块内进行独立的压缩处理。块压缩通常适用于文件、存储、传输等离线数据处理场景。 字典压缩 字典压缩是一种基于字典的压缩算法,通过建立一个字典来存储一组重复出现的字符串,并将这些字符串替换成字典中相应的索引,从而减少数据的存储和传输。字典压缩算法可以更好地处理数据中的重复模式,因为它们可以通过建立字典来存储和恢复重复出现的字符串。
60 0
C# | 上位机开发新手指南(十一)压缩算法
|
28天前
|
算法 C# 数据安全/隐私保护
C# | 上位机开发新手指南(十)加密算法——ECC
本篇文章我们将继续探讨另一种非对称加密算法——ECC。 严格的说,其实ECC并不是一种非对称加密算法,它是一种基于椭圆曲线的加密算法,广泛用于数字签名和密钥协商。 与传统的非对称加密算法(例如RSA)不同,ECC算法使用椭圆曲线上的点乘法来生成密钥对和进行加密操作,而不是使用大数分解等数学算法。这使得ECC算法具有相同的安全性和强度,但使用更少的位数,因此在资源受限的环境中具有优势。 ECC算法虽然使用公钥和私钥进行加密和解密操作,但是这些操作是基于点乘法实现的,而不是基于大数分解等算法实现的。因此,ECC算法可以被视为一种非对称加密算法的变体,但是它与传统的非对称加密算法有所不同。
151 0
C# | 上位机开发新手指南(十)加密算法——ECC
|
28天前
|
XML 算法 安全
C# | 上位机开发新手指南(九)加密算法——RSA
RSA的特性 非对称性 RSA算法使用公钥和私钥两个不同的密钥,公钥用于加密数据,私钥用于解密数据。公钥可以公开,任何人都可以使用,而私钥只有密钥持有人可以访问。 安全性 RSA算法基于大数分解难题,即将一个大的合数分解成其质数因子的乘积。由于目前没有有效的算法可以在合理的时间内对大质数进行分解,因此RSA算法被认为是一种安全的加密算法。 可逆性 RSA算法既可以用于加密,也可以用于解密。加密和解密都是可逆的过程,只要使用正确的密钥,就可以还原原始数据。 签名 RSA算法可以用于数字签名,用于验证数据的完整性和真实性。签名过程是将数据使用私钥进行加密,验证过程是将签名使用公钥进行解密。
123 0
C# | 上位机开发新手指南(九)加密算法——RSA
|
28天前
|
算法 搜索推荐 安全
C# | 上位机开发新手指南(八)加密算法——AES
AES——这是在加密算法中相当重要的一种加密方式! 虽然这个世界上已经存在了非对称加密算法(比如RSA、ECC等),但是在对称加密算法中,AES的地位依然相当重要。与非对称加密算法不同,对称加密算法使用的是相同的密钥对数据进行加密和解密,因此其加密和解密速度更快,而且更加高效。而在对称加密算法中,AES是目前最安全、最可靠的加密算法之一,其加密强度和运行效率都非常高。因此,无论是在个人计算机、移动设备,还是在服务器和云计算等领域,AES都被广泛应用于数据的加密和解密过程中。
121 0
C# | 上位机开发新手指南(八)加密算法——AES
|
28天前
|
存储 算法 安全
C# | 上位机开发新手指南(七)加密算法
加密算法是信息安全领域中的重要技术之一,可以保护数据在传输、存储和处理过程中的安全性。 学习加密算法可以帮助我们更好地理解和应用其他相关技术。例如,数字证书、数字签名、安全协议等都与加密算法密切相关,掌握加密算法可以为我们理解和应用这些技术提供帮助。
61 0
C# | 上位机开发新手指南(七)加密算法
|
28天前
|
算法 安全 C#
C# | 上位机开发新手指南(六)摘要算法
你知道摘要算法么?它在保障数据安全方面非常有用! 它能够将任意长度的数据转换成固定长度的消息摘要,从而确保数据的完整性和可靠性。比如说,我们下载软件的时候,就可以用摘要算法来检验软件是否被篡改,保障我们的电脑安全。 那这个算法的工作原理是怎样的呢?大致就是通过一系列复杂的计算,将原始数据转换为一个固定长度的摘要信息。而且无论输入的数据大小,输出的摘要信息长度都是一样的。 那么摘要算法有什么用处呢?比如数字签名,确保数据的来源和内容没有被篡改。还有密码学等领域的应用,可以说是非常厉害了!
42 0
C# | 上位机开发新手指南(六)摘要算法
|
28天前
|
存储 算法 安全
C# | 上位机开发新手指南(五)校验算法——CRC
当我们在进行数据传输时,可能会因为信道噪声、干扰等因素导致数据出现错误,从而影响传输的可靠性和准确性。此时,我们需要一种方法来检测数据是否出现错误,并尽可能快速地发现和纠正错误。CRC(Cyclic Redundancy Check)校验算法就是一种常用的数据校验方法,它通过对数据进行处理生成校验码,从而实现对数据的完整性和准确性进行验证。 使用CRC校验的意义在于能够提高数据传输的可靠性,降低数据传输错误率,确保数据的完整性和准确性。在各个领域中,如通信、网络、存储等,CRC校验都得到了广泛的应用。
110 0
C# | 上位机开发新手指南(五)校验算法——CRC
|
28天前
|
算法 C#
C# | 上位机开发新手指南(四)校验算法
校验算法是一种用于验证数据传输过程中是否出现错误或丢失的算法。 在数据传输过程中,由于噪声、干扰、传输错误等因素的影响,会导致数据传输过程中出现错误或丢失。 为了保证数据传输的准确性,需要在数据传输过程中添加校验码。发送端通过计算数据的校验码并将其附加到数据中一起发送出去,接收端再次计算校验码并将其与接收到的校验码进行比较,如果两者相同,则说明数据传输过程中没有出现错误或丢失。
151 0
C# | 上位机开发新手指南(四)校验算法
|
28天前
|
开发框架 数据可视化 C#
C# | 上位机开发新手指南(三)框架
在上位机开发中,Windows Forms是使用最广泛的C#框架之一。Windows Forms是.NET Framework中的一个GUI框架,提供了丰富的GUI控件和易于使用的编程模型,可以快速开发Windows桌面应用程序。Windows Forms已经存在多年,并且在.NET Framework中得到广泛的支持和优化,因此在上位机开发中得到了广泛应用。除此之外,随着.NET Core的不断发展,越来越多的开发人员开始使用跨平台的C#框架进行上位机开发,例如使用Electron或Avalonia等框架开发基于Web技术的桌面应用程序。
239 0
C# | 上位机开发新手指南(三)框架