1

我正在尝试做一个从网站读取字符串的程序,并将其发送给另一个。第一个读取字符串的过程可以正常工作,并且将字符串发送到其他工作的函数也可以正常工作,但是从计时器调用此函数时会出现问题..

这是我的代码的一部分:

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.Timers;
using System.Windows.Forms;
using CefSharp;
using CefSharp.WinForms;
using CefSharp.WinForms.Internals;

namespace CodePinger
{
    public partial class Form1 : Form
    {
        public bool login = true;
        private static System.Timers.Timer TimerCheck;
        public Form1()
        {
            InitializeComponent();
            TimerCheck = new System.Timers.Timer(5000);
            TimerCheck.Elapsed += new ElapsedEventHandler(CheckEvent);
            TimerCheck.AutoReset = true;
            CheckForIllegalCrossThreadCalls = false;
            webBrowser1.ScriptErrorsSuppressed = true;
            chromiumWebBrowser1.Load("https://firstwebsite.com");
            webBrowser1.Navigate("http://secondwebsite.com");
        }
        private async void CheckEvent(Object source, ElapsedEventArgs e)
        {
            if (login) { 
                string script = "document.getElementById('SecondSDISP').innerText;";
                JavascriptResponse jr = chromiumWebBrowser1.EvaluateScriptAsync(script).Result;
                if (jr.Success)
                {
                    if (jr.Result.ToString().Contains("01"))
                    {
                        label2.ForeColor = Color.Red;
                        label2.Text = jr.Result.ToString();
                        sendCode(jr.Result.ToString());
                    }
                }
                else
                {
                    label2.ForeColor = Color.Black;
                    label2.Text = "no data";
                }
                label4.Text = timer.ToString();
            }
        }
        private void button2_Click(object sender, EventArgs e)
        {
            TimerCheck.Enabled = true;
            TimerCheck.Start();
            button2.Enabled = false;
        }
        public void sendCode(string code)
        {
            string msg = "";
            if (code == "1") msg = "1 coda";
            else msg = code.ToString() + " code";
            var textarea = webBrowser1.Document.GetElementsByTagName("textarea")[0];
            textarea.InnerHtml = msg;
            textarea.Focus();
            var allsvg = webBrowser1.Document.GetElementsByTagName("svg");
            foreach (HtmlElement svg in allsvg)
            {
                if (svg.GetAttribute("className").Contains("-send"))
                {
                    svg.InvokeMember("click");
                    break;
                }
            }
        }
        private void button4_Click(object sender, EventArgs e)
        {
            sendCode("1");
        }
    }
}

同样在我启动计时器后,如果我单击按钮 4 来测试功能,它可以正常工作。而不是当它从计时器调用时

错误是:

System.InvalidCastException
  HResult=0x80004002
  Message=Specified cast is not valid.
  Source=System.Windows.Forms
  StackTrace:
   at System.Windows.Forms.UnsafeNativeMethods.IHTMLDocument2.GetLocation()
   at System.Windows.Forms.WebBrowser.get_Document()
   at CodePinger.Form1.sendCode(String code) in C:\Users\Flynns82\source\repos\CodePinger\Form1.cs:line 105
   at CodePinger.Form1.<CheckEvent>d__9.MoveNext() in C:\Users\Flynns82\source\repos\CodePinger\Form1.cs:line 63

指示的行是:

105) var textarea = webBrowser1.Document.GetElementsByTagName("textarea")[0];
63) sendCode(jr.Result.ToString());

有人可以解释一下有什么问题吗?

4

1 回答 1

0

您的问题很可能是您试图从非 STA 线程(又名“UI 线程”)访问 WebBrowser

当您使用 button4_click 处理程序时,您的代码在 STA 线程上运行(默认情况下),但是,当回调时间事件处理程序时,它发生在不同的线程(不是 STA 线程)中,因此您将在调用/如果您不“调用”回 STA,则访问 ActiveX/组件(驻留在 STA 线程中)上的属性。

我建议查看以下SO Question: STA vs MTA以获得技术解释。

要解决调用问题,请查看以下SO 问题:自动化 InvokeRequired 代码模式

另一方面,浏览器暴露了一个NavigateComplete事件,你不需要花时间检查页面何时加载,只需将自己挂钩到该事件并在导航后等待它,一旦该事件触发,DOM 将是稳定的。

于 2021-01-11T16:48:59.300 回答