|
Polski Portal COLOBOTa COLOBOT Polish Portal |
|
Zasoby - Rozbudowany program bota obsługującego hutę
pipok - 25-06-2009, 16:55 Temat postu: Rozbudowany program bota obsługującego hutę Rozbudowany program dla bota transportowego obsługującego hutę. Jak każdy inny tego rodzaju, działa w cyklu: zanieś rudę do huty, wyjmij kostkę tytanu i odłóż na wolne miejsce, idź po następną bryłę rudy.
Założenia:* drobne ułatwienia: można uruchomić nawet jeśli bot trzyma coś w chwytaku; kiedy dotrze z surowcem do huty, a zastanie w niej gotową kostkę tytanu, prawidłowo się zachowa itp.
* dba o odpowiedni zapas energii, przewidując zużycie na trasie
* optymalnie wykorzystuje elektrownie, także jeśli jest więcej niż jedna
* podjeżdża blisko elektrowni i jeśli nie może zająć w niej miejsca, to czeka w pobliżu aż będzie dostępna
* sam planuje lądowania techniczne dla ostudzenia silników (jeśli latający)
* wybiera bezpieczne miejsca do lądowania: suche i w miarę możliwości płaskie (dla uniknięcia ryzyka ześlizgnięcia się do wody lub w przepaść). W zasadzie program można uruchomić na więcej niż jednym bocie. Nie pamiętam, czy testowałem taką sytuację, bo ostatnie poprawki robiłem w 2005 roku.
pipok - 26-06-2009, 07:59 Temat postu: Re: Rozbudowany program bota obsługującego hutę Sam skomentuję, skoro nikogo nie korci
Ułatwiajmy sobie życie
Główny program wygląda tak: Kod: | extern void object::ZbierajTytan()
{
int surowiec = TitaniumOre;
int przetwornia = Converter;
int produkt = Titanium;
object rzecz, budynek;
point tutaj;
float odl;
// co trzymasz w łapie, robocie?
WyrzucInne(surowiec);
while (true) { // powtarzaj w nieskończoność
WezSurowiec(surowiec);
// znajdź przetwórnię i idź tam
budynek = radar(przetwornia);
// opróznij przetwórnię, jeśli jest w niej produkt
OproznijPrzetwornie(surowiec, produkt, przetwornia);
// upewnij się, że masz w łapce surowiec, zostaw go w przetwórni
WezSurowiec(surowiec);
Idz(budynek.position); drop();
// poczekaj na produkt i odłóż go
ObsluzProdukt(surowiec, produkt, przetwornia);
}
} | Pierwsze trzy linijki ciała programu (deklaracje i inicjowanie zmiennych typu int) mają za zadanie ułatwić przemianę zbieracza tytanu w zbieracza uranu.
Dalsza część to trzy inne deklaracje, i... do roboty... jednokrotne wywołanie funkcji WyrzucInne oraz pętla zasadniczej pracy. Trzy z czterech użytych funkcji własnych występują w kodzie jednokrotnie, więc dlaczego ich treść nie jest po prostu wpisana w główny program, po co zrobiłem funkcje z kawałków kodu? Program jest dość obszerny (ponad 10KB) -- podzielenie go na części ułatwia zrozumienie jego działania, testowanie i modyfikowanie. Wyobraźmy sobie zagnieżdżenie sześciu dlugaśnych pętli, jedna w drugiej... Dla początkujących programistów: jeśli kod jest dłuższy niż dwa-trzy ekrany, podziel go na kawałki.
Innego rodzaju ułatwienie występuje w pierwszej własnej funkcji. Pierwszym działaniem bota w cyklu pracy jest znalezienie, dojazd i wzięcie rudy tytanu. Bardzo irytowało mnie, że przed uruchomieniem programu zbieracza musiałem upewnić się, że na pusty chwytak. W przeciwnym razie jechał sobie do najbliższej bryły i próbował ją podnieść mając zajęte "ręce". Na dodatek błąd pojawia się dopiero przy próbie podniesienia! Czy powinien mieć wolne łapki? Odpowiedni fragment programu Kempiora wygląda tak: Kod: | while( load == null )
{
item = radar(TitaniumOre);
[ ... ]
goto(item.position);
grab();
}
item = radar(Converter);
while ( item == null );
while(goto(item.position) != 0 )
| Jeśli bot nic nie niesie, to ma iść i zabrać rudę tytanu. Następnie udaje się do huty. Co się stanie, jeśli przed uruchomieniem programu bot już trzyma rudę? Wtedy nie pójdzie po rudę, tylko od razu do huty. W porządku. Ale zrobi to samo, jeśli w chwytaku ma coś innego: kostkę tytanu, ogniwo, uran... Nie uda się po rudę, ale z tym, co trzyma, od razu pójdzie do huty. I tam zostawi, czekając aż huta przerobi to na kostkę
Warto upewnić się przed rozpoczęciem pracy, czy coś trzyma. Ale... Jeśli tak, to powinna to być ruda. Inne przedmioty powinien odłożyć na bok. Kod: | void object::WyrzucInne(int surowiec)
{
if (load != null) {
if (load.category != surowiec) {
// trzyma coś, ale nie surowiec!
// zostaw to gdzieś w pobliżu
goto(space()); drop();
}
}
} |
pipok - 26-06-2009, 09:50 Temat postu: Re: Rozbudowany program bota obsługującego hutę Kolejny klocek to funkcja nakazująca botowi udanie się w miejsce, gdzie leży najbliższa bryła tytanu i zabranie jej:
Kod: | void object::WezSurowiec(int surowiec)
{
int err;
object rzecz;
if (load == null) { // puste łapki - weź surowiec
// idź do surowca i podnieś go
do {
rzecz = radar(surowiec); // znajdź
Idz(rzecz.position); // idź do surowca
// podnieś go
errmode(0); err=grab(); errmode(1);
} while (err != 0);
}
} | Co tutaj robi instrukcja warunkowa? Kod: | if (load == null) { // puste łapki - weź surowiec | Upraszcza kod. Mamy jedną funkcję do znajdowania i zabierania surowca, a na samym początku być może bot trzyma już w chwytaku rudę - wówczas nie musi nigdzie chodzić i niczego brać. Poza sprawdzeniem bot nic wówczas nie robi. Z kolei Kod: | ierrmode(0); err=grab(); errmode(1); | obsługuje sytuację, kiedy z jakiegoś powodu podniesienie surowca się nie powiodło. Szukanie, ewentualna podróż i próba zabrania są powtarzanie póki nie uda się pomyślnie pochwycić surowca. W szczególności błąd chwytania pojawi się, kiedy ktoś już zabrał surowiec podczas gdy nasz bot do niego podróżował. Dzięki umieszczeniu "szukania" surowca wewnątrz pętli, bot spróbuje wtedy dotrzeć do innej, położonej najbliżej niego bryły.
Użyta do przemieszczenia bota funkcja Idz(), przyjmująca parametr typu point, odpowiada wbudowanej funkcji goto() użytej z jednym parametrem. Dodatkowo zapewnia obsługę podładowywania ogniwa i studzenia silników w razie potrzeby.
Kempior - 26-06-2009, 15:21
Kod: | while( load == null )
{
item = radar(TitaniumOre);
[ ... ]
goto(item.position);
grab();
} |
Ten kod miał sprawić że robot nie pojedzie bez rudy do huty.
pipok - 26-06-2009, 16:31
Kempior napisał/a: | Ten kod miał sprawić że robot nie pojedzie bez rudy do huty. | Ale jeśli nie uda mu się chwycić rudy, to nie powtórzy zawartości pętli, tylko przerwie działanie z powodu błędu, prawda?
pipok - 26-06-2009, 16:53
Kiedy bot ma już w chwytaku bryłę rudy tytanu (surowiec), powinien udać się z nią do huty (przetwórni), żeby ją tam umieścić. Co jednak, jeśli w przetwórni leży już wyprodukowana kostka tytanu?
Jak sprawdzić, czy w hucie leży tytan? Wystarczy sprawdzić, czy w promieniu 3 metrów od centrum tego budynku znajduje się kostka tytanu. Współrzędne budynku odnalezionego funkcją radar są współrzędnymi środka budowli. Niestety, funkcja radar daje możliwość przeszukania zadanego wycinka koła, z ograniczeniem odległości, ale centrum poszukiwań stanowi nasz bot. Dlatego poszukamy kostki tytanu znajdującej się w stosunku do bota w odległości takiej jak budynek huty, plus minus 3 metry. Ponadto, ta kostka musi być z pozycji bota widoczna w tym samym kierunku, co budynek, z dokładnością do powiedzmy 5 stopni.
Stąd wziął się występujący w funkcji fragment:
Kod: | budynek = radar(przetwornia);
odl=distance(position, budynek.position);
rzecz=radar(produkt,direction(budynek.position),5, odl-3,odl+3); | Alternatywną metoda sprawdzenia jest szukanie w pętli kolejnych kostek tytanu i sprawdzanie, czy odległość miedzy kostką a hutą jest mniejsza od 3 metry.
Jeżeli w hucie (przetwórni) jest kostka tytanu, to bot powinien pojechać na jakieś wolne miejsce w pobliżu huty, odłożyć trzymaną rudę, a następnie wyciągnąć z huty kostkę tytanu i położyć ją w miejscu składowania kostek (produktów). Po to, żeby w kolejnym kroku móc swoją rude umieścić w opróżnionym budynku huty. Całość wygląda więc tak: Kod: | void object::OproznijPrzetwornie(int surowiec, int produkt, int przetwornia)
{
object rzecz, budynek;
float odl;
int err;
// czy w przetwórni nie ma aby gotowego produktu? - wyjmij go
budynek = radar(przetwornia);
odl=distance(position, budynek.position);
rzecz=radar(produkt,direction(budynek.position),5, odl-3,odl+3);
if (rzecz != null) {
// przetwórnia zajęta, leży w niej gotowy produkt
if (load != null) {
// trzyma coś, trzeba to odłożyć, żeby wyjąć produkt
// odłóz w poblizu przetwórni
Idz(space(budynek.position,5,20,2)); drop();
}
Idz(budynek.position);
errmode(0); err=grab(); errmode(1); // wyjmij produkt
if (err == 0) { OdlozProdukt(produkt); }
}
} | Warto zauważyć, że próba podniesienia kostki z huty nie jest zamknięta w pętli. Dlaczego? Ponieważ sprawdzenia zajętości huty dokonaliśmy zanim do niej bot wyruszył. Kiedy już dotarł w pobliże huty, być może kostki już nie ma w środku (astronauta ją zabrał?).
Z drugiej strony, jak zauważył Kempior, powstaje problem, gdy równocześnie z naszym obsługuje inny bot. Kiedy huta jest pusta w chwili, gdy bot sprawdza (ma rudę, ale jeszcze nie ruszył z nią w drogę), może się zdarzyć, że zanim dotrze na miejsce, drugi bot włoży do huty rudę. A ponownego sprawdzenia już nie ma...
pipok - 26-06-2009, 16:59
Kempior napisał/a: | Ten kod miał sprawić że robot nie pojedzie bez rudy do huty. | Dopiero teraz zwróciłem uwagę, że wyłączyłeś globalnie, na samym początku programu, zatrzymywanie działania w przypadku błędu.
colobotwymiiata - 17-08-2009, 16:04
Istnieje też funkcja "search", która nie jest opisana w oficjalnym podręczniku. Szuka obiektu jak "radar", ale można ustalić pozycję.
pipok - 25-08-2009, 15:30
colobotwymiiata napisał/a: | Istnieje też funkcja "search", która nie jest opisana w oficjalnym podręczniku. Szuka obiektu jak "radar", ale można ustalić pozycję. | Słusznie. Zupełnie o niej zapomniałem, a jest bardzo przydatna w podobnych sytuacjach Cytat: | search ( kategoria, pozycja );
Znajduje obiekt o określonej kategorii, będący najbliżej podanej pozycji. |
|
|