Pojęcie współprogramu ma dwie odmienne definicje. Obie definicje zgodnie stwierdzają, że współprogram cechuje się posiadaniem ciągu instrukcji do wykonania i ponadto możliwością zawieszania wykonywania jednego współprogramu A i przenoszenia wykonywania do innego współprogramu B. W szczególności można wznowić pracę zawieszonego współprogramu A, a wykonywanie będzie podjęte w miejscu, w którym zostało zawieszone. Tym co różni obie definicje jest zdolność współpracy z rekurencyjnymi procedurami (W językach programowania funkcyjnego koncepcja współprogramu istnieje pod postacią kontynuacji – pojęcia wprowadzonego niemal równocześnie ze współprogramami).
Obiekt współprogramu jest quasi-wątkiem. Tak jak wątek ma ciąg instrukcji do wykonania, w odróżnieniu od wątków obiekty współprogramów nie działają równolegle. Jest niezmiennikiem systemu współprogramów to, że w każdej chwili, dokładnie jeden obiekt współprogramu wykonuje swoje instrukcje:
W literaturze znaleźć można termin włókno (ang. fiber) dla odróżnienia od wątku (ang.thread).
moduły coroutine, tj. współprogramów. Składnia współprogramu różni się od składni klasy tym,
że zamiast słowa class piszemy coroutine,
i co ważniejsze, wewnątrz takiego modułu wolno używać
instrukcji atomowych attach oraz detach. Instrukcje takie
mogą się też pojawiać wewnątrz metod zadeklarowanych we współprogramie.
Stwarza to nowe i interesujące możliwości
współpracy współprogramów i rekursji.
"Subroutines are special cases of ... coroutines." –Donald Knuth[1].
W assemblerze od dawna występuje pojęcie podprogramu – nie należy go mylić z pojęciem procedury. Podprogram istnieje w kodzie programu i ma co najwyżej jedną instancję. Nie jest możliwe rekurencyjne wykonywanie podprogramów.
Uwagi. 1°.Mamy tu do czynienia z dynamicznym systemem współprogramów. Wyrażenia generujące postaci
new Traverser(...) umożliwiają stworzenie wielu
obiektów typu współprogram Traverser. 2°. Obiekt współprogramu Traverser wywołuje metodę traverse(). Łańcuch dynamiczny tego obiektu wydłuża się o rekord aktywacji metody traverse. Łańcuch taki może być tak długi jak gałąź odwiedzanego drzewa BST. 3°. Instrukcje attach()
oraz detach mogą wystąpić nie tylko w ciągu instrukcji
współprogramu, lecz także w metodach współprogramu. 4°. Przełączenie
wykonywania odbywa się pomiędzy łańcuchami
dynamicznymi współprogramów.
Uwagi. 1°. Ten system współprogramów jest statyczny: zawiera dwa współprogramy.
Deklaracja coroutine automatycznie tworzy obiekty typu produkuj i konsumuj. 2°. W tym systemie są dwa punkty wejścia. Otrzymany graf składa się z dwu współprogramów i dwu przełączeń yield.
W latach 60 XX wieku współprogram był fragmentem kodu napisanego w assemblerze. wątki o następujących własnościach:
dokładnie jeden współprogram wykonuje swoje instrukcje, tzn. jest aktywny,
współprogram aktywny może przejść w stan pasywny wskazując przy tym na inny wątek, który ma być uaktywniony,
współprogram uaktywniony w efekcie wykonania instrukcji (w dotychczas aktywnym współprogramie ) kontynuuje wykonywanie instrukcji od odpowiedniego punktu wejścia, dokładniej: pierwsze uruchomienie instrukcji wątku współprogramu spowoduje wykonanie pierwszej instrukcji wątku, każda następna instrukcja wznawiająca wykonywanie wątku współprogramu rozpoczyna wykonywanie instrukcji od punktu wejścia wyznaczonego przez ostatnio wykonaną w nim instrukcję
Zasada działania współprogramów
System współprogramów można nazwać systemem quasi-współbieżnym. Nazwa ta jest uzasadniona dwojako: liczne przykłady programów współbieżnych np. producent-konsument, czytelnicy-pisarze itd. zapisane przy pomocy współprogramów okazują się wystarczająco adekwatne do zastosowań. Inny argument wspierający użycie tej nazwy to fakt, że od bardzo dawna stosuje się współprogramy do symulacji systemów, np. w Simuli67, Loglanie'82 i in. Odpowiednia klasa Simulation dostarcza klasę wewnętrzną simproces – obiekty klas pochodnych od klasy simproces symulują rzeczywiste procesy np. pacjentów w systemie symulacji epidemii choroby, pojazdy w systemie symulacji ruchu w mieście itp.
Wielu autorów uważa, iż „współprogramy to podprogramy wykonywane w taki sposób, że sterowanie może zostać przekazywane pomiędzy nimi wielokrotnie, przy czym wywołanie danego współprogramu powoduje wykonywanie instrukcji od miejsca ostatniego przerwania wykonania (ostatniego punktu wyjścia), a nie od początku”. Nie jest to całkiem ścisłe. Podprogramy (funkcja, metoda) tworzą rekordy aktywacji. Po opuszczeniu takiego rekordu jest on automatycznie usuwany i nie ma możliwości wznowienia go. Współprogramy wymagają więc wątków i są realizowane jako obiekty odpowiednich klas, a nie jako podprogramy czy procedury. Cytowany wyżej pogląd mocno zawęża koncepcję współprogramów. Co więcej, nie można zapominać, że instrukcjami wątku współprogramu mogą być instrukcje wywołania jego prywatnych metod(procedur). Metody te mogą zawierać instrukcje przekazujące sterowanie z jednego do innego współprogramu. Dokładniej, instrukcja attach przekazując sterowanie z jednego do drugiego współprogramu przenosi je z łańcucha dynamicznego jednego współprogramu do łańcucha dynamicznego innego współprogramu.
Łańcuch dynamiczny współprogramu zawiera obiekt i wątek współprogramu i ponadto, jeśli wykonano instrukcję procedury, to do łańcucha dynamicznego dołączony jest rekord aktywacji procedury. Zakończenie wykonywania instrukcji procedury(metody) powoduje skrócenie łańcucha dynamicznego. Instrukcja wykonana w rekordzie aktywacji procedury powoduje przejście do punktu wejścia w łańcuchu dynamicznym współprogramu x. Punktem wejścia (powrotu) dla dotychczas aktywnego współprogramu jest instrukcja w rekordzie aktywacji procedury występująca bezpośrednio za instrukcją Widać stąd, że liczba punktów wejścia (powrotu) danego współprogramu może być zmienna w czasie i może nie być niczym ograniczona!
Podsumowując, współprogramy to więcej niż obiekty zwyczajnych klas, a mniej niż obiekty aktywne wątków (ang. threads).
Schemat zmian stanów obiektu współprogramu
Współprogramy w języku Loglan 82
instrukcją przenoszenia sterowania z aktywnego współprogramu do drugiego współprogramu jest
moduł współprogramu jest specyficzną klasą (stosuje się słowo ‘coroutine’ zamiast ‘class’),
instrukcja może występować nie tylko w wątku współprogramu, lecz także w prywatnych metodach współprogramu(!),
punktami wejścia do współprogramu są: pierwsza instrukcja wątku współprogramu oraz każda instrukcja następna po instrukcji
można też używać bezparametrowej instrukcji odpowiada instrukcji 'attach(ten współprogram, który ostatnio mnie wezwał)'.
Instrukcje przenoszenia sterowania między współprogramami w różnych językach programowania
Instrukcje przenoszące sterowanie z jednego do drugiego współprogramu to
podobny schemat występuje w wielu sytuacjach np. jeden współprogram zbiera wyniki pomiarów i zapisuje je w bazie danych, a drugi współprogram opracowuje zebrane wyniki, ogólny schemat to producent-konsument,
jeżeli jakaś metoda (funkcja lub procedura) jest wykonywana wielokrotnie z tymi samymi parametrami aktualnymi, to warto utworzyć odpowiednie obiekty współprogramu (tablicę współprogramów) dla każdego zestawu parametrów aktualnych. Następnie każdą instrukcję wywołania procedury zastępujemy odpowiednią instrukcją Zysk może okazać się znaczny, ponieważ wykonanie instrukcji jest znacznie prostsze od tworzenia rekordu aktywacji procedury.
Jeśli współprogramy są klasami wyposażonymi w instrukcję to można tworzyć hierarchie współprogramów wykorzystując dziedziczenie.
główne zastosowanie współprogramów to narzędzia symulacji, takie jak klasy Simulation w Simuli 67 i w Loglanie 82.
Przypisy
↑Donald Ervin Knuth: The Art of Computer Programming. T. 1 Fundamental Algorithms. Addison-Wesley, 1997, s. 193–200. ISBN 0-201-89683-4.
Bibliografia
M.E. Conway. Design of a separable transition-diagram compiler. „Communications of the ACM”, July 1963.
Ole-Johann Dahl, Bjarne Myhrhaug, Kristen Nygaard: Common Base Language (Simula67). Oslo: NCC, 1970.
O.-J. Dahl, A. Wang. Coroutine sequencing in a block structured environment. „BIT”, s. 425–449, 1971.