= Tema - Sheriff of Nottingham =
* Responsabili: Gabriel Țuculină, Florin Mihalache * Deadline: 10.11.2019 - 23:55 * Deadline hard: 17.11.2019 - 23:55 * Data publicării: 19.10.2019 * Data ultimei modificări: 19.10.2019
* familiarizarea cu Java și conceptele de bază ale POO * utilizarea constructorilor și a agregării/moștenirii * folosirea conceptelor de supraîncărcare și suprascriere * dezvoltarea unor abilități de bază de organizare și design orientat-obiect * respectarea unui stil de codare si de comentare * respectarea principiilor SOLID
Se dorește implementarea unei versiuni minimaliste a jocului Sheriff of Nottingham, utilizând conceptele POO. Regulile jocului nu sunt fix cele din boardgame, ele sunt adaptate pentru tema.
Goana după aur a prințului John a mers prea departe! Este imposibil pentru un comerciant să poată să își mai câștige existența, taxele fiind la fel de mari ca pentru un cumpărător obișnuit. Acum, prințului i s-a alăturat și avarul șerif din Nottingham, care verifică pe oricine îi calcă teritoriul, adică verifică toate bunurile ilegale ale unui comerciant și le păstrează după bunul său plac. Este bine că tu îl cunoști mai bine pe acest șerif decât prințul și știi că acea persoană, aparent intimidantă când stă în fața porții orașului, nu refuză o mită generoasă atunci când i se oferă. Tot ce ai nevoie este o strategie bine gandită, iar rolul tău acum este să o găsesti pe cea mai bună.
Pentru acest lucru, se va simula un joc cu maxim 7 jucători, asezati in cerc, fiecare jucător având un anumit tip de strategie. Pentru a obține un rezultat cât mai obiectiv, fiecare jucător va avea două roluri: șerif și comerciant. Fiecare jucător urmăreste să își maximizeze profitul, astfel încât la finalul jocului el să fie declarat jucătorul cu cea mai bună strategie.
Atunci când este în rolul de comerciant, jucătorul încearcă să aducă pe piața din Nottingham cât mai multe bunuri (legale și ilegale) astfel încât să își mărească câștigul. În rolul de șerif, el are posibilitatea de controla sau nu sacul cu bunurile unui comerciant. El poate fi mituit de aceștia sau poate controla comercianții (toți sau niciunul), el la rândul lui urmărind propriul câștig.
Jocul este organizat în runde. La fiecare sub-rundă (o rundă având n sub-runde, n fiind numărul de jucători), doar unul dintre jucători va avea rolul de șerif, iar toti ceilalți vor avea rolul de comercianți, fiecare comerciant extragand cate 10 carti odata. Jocul se încheie atunci când fiecare jucător a avut rolul de șerif de maxim 5 ori. Fiecare jucător va avea în visteria sa la început 80 de monede ce pot fi folosite pentru a mitui un șerif. Numărul maxim de jucatori este de 7.
Fiecare sub-rundă este împărțită în patru etape: crearea sacului, declarararea bunurilor, inspecția și recompletarea bunurilor din mână.
La fiecare sub-rundă, fiecare jucător va deține 10 cărți (bunuri). El va trebui să selecteze între 0 și 8 cărți (fiecare carte poate să fie de orice tip) din cele 10 și să le pună într-un sac pentru a le înmâna șerifului. De asemenea, pentru a fi mai convingător, în funcție de strategia adoptată, el poate adăuga și o sumă de bani (mită) pentru a convinge șeriful să nu îi controleze sacul.
La această etapă, fiecare jucător predă sacul șerifului și își declară bunurile, indicând tipul lor. Indiferent de bunurile din sac, el trebuie să declare un singur tip și poate să spună sau nu adevărul. Bunurile care nu sunt alese, se pot considera arse.
Atunci când șeriful primește sacii de la comercianți, el trebuie sa aleagă pe care dintre comercianti îi va controla (poate să oricâți comercianți, inclusiv pe toți sau pe niciunul), respectând însă următoarele reguli: * dacă a prins un mincinos, atunci el câștigă bani (penalty) pentru fiecare bun ilegal sau nedeclarat. Fiecare bun ilegal sau nedeclarat va fi confiscat. Un bun confiscat va fi adăugat mereu la finalul grămezii (returnata de metoda getAssetIds() din clasa GameInput) cu celelalte cărți rămase în joc. * dacă a controlat un comerciant sincer, atunci el trebuie sa plătească o sumă proporțională cu numărul de bunuri declarate de acesta (assetsCount * penalty) * dacă alege să accepte mita, atunci șeriful va adăuga acea sumă de bani la câștigul propriu, iar comerciantul va trece necontrolat, deci își va putea adăuga toate bunurile din sac pe tarabă. * dacă se controlează sacul implicit se refuză mita dacă există (aceasta întorcându-se la comerciant)
La finalul sub-rundei fiecare jucător își pune bunurile aduse pe taraba și își completează mâna cu urmatoarele 10 cărți (bunuri).
Un comerciant are acces la două tipuri de bunuri (reprezentate sub forma unor cărți de joc): * bunuri legale: Apple, Cheese, Bread, Chicken, Tomato, Corn, Potato, Wine, Salt, Sugar * bunuri de contrabandă (ilegale): Silk, Pepper, Barrel, Beer, Seafood
Fiecare bun are următoarele caracterstici: * profit - suma de bani pe care comerciantul o va câștiga la finalul jocului dacă aduce în Nottingham bunul respectiv; * penalty - suma de bani pe care trebuie să o plătească un comerciant atunci când este inspectat de șerif și prins cu bunuri nedeclarate sau suma de bani pe care trebuie să o plătească șeriful comerciantului pe care l-a inspectat pe nedrept.
De asemenea, la finalul jocului, se calculează pentru fiecare tip de bun legal un clasament în funcție de numărul de bunuri deținute de acel tip. Jucătorul cu cele mai multe bunuri este declarat rege, iar următorul este declarat regină. Mai jos aveți o listă cu punctajele acordate în funcție de tipul bunului și locul în clasament. Dacă un jucator nu reușește să intre în primele 2 locuri din clasament nu primește bonusuri.
Type of Good | King’s Bonus | Queen’s Bonus |
---|---|---|
Apple | 20 | 10 |
Cheese | 19 | 9 |
Bread | 18 | 9 |
Chicken | 17 | 8 |
Tomato | 16 | 7 |
Corn | 15 | 6 |
Potato | 14 | 5 |
Wine | 13 | 4 |
Salt | 12 | 3 |
Sugar | 11 | 2 |
Fiecare jucător va avea o strategie proprie de joc (una din cele descrise mai jos – atât ca șerif cât și ca și comerciant).
În rolul de comerciant, el este jucătorul onest, corect, care spune mereu adevărul. În funcție de cărțile pe care le are în mână, el va cauta tipul de cărți cel mai frecvent. Dacă vor exista mai multe tipuri de cărți cu aceeași frecvență, va selecta tipul de carte care i-ar aduce un profit mai mare. În cazul în care sunt mai multe bunuri cu aceeași frecvență și același profit se alege bunul cu id-ul cel mai mare. Dacă are doar cărți ilegale, el va aduce în sacul său o singură carte, încercând să își minimizeze fapta ilegală (va adăuga totuși cartea care îi aduce cel mai mare profit și va declara că în sac se află mere). Ca șerif, el va controla toți ceilalți jucători, la fiecare sub-rundă, nu va accepta bani de la ceilalți comercianți și deși va risca să rămână fără bani, el va încerca sa împiedice toate bunurile ilegale care ar putea fi aduse în Nottingham. Dacă a rămas fără bani (are o sumă mai mică de 16), acesta nu va mai controla niciun comerciant pana la finalul turului său ca șerif, dar nu va accepta nici mita acestora. Practic își va termina turul ca șerif.
Este un jucător de bază însă, care atunci când este șerif, este ușor de mituit. El inspectează toți comercianții, mai puțin pe cei care îi oferă mită. Când este în rolul de comerciant, el nu oferă mită, deși în rundele pare, după ce aplică strategia de bază, el adaugă în sacul său (dacă nu are deja 8 bunuri în sac) un bun ilegal, alegând bineînțeles cartea ilegală cu profitul cel mai mare. Dacă într-o sub-runda a rundei pare el nu are nicio astfel de carte, el va juca doar strategia de bază in sub-runda respectiva. Prima rundă se consideră impară.
Jucătorul care mituiește este cel care încearcă să îi păcălească pe ceilalți, dar care la un moment dat riscă să se păcălească pe sine. În rolul de comerciant, el va da mită cât timp venitul îi va permite și va încerca mereu sa pună cat mai multe cărți ilegale (maxim 8, cele ce au profitul cel mai mare) si eventual sa completeze cu legale, după următoarele reguli: * va da 5 monede atunci când în sac va adăuga una sau două bunuri ilegale; * va da 10 monede atunci când în sac va adăuga mai mult de două cărți ilegale; * va completa cu cartile legale cele mai valoroase, iar la egalitate, cu cele cu id-ul cel mai mare, in limita permisa de banii pe care ii are * va declara bunurile ca fiind mere; Dacă ar rămâne fără bani, el nu va putea da mită, deci el va juca corect (va juca la fel ca jucătorul cu strategia de bază). De asemenea, dacă nu va avea niciun bun ilegal, el va aplica tot strategia de bază. El isi va completa sacul (ce ii da serifului la verificare) in functie de penalty-ul care s-ar acumula. Totusi, el nu va lua un bun daca i-ar aduce penalty-ul acumulat suma de bani pe 0. Ca șerif, el va verifica mereu doar comerciantul din stânga, respectiv dreapta sa, in aceasta ordine. De la restul comerciantilor va incerca, ulterior, sa ia mita daca acestia i-ar oferi-o, dar altfel nu ii verifica.
Fișierul de input va conține următoarele elemente: * numărul rundelor de joc * numărul de jucători * ordinea jucătorilor: un vector de string-uri de dimensiune maxima 7, cu numele celor 3 strategii. Jucătorul cu strategia ce corespunde primului element din vector va fi jucătorul care va fi primul în rolul de șerif. Exemplu: [“basic”, “bribe”, “greedy”] * numărul de cărți * cărțile din joc: un vector cu elemente ce reprezinta id-uri al bunurilor.
Exemplu de input:
1 2 bribed bribed 20 5 3 5 23 6 9 0 3 0 9 24 8 9 21 8 5 7 1 0 0
Mai jos avem un tabel in care sunt prezentate informatiile despre bunuri.
Id | Asset | Type | Profit | Penalty | Bonus |
---|---|---|---|---|---|
0 | Apple | Legal | 2 | 2 | Nothing |
1 | Cheese | Legal | 3 | 2 | Nothing |
2 | Bread | Legal | 4 | 2 | Nothing |
3 | Chicken | Legal | 4 | 2 | Nothing |
4 | Tomato | Legal | 3 | 2 | Nothing |
5 | Corn | Legal | 2 | 2 | Nothing |
6 | Potato | Legal | 3 | 2 | Nothing |
7 | Wine | Legal | 5 | 2 | Nothing |
8 | Salt | Legal | 2 | 2 | Nothing |
9 | Sugar | Legal | 3 | 2 | Nothing |
20 | Silk | Illegal | 9 | 4 | 3 * Cheese |
21 | Pepper | Illegal | 8 | 4 | 2 * Chicken |
22 | Barrel | Illegal | 7 | 4 | 2 * Bread |
23 | Beer | Illegal | 6 | 4 | 4 * Wine |
24 | Seafood | Illegal | 12 | 4 | 2 * Tomato + 3 * Potato + 1 * Chicken |
Cărțile ilegale sunt cele valoroase. Pe lângă profitul mai mare, unele dintre ele aduc și un bonus deținătorului, iar acel bonus va fi de asemenea adăugat la punctajul final.
Exemplu: Pentru un jucător care deține o carte de tip Pepper, la punctajul final se va adăuga profitul corespunzător (8 monede), cat și bonusul (2 cărți de tip Chicken). Așadar, o carte de tip Pepper va aduce la scor o creștere de 8 + 2 * 4 = 16 puncte.
La final jocului ne interesează un simplu clasament în care avem id-ul jucătorului (al câtelea a fost în vectorul inițial), strategia utilizată de acesta, împreună cu punctajul său final. Acest clasament trebuie afișat sortat dupa numărul de puncte descrescător.
2 BASIC 140 4 BRIBED 91 1 BASIC 89 0 GREEDY 70 3 GREEDY 20
Clasamentul se va afișa pe consolă.
Fiecare jucător își va verifica bunurile aduse în Nottingham și pentru fiecare bun ilegal va adăuga (dacă este cazul) pe taraba sa bonusul adus de respectiva carte.
Scorul final al unui jucător va fi reprezentat de suma dintre: * numărul de monede rămase în visterie; * profitul provenit de la bunurile legale aduse în Nottingham; * profitul provenit de la bunurile ilegale aduse în Nottingham; * bonusul în funcție de rolul (rege/regină) pe care jucătorul poate să îl dețină pentru un anumit tip de bun.
Pentru realizarea implementării veți avea nevoie de anumite structuri de date pentru a stoca elementele sau a le ordona. Pachetul java.util oferă mai multe clase și interfețe pentru reprezentarea şi manipularea colecţiilor. În continuare sunt prezentate două exemple ce ilustrează folosirea interfețelor Map și List.
class Cookie { String type; int weight; Cookie(String type, int weight) { this.type = type; this.weight = weight; } }
Pentru a crea o listă cu obiecte de tip Cookie procedam astfel:
class Test { public static void main(String[] args) { List<Cookie> cookies = new LinkedList<Cookie>(); cookies.add(new Cookie("chocolate", 125)); cookies.add(new Cookie("peanuts", 100)); cookies.add(new Cookie("vanilla", 120)); for(Cookie cookie : cookies) { System.out.println(cookie.type); } } }
Dacă vrem să avem obiectele sortate în funcție de câmpul weight avem nevoie de un comparator pentru obiectele de tip Cookie:
class CookieComparator implements Comparator<Cookie> { @Override public int compare(Cookie c1, Cookie c2) { return c1.weight - c2.weight; } }
class Test { public static void main(String[] args) { CookieComparator cookieComparator = new CookieComparator(); List<Cookie> cookies = new LinkedList<Cookie>(); cookies.add(new Cookie("chocolate", 125)); cookies.add(new Cookie("peanuts", 100)); cookies.add(new Cookie("vanilla", 120)); Collections.sort(cookies, cookieComparator); for(Cookie cookie : cookies) { System.out.println(cookie.type); } } }
Atunci când avem nevoie să reținem asocieri de tip cheie - valoare folosit interfața Map. Următorul exemplu numără aparițiile fiecărui cuvânt dintr-un vector de String-uri:
String[] flavors = {"chocolate", "chocolate", "vanilla", "peanuts", "strawberry", "peanuts"}; Map<String, Integer> countFlavors = new HashMap<String, Integer>(); for (String flavour : flavors) { countFlavors.put(flavour, countFlavors.getOrDefault(flavour, 0) + 1); } for (String key : countFlavors.keySet()) { System.out.println(key + ": " + countFlavors.get(key)); }
Unul din obiectivele temei este învățarea respectării code-style-ului limbajului pe care îl folosiți. Aceste convenții (de exp. cum numiți fișierele, clasele, variabilele, cum indentați etc) sunt verificate pentru temă de către tool-ul checkstyle.
Pe pagina de Recomandări cod găsiți câteva exemple de coding style.
Dacă numărul de erori depistate de checkstyle depășește 30, atunci punctele pentru coding-style nu vor fi acordate. Dacă punctajul este negativ, acesta se trunchiază la 0.
Exemple:
punctaj_total = 100
și nr_erori = 200
⇒ nota_finala = 80
punctaj_total = 100
și nr_erori = 29
⇒ nota_finala = 100
punctaj_total = 80
și nr_erori = 30
⇒ nota_finala = 80
punctaj_total = 80
și nr_erori = 31
⇒ nota_finala = 60
Arhiva pe care o veţi urca pe VMChecker va trebui să conţină în directorul rădăcină:
README
Pe GitHub găsiți un schelet de cod care face parsarea din fișierul de intrare. schel
Pentru a folosi fileio din schelet, urmăriți tutorialul de aici. tutorial-io
Exemple de rezolvari pe foaie
a testelor de input:
* 1round2players-mixed (basic-basic): pmpjwg
* 2rounds2players-legal (basic-greedy): pmpp71