我正在使用WatiN来解析我的网站.我有一个启动过程的按钮.我打开一个浏览器窗口并导航到我需要去的地方,然后创建一个调用DoWork方法的新任务.
我的问题是,如果我在DoWork结束时调用一个新方法来做其他事情,当我尝试让程序导航我的网站时,我会得到奇怪的结果,但是,如果我不从DoWork调用这个新方法而只是挂钩单击按钮的新方法一切正常.所以我的问题是,我没有从背景处理方法中正确地调用我的新方法,Dowork?
码:
IE browser = new IE("http://www.mywebsite.com/");
string startYear;
string endYear;
int NumRows;
Task myThread;
public Form1()
{
InitializeComponent();
}
private void Start_Click(object sender, EventArgs e)
{
startYear = txtStartYear.Text;
endYear = txtEndYear.Text;
//website navigation work removed for brevity
browser.Button(Find.ById("ContentPlaceHolder1_btnApplyFilter")).Click();
int numRows = browser.Div(Find.ById("scroller1")).Table(Find.First()).TableRows.Count -1;
NumRows = numRows;
lblTotalRows.Text = numRows.ToString();
myThread = Task.Factory.StartNew(() => DoWork());
}
public void DoWork()
{
List<string> myList = new List<string>(NumRows);
txtStartYear.Text = startYear;
txtEndYear.Text = endYear;
for (int i = 1; i < NumRows; i++)
{
TableRow newTable = browser.Div(Find.ById("scroller1")).Table(Find.First()).TableRows[i];
string coll = string.Format("{0},{1},{2},{3},{4}", newTable.TableCells[0].Text, newTable.TableCells[1].Text, newTable.TableCells[2].Text, newTable.TableCells[3].Text, newTable.TableCells[4].Text);
myList.Add(coll);
label1.Invoke((MethodInvoker)delegate
{
label1.Text = i.ToString();
});
}
//database work removed for brevity.
browser.Button(Find.ById("btnFilter")).Click();
newMethod();
}
public void newMethod()
{
int start = int.Parse(startYear);
start++;
startYear = start.ToString();
int end = int.Parse(endYear);
end++;
endYear = end.ToString();
browser.SelectList(Find.ById("selStartYear")).SelectByValue(startYear);
browser.SelectList(Find.ById("selEndYear")).SelectByValue(endYear);
//removed for brevity
}
}
重申一下,如果我从Dowork调用newMethod行browser.SelectList(Find.ById(“selStartYear”)).SelectByValue(startYear)行为不正常,但如果我从Dowork删除对newMethod的调用并挂起newMethod一个按钮它工作正常.我想知道它是否与DoWork作为后台任务有关?
当我说它表现不正常时,我的意思是当你从下拉列表中选择一个项目时,页面会自动回发,但是上面的代码行会选择它但页面不会回发,这不应该是可能.如果我不在DoWork中调用该方法,我就没有这个问题.
最佳答案 您正在从非UI线程修改UI元素.你已经有了通过Control.Invoke在DoWork中处理的代码 – 你需要为newMethod做同样的事情.在UI线程中调用整个方法可能最简单:
// At the end of DoWork
Action action = newMethod;
label.BeginInvoke(action);
(我正在使用label.BeginInvoke,因为我不确定浏览器本身是否是一个“正常”控件 – 但是使用label仍然会到达正确的线程.如果browser.BeginInvoke编译,那就更清楚了.)