W swoim pytaniu zadałeś kilka pytań. Rozbiorę je nieco inaczej niż ty. Ale najpierw pozwól mi bezpośrednio odpowiedzieć na pytanie.
Wszyscy chcemy aparatu, który jest lekki, wysokiej jakości i tani, ale jak mówi przysłowie, można uzyskać maksymalnie dwa z tych trzech. Jesteś w tej samej sytuacji tutaj. Chcesz rozwiązania, które jest wydajne, bezpieczne i dzieli kod między ścieżkami synchronicznymi i asynchronicznymi. Dostaniesz tylko dwa z nich.
Pozwól mi wyjaśnić, dlaczego tak jest. Zaczniemy od tego pytania:
Widziałem, że ludzie mówią, że możesz użyć GetAwaiter().GetResult()
metody asynchronicznej i wywołać ją z metody synchronizacji? Czy ten wątek jest bezpieczny we wszystkich scenariuszach?
Pytanie to brzmi: „czy mogę współdzielić ścieżki synchroniczne i asynchroniczne, czyniąc ścieżkę synchroniczną po prostu synchronicznym oczekiwaniem w wersji asynchronicznej?”
Pozwól, że wyrażę się jasno w tej kwestii, ponieważ jest to ważne:
NALEŻY NATYCHMIAST PRZESTAĆ SKORZYSTAĆ Z JAKICHKOLWIEK PORAD OD TYCH LUDZI .
To bardzo zła rada. Synchroniczne pobieranie wyniku z zadania asynchronicznego jest bardzo niebezpieczne, chyba że masz dowody, że zadanie zakończyło się normalnie lub nieprawidłowo .
Powodem, dla którego jest to bardzo zła rada, jest, cóż, rozważ ten scenariusz. Chcesz kosić trawnik, ale ostrze kosiarki jest zepsute. Zdecydowałeś się przestrzegać tego przepływu pracy:
- Zamów nowe ostrze ze strony internetowej. Jest to asynchroniczna operacja o dużym opóźnieniu.
- Synchronicznie czekaj - to znaczy śpij, dopóki nie będziesz mieć ostrza pod ręką .
- Okresowo sprawdzaj skrzynkę pocztową, aby zobaczyć, czy ostrze dotarło.
- Wyjmij ostrze z pudełka. Teraz masz to w ręku.
- Zamontować ostrze w kosiarce.
- Skosić trawnik.
Co się dzieje? Śpisz wiecznie, ponieważ operacja sprawdzania poczty jest teraz uzależniona od czegoś, co dzieje się po jej nadejściu .
Jest to bardzo proste , aby dostać się do tej sytuacji, kiedy synchronicznie czekać na dowolnym zadaniu. To zadanie mogło mieć zaplanowaną pracę w przyszłości wątku, który teraz czeka , a teraz ta przyszłość nigdy nie nadejdzie, ponieważ na nią czekasz.
Jeśli wykonujesz asynchroniczne oczekiwanie, wszystko jest w porządku! Od czasu do czasu sprawdzasz pocztę, a podczas oczekiwania robisz kanapkę, płacisz podatki lub cokolwiek innego; wykonujesz pracę podczas oczekiwania.
Nigdy nie czekaj synchronicznie. Jeśli zadanie zostało wykonane, nie jest konieczne . Jeśli zadanie nie zostanie wykonane, ale zaplanowane uruchomienie z bieżącego wątku, jest nieefektywne, ponieważ bieżący wątek może obsługiwać inną pracę zamiast czekać. Jeśli zadanie nie zostanie wykonane, a harmonogram uruchomi się na bieżącym wątku, zawiesi się, aby synchronicznie czekać. Nie ma żadnego dobrego powodu, aby ponownie synchronicznie czekać, chyba że już wiesz, że zadanie zostało zakończone .
Więcej informacji na ten temat można znaleźć na stronie
https://blog.stephencleary.com/2012/07/dont-block-on-async-code.html
Stephen wyjaśnia scenariusz ze świata rzeczywistego znacznie lepiej niż potrafię.
Rozważmy teraz „inny kierunek”. Czy możemy współdzielić kod, czyniąc wersję asynchroniczną po prostu wykonując wersję synchroniczną w wątku roboczym?
Jest to prawdopodobnie i prawdopodobnie zły pomysł z następujących powodów.
Jest nieefektywny, jeśli operacja synchroniczna działa we / w z dużym opóźnieniem. To zasadniczo zatrudnia pracownika i sprawia, że ten pracownik śpi aż do wykonania zadania. Wątki są niesamowicie drogie . Domyślnie zużywają milion bajtów przestrzeni adresowej, zabierają czas, zajmują zasoby systemu operacyjnego; nie chcesz palić nici wykonując bezużyteczną pracę.
Operacja synchroniczna może nie zostać zapisana jako bezpieczna dla wątków.
Jest to bardziej rozsądna technika, jeśli praca z dużymi opóźnieniami jest związana z procesorem, ale jeśli tak, to prawdopodobnie nie chcesz po prostu przekazywać jej do wątku roboczego. Prawdopodobnie chcesz użyć biblioteki zadań równoległych do zrównoleglenia jej z jak największą liczbą procesorów, prawdopodobnie potrzebujesz logiki anulowania i nie możesz po prostu zmusić wersji synchronicznej do tego wszystkiego, ponieważ wtedy byłaby to już wersja asynchroniczna .
Dalsza lektura; ponownie Stephen wyjaśnia to bardzo wyraźnie:
Dlaczego nie skorzystać z Task.Run:
https://blog.stephencleary.com/2013/11/taskrun-etiquette-examples-using.html
Więcej scenariuszy „rób i nie rób” dla Task.Run:
https://blog.stephencleary.com/2013/11/taskrun-etiquette-examples-dont-use.html
Co nam zatem pozostawia? Obie techniki udostępniania kodu prowadzą do impasu lub znacznej nieefektywności. Doszliśmy do wniosku, że musisz dokonać wyboru. Czy chcesz program, który jest wydajny i poprawny i zachwyca dzwoniącego, czy chcesz zaoszczędzić kilka naciśnięć klawiszy związanych z duplikowaniem niewielkiej ilości kodu między ścieżkami synchronicznymi i asynchronicznymi? Obawiam się, że nie dostaniesz obu.