WebBrowser oraz Thread

0

Witam, próbuję uruchomić klikanie w divy przy użyciu webbrowser w osobnym thredzie lecz otrzymuje błąd podczas gdy pętla foreach zaczyna bawić się z przeglądarką kod:

Blad pojawia się w linijce: foreach (String s in u.Document.All)
A funkcję Like odpalam za pomocą:

                        LikeThread = new Thread(() => Like(webBrowser1));
                        LikeThread.Start();
        public void Like(WebBrowser u)
        {
                    

            do
            {
               
                
                  iloscdorefa++;
                    //lajkowanie
                  
                      foreach (String s in u.Document.All)
                      {
                          MessageBox.Show("" + s);
                      }

                      HtmlElementCollection lol = u.Document.GetElementsByTagName("like hintable");

                      foreach (HtmlElement ee in lol)
                          if (ee.GetAttribute("className") == "like hintable")
                          {
                              ee.InvokeMember("click");
                              break;
                          }
                      //foreach (HtmlElement ee in webBrowser1.Document.All)
                      //    if (ee.GetAttribute("className") == "like hintable")
                      //    {
                      //        ee.InvokeMember("click");
                      //        break;
                      //    }

                      //rozwijanie
                      if (iloscdorefa == 10)
                      {

                          foreach (HtmlElement ee in u.Document.All)
                              if (ee.GetAttribute("className") == "submit-button-more submit-button-more-active")
                                  ee.InvokeMember("click");

                          //Thread.Sleep(rnd.Next(500, 2000));
                          //foreach (HtmlElement ee in webBrowser1.Document.All)
                          //    if (ee.GetAttribute("className") == "submit-button-more submit-button-more-active")
                          //        ee.InvokeMember("click");

                          iloscdorefa = 0;
                      }

                    Thread.Sleep(rnd.Next(1000, 3500));
                
            }while (Dziala);

          }

Blad:

System.InvalidCastException was unhandled
  Message=Określone rzutowanie jest nieprawidłowe.
  Source=System.Windows.Forms
  StackTrace:
       w System.Windows.Forms.UnsafeNativeMethods.IHTMLDocument2.GetLocation()
       w System.Windows.Forms.WebBrowser.get_Document()
       w AutomatyczneLajki.Form1.Like(WebBrowser u) w C:\Szkola\AutomatyczneLajki\AutomatyczneLajki\Form1.cs:wiersz 38
       w AutomatyczneLajki.Form1.<butLikeAll_Click>b__0() w C:\Szkola\AutomatyczneLajki\AutomatyczneLajki\Form1.cs:wiersz 124
       w System.Threading.ThreadHelper.ThreadStart_Context(Object state)
       w System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean ignoreSyncCtx)
       w System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
       w System.Threading.ThreadHelper.ThreadStart()
  InnerException: 

0

Problem polega na tym, że browser przechowuje część danych w obszarze pośrednio dostępnym dla jednego tylko wątku (thread local storage). Wątek, który uruchomiłeś, nie ma zainicjowanych tych danych (wskaźników), a browser, ufnie traktując owe zerowe wskaźniki, powoduje co widać u Ciebie.

Wyjścia są conajmniej dwa. Pierwszy i najprostszy w natywnych, 32-bitowych programach, jest zapisanie wskaźnika [fs:0x18] będąc w wątku tworzącym browsera, i podmiana go w wątku, który utworzyłeś. Wskaźnik ten zwraca np. natywna funkcja NtCurrentTeb(), ale nie ma systemowej funkcji ustawiającej ten wskaźnik.

Drugi sposób to skopiowanie tych ukrytych danych z thread local storage, z jednego wątku do innego. Po zainicjowaniu browsera pobierz index pierwszego wolnego slotu TLS funkcją TlsAlloc() do zmiennej TlsCount, stwórz tablicę wskaźników (dostępną dla wszystkich wątków w programie) o TlsCount elementach i wypełnij ją wartościami zwróconymi z funkcji TlsGetValue(index), gdzie index przyjmuje wartości od zera do TlsCount-1. Zwolnij slot który pobrałeś na początku, funkcją TlsFree(TlsCount), ale pozostaw zmienną TlsCount niezmienioną.

Teraz możesz uruchomić swój wątek, a w nim, na początku, utwórz identyczną pętlę jak na początku, od zera do TlsCount-1, ale zamiast TlsGetValue, użyj TlsSetValue(index, tablica[index]).

To powinno wystarczyć, by browser działał stabilnie. Sposób jest trochę naiwny, bo kopiuje wszystkie dane TLS, od zerowego do ostatniego. Zanim browser zostanie utworzony, jakieś dane TLS mogą już istnieć. Funkcja TlsAlloc zwróci ich ilość (zmienna TlsStart), więc możesz pominąc te dane w obu pętlach zaczynając nie od zera, a od TlsStart.

TlsStart = TlsAlloc() // pseudokod, TlsStart, TlsCount i TlsTab to globalne zmienne
TlsFree(TlsStart)

browser = new WebBrowser // ewentualnie otwarcie okna z browserem
browser.Navigate("about:blank") // niech się wszystko zainicjuje
// tu warto wstawić coś w stylu DoEvents(), dopóki browser.complete jest false

TlsCount = TlsAlloc()
TlsFree(TlsCount)

TlsTab = new array [TlsCount] of pointers // zapiszmy wszystko
for index = 0 to TlsCount-1
   TlsTab[index] = TlsGetValue(index)
next


public void Like(WebBrowser u) // wątek
{
	for index = TlsStart to TlsCount-1 // ustawmy tylko to, co z browsera
		TlsSetValue(index, TlsTab[index])
	next
// dalej już bez zmian
0

U mnie nie ma żadnych funkcji których podałeś. A poza tym to składnia z VB? :/

0

Jest może jakaś inna możliwość napisania tego lub może inne api przeglądarki które to umożliwi?

0

Nie wiem czy tk kestia api (brak funkcji) ale jesli chodzi o watki to to co probujesz wykonac jwst rownowazne z otwarciem przegladarki (jedna kontrolkaa) i klikaniem wieloma kursorami na jednej zakladce... jezeli oczywscie byla by mozliwosc obslugi wielu myszek. Do tego sluzyly by Ci watki.
Poniewasz kontrolka i tak tworzy watek do obslugi wgrywania danych mozesz to robic na zdarzeniach... po wgraniu strony, chyba documentloaded dobierasz sie w obsludze zdarzenia do zawartosci i klikasz to wszytko bedzie w watku Gui i nie bedziesz potrzebowal swojego watku.

I tak mi sie jeszcze z api przypomnialo... czy uzywasz Forms czy WPF tu moze byc rozwiazanie brakujacej funkcji.

0

Może sobie z tym nie poradziłem, lecz rozwiązałem to w inny sposób. Zamiast nowego thredu odświerzam aplikacje za pomocą Application.DoEvents() w pentli w której klikam ;)

1 użytkowników online, w tym zalogowanych: 0, gości: 1