WebBrowser oraz Thread

WebBrowser oraz Thread
SA
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 11 lat
  • Postów:11
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ą:

Kopiuj
                        LikeThread = new Thread(() => Like(webBrowser1));
                        LikeThread.Start();
Kopiuj
        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:

Kopiuj
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: 

edytowany 1x, ostatnio: Sahee
SA
  • Rejestracja:około 21 lat
  • Ostatnio:ponad 9 lat
  • Postów:513
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.

Kopiuj
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
SA
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 11 lat
  • Postów:11
0

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

n0name_l
@somekind to raczej wymyslony jezyk przez autora posta.
SA
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 11 lat
  • Postów:11
0

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

DU
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 11 lat
  • Postów:25
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.

SA
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 11 lat
  • Postów:11
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 ;)

Zarejestruj się i dołącz do największej społeczności programistów w Polsce.

Otrzymaj wsparcie, dziel się wiedzą i rozwijaj swoje umiejętności z najlepszymi.