IT rekvalifikácia. Seniorní programátori zarábajú až 6 000 €/mesiac a rekvalifikácia je prvým krokom. Zisti, ako na to!

5. diel - Bojovník do arény

V predchádzajúcom cvičení, Riešené úlohy k 4. lekcii OOP v Jave, sme si precvičili získané skúsenosti z predchádzajúcich lekcií.

Už teda vieme, ako fungujú referencie a ako môžeme s objektmi zaobchádzať. Bude sa nám to hodiť dnes aj nabudúce. Tento a budúci tutoriál budú totiž venované dokončeniu našej arény. Hraciu kocku už máme, ešte nám chýbajú ďalšie 2 objekty: bojovník a samotná aréna. Dnes sa budeme venovať bojovníkovi. Najprv si popíšme, čo má bojovník vedieť, potom sa pustíme do písania kódu.

Atribúty

Bojovník sa bude nejako volať a bude mať určitý počet hp (teda bodov života, napr. 80hp). Budeme uchovávať jeho maximálny život (bude sa líšiť pri každej inštancii) a jeho súčasný život, teda napr. zranený bojovník bude mať 40hp z 80hp. Bojovník má určitý útok a obranu, oboje vyjadrené opäť v hp. Keď bojovník útočí s útokom 20hp na druhého bojovníka s obranou 10hp, uberie mu 10hp života. Bojovník bude mať referenciu na inštancii objektu RollingDie. Pri útoku či obrane si vždy hodí kockou a k útoku/obrane pripočíta padnuté číslo. (Samozrejme by mohol mať každý bojovník svoju kocku, ale chcel som sa priblížiť stolovej podobe hry a ukázať, ako OOP naozaj simuluje realitu. Bojovníci teda budú zdieľať jednu inštanciu kocky.) Kockou dodáme hre prvok náhody, v realite sa jedná vlastne o šťastie, ako sa útok alebo obrana vydarí. Napokon budeme chcieť, aby bojovníci podávali správy o tom, čo sa deje, pretože inak by z toho užívateľ nič nemal. Správa bude vyzerať napr. "Zalgoren útočí s úderom za 25hp.". Správami sa zatiaľ nebudeme zaťažovať a vrátime sa k nim až nakoniec.

Už vieme, čo budeme robiť, poďme na to! :) K projektu ArenaFight si pridajme triedu Warrior a dodajme ju patričné atribúty. Všetky budú privátne:

public class Warrior {
    /**
     * Warrior's name
     */
    private String name;
    /**
     * Health in HP
     */
    private int health;
    /**
     * Maximum health in HP
     */
    private int maxHealth;
    /**
     * Damage in HP
     */
    private int damage;
    /**
     * Defense in HP
     */
    private int defense;
    /**
     * The rolling die instance
     */
    private RollingDie die;
}

Trieda RollingDie musí samozrejme byť v našom projekte.

Metódy

Poďme pre atribúty vytvoriť konštruktor, nebude to nič ťažké. Komentáre tu vynechám, vy si ich dopíšte podobne, ako u atribútov vyššie. Nebudem ich písať ani pri ďalších metódach, aby sa tutoriál zbytočne nerozťahoval a zostal prehľadný (prípadne sa pozrite do archívu).

public Warrior(String name, int health, int damage, int defense, RollingDie die) {
    this.name = name;
    this.health = health;
    this.maxHealth = health;
    this.damage = damage;
    this.defense = defense;
    this.die = die;
}

Všimnite si, že maximálne zdravie si v konštruktore odvodíme a nemáme naň parameter v hlavičke metódy. Predpokladáme, že bojovník je pri vytvorení plne zdravý, stačí nám teda poznať iba jeho život a maximálny život bude rovnaký.

Prejdime k metódam, opäť sa najskôr zamyslime nad tým, čo by mal bojovník vedieť. Začnime tým jednoduchším, budeme chcieť nejakú textovú reprezentáciu, aby sme mohli bojovníka vypísať. Prekryjeme teda metódu toString(), ktorá vráti meno bojovníka. Určite sa nám bude hodiť metóda vracajúca, či je bojovník nažive (teda typu boolean). Aby to bolo trochu zaujímavejšie, budeme chcieť kresliť život bojovníka do konzoly, nebudeme teda písať, koľko má života, ale „vykreslíme“ ho takto:

[#########    ]

Vyššie uvedený život by zodpovedal asi 70 %. Doteraz spomínané metódy nepotrebovali žiadne parametre. Samotný útok a obranu nechajme na neskôr a poďme si implementovať metódy toString(), isAlive() a healthBar(). Začnime s metódou toString(), tam nie je čo vymýšľať:

@Override
public String toString() {
    return name;
}

Teraz implementujme metódu isAlive(), opäť to nebude nič ťažké. Stačí skontrolovať, či je život väčší ako 0 a podľa toho sa zachovať. Mohli by sme ju napísať napríklad takto:

public boolean isAlive() {
    if (health > 0) {
        return true;
    } else {
        return false;
    }
}

Keďže aj samotný výraz (health > 0) je vlastne logická hodnota, môžeme vrátiť tú a kód sa značne zjednoduší:

public boolean isAlive() {
    return (health > 0);
}

Grafický život

Ako som sa už zmienil, metóda healthBar() bude umožňovať vykresliť ukazovateľ života v grafickej podobe. Už vieme, že z hľadiska objektového návrhu nie je vhodné, aby metóda objektu priamo vypisovala do konzoly (pokiaľ nie je na výpis objekt určený), preto si znaky uložíme do reťazca a ten vrátime pre neskoršie vypísanie. Ukážeme si kód metódy a následne podrobne popíšeme:

public String healthBar() {
    String healthBar = "[";
    int total = 20;
    double count = Math.round(((double)health / maxHealth) * total);
    if ((count == 0) && (isAlive())) {
        count = 1;
    }
    for (int i = 0; i < count; i++) {
        healthBar += "#";
    }
    for (int i = 0; i < total - count; i++) {
        healthBar += " ";
    }
    healthBar += "]";
    return healthBar;
}

Pripravíme si reťazec healthBar a vložíme doň úvodný znak [. Určíme si celkovú dĺžku ukazovateľa života do premennej total (napr. 20). Teraz v podstate nepotrebujeme nič iné, než trojčlenku. Ak maxHealth zodpovedá total dielikov, premenná health bude zodpovedať premennej count. Premenná count obsahuje počet dielikov aktuálneho zdravia.

Matematicky platí, že count = (health / maxHealth) * total. My ešte doplníme zaokrúhlenie na celé dieliky a tiež pretypovanie jedného z operandov na double, aby Java chápala delenie ako neceločíselné.

Mali by sme ošetriť prípad, keď je život taký nízky, že nám vyjde na 0 dielikov, ale bojovník je stále nažive. V tom prípade vykreslíme 1 dielik, inak by to vyzeralo, že je už mŕtvy.

Ďalej stačí jednoducho cyklom for pripojiť k reťazcu healthBar patričný počet znakov a doplniť ich medzerami do celkovej dĺžky. Doplnenie vykonáme pomocou cyklu for, ktorý pridáva medzery do dĺžky total. Pridáme koncový znak a reťazec vrátime.

Všetko si vyskúšame, prejdime k metóde main() a vytvorme si bojovníka (a kocku, pretože tu musíme konštruktoru bojovníka odovzdať). Následne vypíšme, či je nažive a jeho život graficky:

        RollingDie die = new RollingDie(10);
        Warrior warrior = new Warrior("Zalgoren", 100, 20, 10, die);

        System.out.printf("Warrior: %s%n", warrior); // test toString();
        System.out.printf("Alive: %s%n", warrior.isAlive()); // test isAlive();
        System.out.printf("health: %s%n", warrior.healthBar()); // test healthBar();
import java.util.Random;

public class RollingDie {
    private Random random;
    private int sidesCount;

    public RollingDie() {
        sidesCount = 6;
        random = new Random();
    }

    public RollingDie(int sidesCount) {
        this.sidesCount = sidesCount;
        random = new Random();
    }

    public int getSidesCount() {
        return sidesCount;
    }

    public int roll() {
        return random.nextInt(sidesCount) + 1;
    }

    @Override
    public String toString() {
        return String.format("Rolling die with %s sides", sidesCount);
    }

}
public class Warrior {
    private String name;
    private int health;
    private int maxHealth;
    private int damage;
    private int defense;
    private RollingDie die;

    public Warrior(String name, int health, int damage, int defense, RollingDie die) {
        this.name = name;
        this.health = health;
        this.maxHealth = health;
        this.damage = damage;
        this.defense = defense;
        this.die = die;
    }

    @Override
    public String toString() {
        return name;
    }

    public boolean isAlive() {
        return (health > 0);
    }

    public String healthBar() {
        String healthBar = "[";
        int total = 20;
        double count = Math.round(((double)health / maxHealth) * total);
        if ((count == 0) && (isAlive())) {
            count = 1;
        }
        for (int i = 0; i < count; i++) {
            healthBar += "#";
        }
        for (int i = 0; i < total - count; i++) {
            healthBar += " ";
        }
        healthBar += "]";
        return healthBar;
    }

}
Konzolová aplikácia
Warrior: Zalgoren
Alive: true
health: [####################]

Boj

Dostávame sa k samotnému boju. Implementujeme metódy na útok a obranu.

Obrana

Začnime obranou. Metóda defend() bude umožňovať brániť sa úderu, ktorého sila bude odovzdaná metóde ako parameter. Metódu si opäť ukážeme a potom popíšeme:

public void defend(int hit) {
    int injury = hit - (defense + die.roll());
    if (injury > 0) {
        health -= injury;
        if (health <= 0) {
            health = 0;
        }
    }
}

Najprv spočítame skutočné zranenie a to tak, že z útoku nepriateľa odpočítame našu obranu zvýšenú o číslo, ktoré padlo na hracej kocke. Ak sme zranenie celé neodrazili (injury > 0), budeme znižovať náš život. Táto podmienka je dôležitá, keby sme zranenia odrazili a bolo napr. -2, bez podmienky by sa život bojovníka zvýšil. Po znížení života skontrolujeme, či nie je v zápornej hodnote a prípadne ho dorovnáme na nulu.

Útok

Metóda attack() bude brať ako parameter inštanciu bojovníka, na ktorého sa útočí. To preto, aby sme na ňom mohli zavolať metódu defend(), ktorá na náš útok zareaguje a zmenší protivníkov život. Tu vidíme výhody referencií v Jave, môžeme si inštancie jednoducho odovzdávať a volať na nich metódy bez toho, aby došlo k ich skopírovaniu. Ako prvý vypočítame úder, podobne ako pri obrane, úder bude náš útok + hodnota z hracej kocky. Na súperovi následne zavoláme metódu defend() s hodnotou úderu:

public void attack(Warrior enemy) {
    int hit = damage + die.roll();
    enemy.defend(hit);
}

To by sme mali, poďme si skúsiť v našom ukážkovom programe zaútočiť a potom znova vykresliť život. Pre jednoduchosť nemusíme zakladať ďalšieho bojovníka, ale môžeme zaútočiť sami na seba:

        RollingDie die = new RollingDie(10);
        Warrior warrior = new Warrior("Zalgoren", 100, 20, 10, die);

        System.out.printf("Warrior: %s%n", warrior); // test toString();
        System.out.printf("Alive: %s%n", warrior.isAlive()); // test isAlive();
        System.out.printf("health: %s%n", warrior.healthBar()); // test graphicHealth();

        warrior.attack(warrior); // attack test
        System.out.printf("Health after the hit:: %s%n", warrior.healthBar());
import java.util.Random;
public class RollingDie {
    private Random random;
    private int sidesCount;

    public RollingDie() {
        sidesCount = 6;
        random = new Random();
    }

    public RollingDie(int sidesCount) {
        this.sidesCount = sidesCount;
        random = new Random();
    }

    public int getSidesCount() {
        return sidesCount;
    }

    public int roll() {
        return random.nextInt(sidesCount) + 1;
    }

    @Override
    public String toString() {
        return String.format("Rolling die with %s sides", sidesCount);
    }

}
public class Warrior {
    private String name;
    private int health;
    private int maxHealth;
    private int damage;
    private int defense;
    private RollingDie die;

    public Warrior(String name, int health, int damage, int defense, RollingDie die) {
        this.name = name;
        this.health = health;
        this.maxHealth = health;
        this.damage = damage;
        this.defense = defense;
        this.die = die;
    }

    @Override
    public String toString() {
        return name;
    }

    public boolean isAlive() {
        return (health > 0);
    }

    public String healthBar() {
        String healthBar = "[";
        int total = 20;
        double count = Math.round(((double)health / maxHealth) * total);
        if ((count == 0) && (isAlive())) {
            count = 1;
        }
        for (int i = 0; i < count; i++) {
            healthBar += "#";
        }
        for (int i = 0; i < total - count; i++) {
            healthBar += " ";
        }
        healthBar += "]";
        return healthBar;
    }

    public void defend(int hit) {
        int injury = hit - (defense + die.roll());
        if (injury > 0) {
            health -= injury;
            if (health <= 0) {
                health = 0;
            }
        }
    }

    public void attack(Warrior enemy) {
        int hit = damage + die.roll();
        enemy.defend(hit);
    }

}
Konzolová aplikácia
Warrior: Zalgoren
Alive: true
health: [####################]
Health after the hit: [##################  ]

Zdá sa, že všetko funguje, ako má. Prejdime k poslednému bodu dnešného tutoriálu a to k správam.

Správy

Ako už bolo povedané, o útokoch a obrane budeme užívateľov informovať výpisom na konzole. Výpis nebude vykonávať samotná trieda Warrior, tá bude len vracať správy ako textové reťazce. Jedna možnosť by bola nastaviť návratový typ metód attack() a defend() na String a pri ich volaní vrátiť aj správu. Problém by však nastal v prípade, ak by sme chceli získať správu od metódy, ktorá už niečo vracia. Metóda samozrejme nemôže jednoducho vrátiť 2 veci.

Poďme na vec univerzálnejšie, správu budeme ukladať do privátnej premennej message a urobíme metódy na jej uloženie a vrátenie. Samozrejme by sme mohli urobiť premennú verejnú, ale nie je tu dôvod, prečo umožniť zvonku zápis do správy a tiež by skladanie zložitejšej správy vo vnútri triedy mohlo byť niekedy problematické.

K atribútom triedy teda pridáme:

private String message;

Teraz si vytvoríme dve metódy. Najprv privátnu metódu setMessage(), ktorá berie ako parameter text správy a slúži na vnútorné účely triedy, kde nastaví správu do privátnej premennej:

private void setMessage(String message) {
    this.message = message;
}

Nič zložité. Podobne jednoduchá bude verejná metóda na vrátenie správy:

public String getLastMessage() {
    return message;
}

O práci so správami obohatíme naše metódy attack() a defend(), teraz budú vyzerať takto:

public void attack(Warrior enemy) {
    int hit = damage + die.roll();
    setMessage(String.format("%s attacks with a hit worth %s hp", name, hit));
    enemy.defend(hit);
}

public void defend(int hit) {
    int injury = hit - (defense + die.roll());
    if (injury > 0) {
        health -= injury;
        message = String.format("%s defended against the attack but still lost %s hp", name, injury);
        if (health <= 0) {
            health = 0;
            message += " and died";
        }

    } else {
        message = String.format("%s blocked the hit", name);
    }
    setMessage(message);
}

Všetko si opäť vyskúšame, tentoraz už vytvoríme druhého bojovníka:

        RollingDie die = new RollingDie(10);
        Warrior warrior = new Warrior("Zalgoren", 100, 20, 10, die);

        System.out.printf("Health: %s%n", warrior.healthBar());

        // warrior attack phase
        Warrior enemy = new Warrior("Shadow", 60, 18, 15, die);
        enemy.attack(warrior);
        System.out.println(enemy.getLastMessage());
        System.out.println(warrior.getLastMessage());
        System.out.printf("Health: %s%n", warrior.healthBar());
import java.util.Random;

public class RollingDie {
    private Random random;
    private int sidesCount;

    public RollingDie() {
        sidesCount = 6;
        random = new Random();
    }

    public RollingDie(int sidesCount) {
        this.sidesCount = sidesCount;
        random = new Random();
    }

    public int getSidesCount() {
        return sidesCount;
    }

    public int roll() {
        return random.nextInt(sidesCount) + 1;
    }

    @Override
    public String toString() {
        return String.format("Rolling die with %s sides", sidesCount);
    }

}
class Warrior {
    private String name;
    private int health;
    private int maxHealth;
    private int damage;
    private int defense;
    private RollingDie die;
    private String message;

    public Warrior(String name, int health, int damage, int defense, RollingDie die) {
        this.name = name;
        this.health = health;
        this.maxHealth = health;
        this.damage = damage;
        this.defense = defense;
        this.die = die;
    }

    public boolean isAlive() {
        return (health > 0);
    }

    public String healthBar() {
        String healthBar = "[";
        int total = 20;
        double count = Math.round(((double)health / maxHealth) * total);
        if ((count == 0) && (isAlive())) {
            count = 1;
        }
        for (int i = 0; i < count; i++) {
            healthBar += "#";
        }
        for (int i = 0; i < total - count; i++) {
            healthBar += " ";
        }
        healthBar += "]";
        return healthBar;
    }

    public void attack(Warrior enemy) {
        int hit = damage + die.roll();
        setMessage(String.format("%s attacks with a hit worth %s hp", name, hit));
        enemy.defend(hit);
    }

    public void defend(int hit) {
        int injury = hit - (defense + die.roll());
        if (injury > 0) {
            health -= injury;
            message = String.format("%s defended against the attack but still lost %s hp", name, injury);
            if (health <= 0) {
                health = 0;
                message += " and died";
            }

        } else {
            message = String.format("%s blocked the hit", name);
        }
        setMessage(message);
    }

    private void setMessage(String message) {
        this.message = message;
    }

    public String getLastMessage() {
        return message;
    }

    @Override
    public String toString() {
        return name;
    }
}
Konzolová aplikácia
Health: [####################]
Shadow attacks with a hit worth 24 hp
Zalgoren defended against the attack but still lost 10 hp
Health: [##################  ]

Máme kocku aj bojovníka, teraz už chýba len aréna.

V ďalšej lekcii, Java - Aréna s bojovníkmi, si vytvoríme arénu.


 

Mal si s čímkoľvek problém? Stiahni si vzorovú aplikáciu nižšie a porovnaj ju so svojím projektom, chybu tak ľahko nájdeš.

Stiahnuť

Stiahnutím nasledujúceho súboru súhlasíš s licenčnými podmienkami

Stiahnuté 11x (4.59 kB)
Aplikácia je vrátane zdrojových kódov v jazyku Java

 

Predchádzajúci článok
Riešené úlohy k 4. lekcii OOP v Jave
Všetky články v sekcii
Objektovo orientované programovanie v Jave
Preskočiť článok
(neodporúčame)
Java - Aréna s bojovníkmi
Článok pre vás napísal David Hartinger
Avatar
Užívateľské hodnotenie:
29 hlasov
David je zakladatelem ITnetwork a programování se profesionálně věnuje 15 let. Má rád Nirvanu, nemovitosti a svobodu podnikání.
Unicorn university David sa informačné technológie naučil na Unicorn University - prestížnej súkromnej vysokej škole IT a ekonómie.
Aktivity