c# - 访问 Windows 窗体控件时,BackgroundWorker 线程失败

标签 c# winforms backgroundworker

我正在尝试从后台工作进程更新 Windows 窗体控件,如下面的代码所示。不幸的是,我收到以下错误:

Cross-thread operation not valid: Control 'richTextBox1' accessed from a thread other than the thread it was created on.

这是我正在使用的代码:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Net;
using System.IO;
using System.Text.RegularExpressions;

namespace WindowsFormsApplication1 {
    public partial class Form1 : Form {
        private CookieContainer cookies;
        public Form1() {
            InitializeComponent();
            backgroundWorker1.DoWork += 
                new DoWorkEventHandler(backgroundWorker1_DoWork);    
        }

        public string od_auth(string s) {    
            if (s.Contains(":")) {
                string[] acc = s.Split(':');
                cookies = new CookieContainer();
                HttpWebResponse res;
                try {
                    HttpWebRequest req = 
                        (HttpWebRequest)HttpWebRequest.Create(
                            "http://m.site.ru/dk?" + 
                            "bk=GuestMain" + 
                            "&st.cmd=main" + 
                            "&tkn=1110" +
                            "&fr.posted=set" +
                            "&fr.needCaptcha=" + 
                            "&fr.login=" + acc[0] + 
                            "&fr.password=" + acc[1] +
                            "&fr.remember=on" + 
                            "&button_login=%D0%92%D0%BE%D0%B9%D1%82%D0%B8"
                        );
                    req.AllowAutoRedirect = false;
                    req.Method = "HEAD";
                    req.CookieContainer = cookies;
                    req.UserAgent = "Opera/9.80 (Windows NT 6.1; U; ru) " +
                                    "Presto/2.2.15 Version/10.10";
                    req.ContentType = "application/x-www-form-urlencoded";
                    res = (HttpWebResponse)req.GetResponse();
                    res.Close();

                    string cook = "";
                    try {
                        if (res.Headers["Location"].Contains("cmd=userMain")) {    
                            cook = "AUTHCODE=" + Regex.Match(res.Headers["Set-Cookie"], @"HCODE=(?<id>[^;]+);").Groups["id"].Value + "; ";
                            cook += "JSESSIONID=" + Regex.Match(res.Headers["Set-Cookie"], @"ESSIONID=(?<id>[^;]+);", RegexOptions.RightToLeft).Groups["id"].Value + ";";
                            richTextBox1.Text += "[+] Авторизовались успешно на " + acc[0] + ":" + acc[1] + "\r\n";
                            return cook;
                        } else {
                            cook = "badacc";
                            richTextBox1.Text += "[-] Аккаунт " + acc[0] + ":" + acc[1] + " невалидный\r\n";
                            return cook;
                        }
                    } catch {
                        cook = "badacc";
                        richTextBox1.Text += "[-] Аккаунт " + acc[0] + ":" + acc[1] + " невалидный\r\n";
                        return cook;
                    }
                }
                catch { richTextBox1.Text += "[?] network error \r\n"; ;return "err"; }
            }

            richTextBox1.Text += "[?] some error \r\n";
            return "err";
        }

        public void od_info_changer(string cook) {
            HttpWebRequest request = (HttpWebRequest)WebRequest.Create("http://m.site.ru/dk?bk=UserSettingsProfile&st.cmd=userSettingsProfile&tkn=2154");
            request.Method = "POST";
            request.Headers.Add("Cookie: "+cook);
            request.AllowAutoRedirect = false;
            request.ContentType = "application/x-www-form-urlencoded";
            byte[] EncodedPostParams = Encoding.UTF8.GetBytes("fr.posted=set&fr.name=D1%8F&fr.surname=%D0&fr.gender=1&fr.birthday=12&fr.bmonth=10&fr.byear=1949&fr.country=10414533690&fr.city=%D0%95%D0%BA%D0%B0%D1%82%D0%B5%D1%80%D0%B8%D0%BD%D0%B1%D1%83%D1%80%D0%B3&button_save=%D0%A1%D0%BE%D1%85%D1%80%D0%B0%D0%BD%D0%B8%D1%82%D1%8C");
            request.ContentLength = EncodedPostParams.Length;
            request.GetRequestStream().Write(EncodedPostParams, 0, EncodedPostParams.Length);
            request.GetRequestStream().Close();

            HttpWebResponse response = (HttpWebResponse)request.GetResponse();
            string html = new StreamReader(response.GetResponseStream(), Encoding.UTF8).ReadToEnd();
            if (response.Headers["Location"].IndexOf("st.cmd=userSettings") != -1) {
                richTextBox1.Text += "[+] Cменили пароль\r\n";
            } else {
                richTextBox1.Text += "[-] Не смогли сменить имя\r\n";
            }               
        }

        List<string> accs=new List<string>();
        private void открытьАккаунтыToolStripMenuItem_Click(object sender, EventArgs e) {
            OpenFileDialog openFileDialog1 = new OpenFileDialog();
            openFileDialog1.Filter = "txt files (*.txt)|*.txt|All files (*.*)|*.*";
            if (openFileDialog1.ShowDialog() == DialogResult.OK) {
                accs.Clear();
                foreach (string s in File.ReadAllLines(openFileDialog1.FileName))
                    accs.Add(s);

                richTextBox1.Text += "[+] Загрузили " + accs.Count.ToString() + " аккаунтов\n";
            }
        }

        private void Auth_Click(object sender, EventArgs e) {
            Auth.Enabled = false; // На время расчета блокируем опасные кнопки
            backgroundWorker1.RunWorkerAsync();
        }    

        public void go() {
            foreach (string acc in accs) {
                string cook = od_auth(acc);
                if (cook!="badacc")
                    od_info_changer(cook);
            }    
        }

        private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e) {
            go();
        }
    }
}

但是,加载 .txt 后文件并按下我的按钮,我收到:

this error message

出了什么问题?

最佳答案

您无法直接从后台线程更新控件。您需要调用回主线程并在那里执行操作:

delegate void updateDelegate(string val);

private void UpdateText(string updateVal)
{
    if (richTextBox1.InvokeRequired)
        richTextBox1.Invoke(new updateDelegate(UpdateText), updateVal);
    else
        richTextBox1.Text += updateVal;

}

然后在后台工作代码中,您可以直接调用 UpdateText 方法,而不是设置 richTextBox1.text

UpdateText("Whatever Value you want");

关于c# - 访问 Windows 窗体控件时,BackgroundWorker 线程失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13653810/

相关文章:

c# - Entityframework Core 无法将类型为 'System.Int32' 的对象转换为类型 'System.Int64'

c# - 如何使用 ActionResult<T> 进行单元测试?

c# - 根据 C# 中的表单最大化调整面板宽度

处理 "heavy"数据时的 WinForms UI 响应能力

c# - 由于未注册 Windows,Windows 10 分配的访问权限应用无法启动。启动契约(Contract)

c# - ASP.NET MVC4 重定向到登录页面

c# - 动态显示数百张图片

c# - 一种形式关闭 - 第二次更新 DataGridView

c# - 在 BackgroundWorker 运行时显示模态窗口,而不会出现 STA/MTA 问题

C# BackgroundWorker 和 Com 端口问题