1

结果发现 GetUserByID 方法存在问题,然后更新了库,问题似乎已经消失了,仍然学习了如何更好地访问线程外的 GUI。

我使用 TweetInvi 库编写了一个应用程序,它检索用户关注者和关注者,以及他们的图片,图片链接和 Twitter ID。

然后它遍历返回的列表并显示它们(都在不同的列表中)

现在,当我第一次开始使用这个应用程序时,我在 _Click 事件上运行了所有东西,当然,我冻结了 UI 直到它完成。

我现在已将代码移至后台工作线程,这导致了一些奇怪的问题。

有时它会“选择”不填充某些列表,有时它会。有时它会正确加载所有列表,除了“关注你”列表,它会过滤你的哪些朋友正在关注你(使用 If 语句过滤已验证的帐户)

起初我读到尝试在单独的线程上更新 UI 会导致奇怪的错误,所以我删除了除了它填充的列表之外的任何 UI 控件更改。

 private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
        {   
            BackgroundWorker worker = sender as BackgroundWorker;


        //user name retrieved from text box, rest of this method will pull various bits of data back

        var username = e.Argument.ToString();
        var user = User.GetUserFromScreenName(username);



        Properties.Settings.Default.LastHandle = boxUsername.Text;
        Properties.Settings.Default.Save();



        var usersTweets = user.GetUserTimeline(Convert.ToInt32(txtTweetAmount.Text)).ToList();

        foreach (var userTweet in usersTweets)
        {
            lstSearchTweetList.Invoke((MethodInvoker)delegate
            {
                var searchList = lstSearchTweetList.Items.Add(userTweet.Text);
                  searchList.SubItems.Add(userTweet.CreatedAt.ToString());
            });
        }


        var show = user.GetFollowers(500).ToList();


        foreach (var friend in show)
        {

            string screenName = "@" + friend.ScreenName;
            lstFriend.BeginInvoke((MethodInvoker)delegate
            {
                lstFriend.Items.Add(screenName); // runs on UI thread
            });

        }

        var friends = user.GetFriends(500);
        var followers = user.GetFollowers(500);


        var result2 = followers.Where(follower => friends.All(friend => follower.Name != friend.Name));
        int i2 = 0;
        foreach (var res2 in result2)
        {
            string screenName = "@" + res2.ScreenName;
            lstFollowingChecker.BeginInvoke((MethodInvoker)delegate
                {
                    lstFollowingChecker.Items.Add(screenName);
                });
                i2++;
             //   lblFollowBackAmount.Text = Convert.ToString(i2);
        }
        var result = friends.Where(friend => followers.All(follower => friend.Name != follower.Name));

        //lblFriendCount.Text = "(" + result.Count().ToString() + ")";
        int i1 = 0;
        foreach (var res in result)
        {
                if (res.Verified != true)
                {
                    string screenName = "@" + res.ScreenName;
                    lstFollowerChecker.BeginInvoke((MethodInvoker)delegate
                    {
                        lstFollowerChecker.Items.Add(screenName);
                    });

                    i1++;
                   // lblCheckerCount.Text = Convert.ToString(i1);
                }
        }



        backgroundWorker1.ReportProgress(1,username);
    }

调用 RunWorkerAsync() 的函数

private void btnFind_Click(object sender, EventArgs e)
    {

        //start backgroundworker and clear friends and search lists
        pctProgressBar.Visible = true;

        lstFriend.Items.Clear();
        lstSearchTweetList.Items.Clear();
        lstFollowerChecker.Items.Clear();
        lstFollowingChecker.Items.Clear();
        lstFriend.Items.Clear();
        lstSearchTweetList.Items.Clear();

        if (txtTweetAmount.Text == "")
        {
            txtTweetAmount.Text = "20";
        }

        backgroundWorker1.RunWorkerAsync();

    }

我的问题是奇怪的无法解释的错误似乎仍然随机发生。

如果这是由在后台工作线程中更新的列表引起的,那么如果我不能用它来做密集的工作,那么后台工作人员有什么用

我还将包括朋友帐户的两张图片,因为它可以更好地展示问题,因此处理等内容将被空白。第一个问题是它有时会多次填充一个列表,并且“不关注你的列表”应该只返回一次@Theilluminati 第一次同屏

它再次返回@Theilluminati,但列出了两次。 同屏第二次

还有一个问题,如果我在任何地方运行下面的代码,后台工作人员不会运行,也就是说,它会拉回图片/名称/位置但后台工作人员不运行,如果我尝试在实际中执行backgroundworker 线程,那么列表将不会填充。

   var username = boxUsername.Text;
    var user = User.GetUserFromScreenName(username);
    //string ImageURL = user.ProfileImageUrl;
    //string biggerImageURL = ImageURL.Replace("_normal", "");
    //txtImageURL.Text = biggerImageURL;
    //pctDisplaypicture.ImageLocation = biggerImageURL;
    //txtTwitterID.Text = user.Id.ToString();
    //lblFriendCount.Text = "(" + user.FollowersCount + ")";

任何帮助都将不胜感激,如果它无法从 UI 线程卸载工作,我现在很难看到 Backgroundworker 的使用,对不起,很长的帖子,感谢阅读。

修复尝试

我在任务运行时禁用了查找按钮,但仍然出现同样的问题。

一旦任务完成一次,我也尝试使用 if(working.Cancellationpending == true) 来打破循环。

我已将列表 foreach 循环分别更改为以下内容,并将用户名作为变量传递,而不是从控件中提取,问题似乎变得更糟,现在根本没有列表填充。

lstSearchTweetList.Invoke((MethodInvoker)delegate
                {
                    lstSearchTweetList.Items.Add(userTweet.Text).SubItems.Add(userTweet.CreatedAt.ToString());
                });

 backgroundWorker1.RunWorkerAsync(boxUsername.Text);

var username = e.Argument.ToString();

我已经尝试了这两个答案作为解决方案,并且仍然导致相同的问题具有不同的严重性,我仍然遇到这样的问题,即取消注释代码以检索名称/图片等仍然会阻止后台工作程序运行。不管它是从哪里跑来的。

4

2 回答 2

4

您可能需要在您尝试在后台线程上更新的列表控件上使用 Invoke 方法,如下所示:

    string  screenName = "@" + friend.ScreenName;
    lstFriend.Invoke((MethodInvoker)delegate {
           lstFriend.Items.Add(screenName); // runs on UI thread
    });

多线程可能遇到的一个问题是,当您尝试从多个线程访问共享资源(集合、文件等)时,可能会发生死锁以及竞争条件。为了安全地执行此操作,在这种情况下将创建一个锁定对象并锁定正在访问共享资源的代码。这样,一次只能访问一个资源。

    //defined globally
    object _MyLockingObject = new object();

并在某种方法中锁定一个列表:

    lock(_MyLockingObject)
    {
       myList.Add(item);
    }
于 2014-03-16T15:57:29.670 回答
0

您违反了 Windows GUI 编程中的一条基本规则:永远不要从与创建控件的线程不同的线程访问控件。当您违反此规则时,会发生不好的事情;)

通过传递用户名值 backgroundWorker1.RunWorkerAsync(boxUsername.Text);,并通过读取它e.Arguments as string

然后,您需要使用BeginInvoke与 UI 控件进行交互。理想情况下,您应该优化此 lambda 以暂停控件的布局,一次调用替换整个项目列表,并恢复控件的布局。

 // execute on the UI thread
 this.BeginInvoke((Action)(() =>
 {
   lstFriend.Items.Add("@" + friend.ScreenName);
 }), null);

我会Control.BeginInvoke在同步Control.Invoke选项上使用异步。似乎没有理由等待控件呈现您的更改。

于 2014-03-16T16:03:42.310 回答