Ash*_*Ash 8 c# wpf user-interface
好吧,我已经尝试了几种让它工作的方法,后台工作者,Dispatcher.Invoke,在被调用类中的线程,似乎没有任何工作.到目前为止,最好的解决方案是一个Extension方法,它调用控件的调用.此外,我已经尝试避免通过我的事件类传递标签的数据,只是在我的处理代码中调用,但这没有任何区别.
关于后台组件,我不断得到异常,说后台工作人员很忙,所以我多次实例化了这个类,但是一旦整个操作完成,标签才会明显改变.
我删除了以前的代码,这里是相关的一切,因为它似乎很难解决.
被调用的方法
private void TestUris()
{
string text = new TextRange(rtxturis.Document.ContentStart, rtxturis.Document.ContentEnd).Text;
string[] lines = Regex.Split(text.Remove(text.Length - 2), "\r\n");
foreach (string uri in lines)
{
SafeUpdateStatusText(uri);
bool result;
string modUri;
if (!uri.Contains("http://"))
{
modUri = uri;
result = StoreData.LinkUriExists(new Uri("http://" + modUri));
}
else
{
modUri = uri.Substring(7);
result = StoreData.LinkUriExists(new Uri(uri));
}
if (!result)
{
Yahoo yahoo = new Yahoo();
yahoo.Status.Sending += (StatusChange);
uint yahooResult = 0;
yahooResult = yahoo.ReturnLinkCount(modUri);
if (yahooResult > 1000 )
{ results.Add(new ScrapeDetails(Guid.NewGuid(), modUri, 1000, "Will be processed", true)); }
else
{ results.Add(new ScrapeDetails(Guid.NewGuid(), modUri, (int)yahooResult, "Insufficient backlinks", false)); }
}
else
{
results.Add(new ScrapeDetails(Guid.NewGuid(), modUri, 0, "Previously been processed", false));
}
}
foreach (var record in results)
{
dgvresults.Items.Add(record);
}
EnableStartButton();
}
Run Code Online (Sandbox Code Playgroud)
雅虎班
public class Yahoo
{
/// <summary>
/// Returns the amount of links each Uri has.
/// </summary>
public uint ReturnLinkCount(string uri)
{
string html;
Status.Update(uri, false); //this is where the status is called
try
{
html = client.DownloadString(string.Format("http://siteexplorer.search.yahoo.com/search?p=http%3A%2F%2F{0}&fr=sfp&bwm=i", uri));
}
catch (WebException ex)
{
ProcessError(ex.ToString());
return 0;
}
return (LinkNumber(html));
}
Run Code Online (Sandbox Code Playgroud)
状态类
public class StatusEventArgs : EventArgs
{
private string _message;
private bool _isidle;
public StatusEventArgs(string message, bool isidle)
{
this._message = message;
this._isidle = isidle;
}
public bool IsIdle
{
get { return _isidle; }
}
public string Message
{
get { return _message; }
}
}
public class Status
{
public Status()
{
}
// Declaring an event, with a custom event arguments class
public event EventHandler<StatusEventArgs> Sending;
// Some method to fire the event.
public void Update(string message, bool isIdle)
{
StatusEventArgs msg = new StatusEventArgs(message, isIdle);
OnUpdate(msg);
}
// The method that invokes the event.
protected virtual void OnUpdate(StatusEventArgs e)
{
EventHandler<StatusEventArgs> handler = Sending;
if (handler != null)
{
handler(this, e);
}
}
}
Run Code Online (Sandbox Code Playgroud)
更改标签内容的方法
private void StatusChange(object sender, StatusEventArgs e)
{
if(!e.IsIdle)
{
lblstatus.Content = e.Message;
lblstatus.Foreground = StatusColors.Green;
lblstatus.Refresh();
}
else
{
lblstatus.Content = e.Message;
lblstatus.Foreground = StatusColors.Grey;
lblstatus.Refresh();
}
}
Run Code Online (Sandbox Code Playgroud)
Refresh静态方法调用:
public static class ExtensionMethods
{
private static Action EmptyDelegate = delegate() { };
public static void Refresh(this UIElement uiElement)
{
uiElement.Dispatcher.Invoke(DispatcherPriority.Render , EmptyDelegate);
}
Run Code Online (Sandbox Code Playgroud)
另一个编辑:盯着我的代码一段时间,我意识到,foreach循环会很快执行,花费时间的操作,是
yahooResult = yahoo.ReturnLinkCount(modUri);
Run Code Online (Sandbox Code Playgroud)
因此,我已经声明了状态类(处理事件并调用标签等)并对其进行了子类化.我得到了更好的结果,虽然它仍然感觉随机,有时我看到一些标签更新,有时即使完全相同的URI传递,所以很奇怪.
小智 8
我希望有某事.很有帮助...
private void button1_Click(object sender, RoutedEventArgs e)
{
ThreadPool.QueueUserWorkItem(o =>
{
int result = 0;
for (int i = 0; i < 9999999; i++)
{
result++;
Dispatcher.BeginInvoke(new Action(() =>
{
this.label1.Content = result;
}));
Thread.Sleep(1);
}
});
}
Run Code Online (Sandbox Code Playgroud)
解决了,是的,WOOHOOOOOOOO 3天的测试、测试、测试。
我决定使用上面的扩展方法和一个简单的 for 循环来启动一个新项目来测试 UI 更新功能。我开始测试不同的 DispatchPrioraties (测试了所有这些)。
奇怪的是,我发现最高优先级的情况更差,例如使用 Send 根本没有更新标签,而 Render 平均更新了两次。这是我在尝试不同优先级时遇到的奇怪行为。我发现背景:
枚举值为4。操作在所有其他非空闲操作完成后进行处理。
现在这听起来正是我不想要的,因为显然标签应该在处理过程中更新,因此我从未尝试过。我猜测一旦我的一个方法完成,在调用下一个方法之前,UI 就会更新。我发现猜测,但它在两个单独的操作上 100% 一致地正确更新。
谢谢大家。