【问题标题】:chat application c# safe thread [duplicate]聊天应用程序c#安全线程[重复]
【发布时间】:2017-08-31 16:21:32
【问题描述】:

我已经使用基于 C# 网络编程中的代码的线程创建了一个 LAN 聊天应用程序 - Richard Blum。但是当我运行程序时(我打开 2 个窗口聊天应用程序),单击一侧的连接,突然代码崩溃并出现错误“附加信息:跨线程操作无效:控制 'lst_show' 从线程以外的线程访问它是在上面创建的。”

有人帮助我,我尝试了几个小时前在这里询问但仍然无法解决。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Net;
using System.Net.Sockets;
using System.Threading;

namespace Chat_Threads
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {

        }
        private static TextBox newText = new TextBox();
        private static ListBox results = new ListBox();
        private static Socket client;
        private static byte[] data = new byte[1024];


        private void btn_listen_Click(object sender, EventArgs e)
        {
            lst_show.Items.Add("Listening for a client...");

            Socket newsock = new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

            IPEndPoint iep = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 9050);

            newsock.Bind(iep);

            newsock.Listen(5);

            newsock.BeginAccept(new AsyncCallback(AcceptConn),newsock);   


        }

        private void AcceptConn(IAsyncResult ar)
        {
            Socket oldserver = (Socket)ar.AsyncState;

            client = oldserver.EndAccept(ar);

            lst_show.Items.Add("Connection from: " + client.RemoteEndPoint.ToString());

            Thread receiver = new Thread(new ThreadStart(ReceiveData));

            receiver.Start();
        }

        private void ReceiveData()
        {
            int recv;
            string stringData;
            while (true)
            {
                recv = client.Receive(data);
                stringData = Encoding.ASCII.GetString(data, 0, recv);
                if (stringData == "bye")
                    break;
                lst_show.Items.Add(stringData);
            }
            stringData = "bye";
            byte[] message = Encoding.ASCII.GetBytes(stringData);
            client.Send(message);
            client.Close();
            lst_show.Items.Add("Connection stopped");
            return;
        }

        private void btn_connect_Click(object sender, EventArgs e)
        {
            lst_show.Items.Add("Connecting...");

            client = new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

            IPEndPoint iep = new IPEndPoint(IPAddress.Parse("127.0.0.1"),9050);

            client.BeginConnect(iep, new AsyncCallback(Connected),client);
        }

        private void Connected(IAsyncResult ar)
        {
            try
            {
                client.EndConnect(ar);

                this.lst_show.Items.Add("Connected to: " + client.RemoteEndPoint.ToString());

                Thread receiver = new Thread(new ThreadStart(ReceiveData));
                receiver.Start();

            }
            catch (SocketException)
            {
                this.lst_show.Items.Add("Error connecting");
            }
        }

        private void btn_send_Click(object sender, EventArgs e)
        {
            byte[] message = Encoding.ASCII.GetBytes(txt_message.Text);

            txt_message.Clear();

            client.BeginSend(message, 0, message.Length, 0, new AsyncCallback(SendData),client);

        }

        private void SendData(IAsyncResult ar)
        {
            Socket remote = (Socket)ar.AsyncState;
            int sent = remote.EndSend(ar);
        }


    }
}

【问题讨论】:

  • 只是为了记录,这是一本 2002 年的书......
  • 是的,这已经过时了。

标签: c# thread-safety


【解决方案1】:

调用者在不同的线程上,所以你必须确保列表和调用者在主 UI 线程上,所以以下是

control.Invoke 是必需的。

     delegate void UpdateStringinTheGrid(string s);

      private void ListAddItem(string s)
        {
            if (lst_show.InvokeRequired())
            {
                var updateDel = new UpdateStringinTheGrid(ListAddItem);
                this.BeginInvoke(updateDel, s);
            }
          else
            lst_show.Items.Add(s);

}
    private void ReceiveData()
            {
                int recv;
                string stringData;
                while (true)
                {
                    recv = client.Receive(data);
                    stringData = Encoding.ASCII.GetString(data, 0, recv);
                    if (stringData == "bye")
                        break;
                   listAddString(stringData);
                }
                stringData = "bye";
                byte[] message = Encoding.ASCII.GetBytes(stringData);
                client.Send(message);
                client.Close();
                lst_show.Items.Add("Connection stopped");
                return;
            }

【讨论】:

  • 感谢斯图尔特·史密斯。我使用的是 2003 年的旧书,发生了很多我无法理解的变化。
  • 很高兴我能帮上忙
【解决方案2】:

大多数 UI 引擎都有一个带有专用处理循环的主线程来更新用户界面。这是因为,如果多个线程尝试进行更改,可能会产生竞争条件或死锁。

您的业务逻辑通常在单独的线程上运行,尤其是在您进行 TCP/IP 通信之类的事情时。当业务逻辑需要对 UI 进行更改时,它会以间接的方式进行,例如将消息放入队列或调用特殊方法。

在 .NET 中,您可以使用 Invoke 方法在主 UI 线程中运行代码,然后它会在有机会时安全地执行它。例如:

lst_show.Items.Invoke((MethodInvoker)delegate 
{
    lst_show.Items.Add("Connection from: " + client.RemoteEndPoint.ToString());
});

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-12-06
    • 1970-01-01
    • 2017-07-17
    • 1970-01-01
    • 1970-01-01
    • 2017-04-02
    相关资源
    最近更新 更多