Cvičenie 4 – Znaky, reťazce, myšacie udalosti

Cieľom cvičení štvrtého týždňa je:

  • práca s reťazcami (String, StringBuilder),
  • spracovanie „myšacích“ udalostí vo WinPane (a tried rozšírujúcich WinPane) a
  • inštančné premenné.

Korytnačka vo svete znakových reťazcov

  • Na čo slúžia objekty triedy String? Akou metódou nám objekt triedy String vie povedať dĺžku reťazca, ktorý uchováva? Akým spôsobom sa vieme objektov triedy String „pýtať“ na jednotlivé znaky reťazca?
  • Čo je to referencia? Čo znamená, že parametrom metódy je referencia na objekt nejakej triedy (napr. String)?
  • Vytvorte triedu StringTurtle rozširujúcu triedu Turtle s nasledujúcimi metódami:
    • countChar vráti počet výskytov znaku c v reťazci referencovanom parametrom s
    • isPalindrome vráti, či reťazec referencovaný parametrom s je palindróm, t.j. číta sa rovnako zľava doprava ako aj sprava doľava: abcba, accaacca sú palindrómy
    • containsDoubleSpace vráti, či reťazec referencovaný parametrom s obsahuje 2 za sebou idúce medzery

1
2
3
public int countChar(String s, char c)
public boolean isPalindrome(String s)
public boolean containsDoubleSpace(String s)
  • Porovnajte nasledujúce fragmenty kódu a diskutujte aký výpis bude realizovaný a prečo.

1
2
3
4
5
6
7
String s = new String("Ahoj");
String t = new String("Ahoj");
if (s == t) {
    System.out.println("rovnake");
} else {
    System.out.println("rozne");
}

1
2
3
4
5
6
7
char a = 'a';
char b = 'a';
if (a == b) {
    System.out.println("rovnake");
} else {
    System.out.println("rozne");
}

Znakové reťazce

Na internete vyhľadajte zoznam metód objektov tried String a StringBuilder. Aké zaujímavé metódy ste našli?

Vytvorte triedu StringTurtle rozširujúcu triedu Turtle s nasledujúcimi metódami:

  • unicharString vráti referenciu na novovytvorený reťazec, ktorý sa skladá len zo znakov c o dĺžke n (vyskúšajte variant so StringBuilder-om aj bez neho, variant bez použitia objektu triedy StringBuilder aj debugujte).
  • isUnicharString vráti true, práve vtedy, keď zadaný reťazec obsahuje všetky znaky úplne rovnaké
  • duplicateChars vráti referenciu na novovytvorený reťazec, ktorý vznikne zduplikovaním znakov v reťazci referencovanom parametrom s („Ahoj“ -> „AAhhoojj“)
  • removeExtraSpaces vráti referenciu na novovytvorený reťazec, ktorý vznikne odstránením zbytočných medzier v reťazci referencovanom parametrom s (zbytočné sú medzery na začiatku a konci reťazca, každú postupnosť 2 a viacerých medzier môžeme nahradiť jednou medzerou).
  • countWords vráti počet slov v reťazci referencovanom parametrom s. Využite fintu, že každý koniec slova môžeme rozoznať tak, že je to vlastne medzera, pred ktorou je nemedzera. Inými slovami stačí nám spočítať počet medzier, pred ktorými je nemedzera. Týmto spôsobom ale nezarátame posledné slovo, ak za ním už nie je žiadna medzera. Toto vyriešime tak, že budeme počítať počet slov v reťazci, ktorý vznikne pridaním medzery na koniec pôvodného reťazca (s = s + " ";).

1
2
3
4
5
public String unicharString(char c, int n)
public boolean isUnicharString(String s)
public String duplicateChars(String s)
public String removeExtraSpaces(String s)
public int countWords(String s)

Generátor hesiel

Naučte objekty triedy StringTurtle metódu na automatické generovanie hesiel zadanej dĺžky. Metóda okrem dĺžky dostane ako parameter 2 reťazce (2 referencie na objekty triedy String). Heslo je generované tak, že každé párne písmeno hesla je náhodné písmeno z prvého reťazca a každé nepárne písmeno je náhodné písmeno z druhého reťazca. Ak máme ako vstup reťazce "aioo" a "kjmjx", tak vygenerované heslo dĺžky 7 môže byť "kajajok".


1
public String generatePassword(int length, String s1, String s2)

Vypĺňanie mriežky

  • Vytvorte triedu GridPane rozširujúcu triedu WinPane. Naučte objekty triedy GridPane metódu drawGrid, ktorá vyplní kresliacu plochu mriežkou, pričom každý zo štvorčekov mriežky má rozmer 50 x 50. Na testovanie môžete využiť ObjectInspector skúmajúci objekty tejto triedy.

1
public void drawGrid()

V „spúšťači“ pridajte volanie metódy drawGrid hneď po vytvorení objektu triedy GridPane.

  • Do triedy GridPane pridajte metódu drawDot, ktorá nakreslí farebnú bodku s polomerom 20 v strede políčka so zadanými súradnicami. Súradnice nech sú podobné, ako pri kresliacej ploche, t.j. políčko v ľavom hornom rohu má súradnice (0, 0), políčko napravo od neho (1, 0), atď.

1
public void drawDot(int column, int row, Color dotColor)
  • Do triedy GridPane pridajte metódu onMouseClicked na obsluhu kliknutia do kresliacej plochy. Pri kliknutí do kresliacej plochy, nech sa v políčku, do ktorého sa kliklo, namaľuje červená bodka s využitím metódy drawDot.

1
protected void onMouseClicked(int x, int y, MouseEvent detail)
  • Upravte metódu onMouseClicked tak, aby sa striedavo kreslili červené a modré bodky. (Návod: využite, že v inštančnej premennej si môžete zapamätať aká bodka sa kreslila ako posledná, resp. koľko bodiek sa doposiaľ nakreslilo).

A opäť reťazce

  • Vytvorte metódu, ktorá overí, či reťazec (objekt triedy String referencovaný z parametra s) obsahuje súvislú podpostupnosť aspoň limit za sebou idúcich znakov '.' (bodka). Napríklad reťazec "aaa....abda.asa..as.asnasa.....asasas" obsahuje súvislú podpostupnosť bodiek dĺžky 3, ale nie dĺžky 8.

1
public boolean checkDots(String s, int limit)

Korytnačky s pamäťou

  • Vytvorte triedu OdoTurtle rozširujúcu triedu Turtle. Korytnačky tejto triedy naučte metódu measuredStep, ktorá spraví presne to isté, čo metóda step avšak s tým rozdielom, že korytnačka si bude pamätať celkovú dĺžku, ktorú prešla pomocou metódy measuredStep.
  • Pre korytnačky triedy OdoTurtle pridajte metódu getTrajectoryLength, ktorá vráti celkovú dĺžku, ktorú korytnačka prešla.

1
2
public void measuredStep(double length)
public double getTrajectoryLength()

Pomôcka pre Java programátorov

Určite viete, že pre názvy premenných a metód v Jave platia isté pravidlá „slušnosti“ (konvencie). V Jave sa používa tzv. lower-camel-case konvencia. V tejto konvencii platí, že názov je jedno slovo, v prípade viacslovného názvu jednotlivé slová zlepíme. Všetky písmena v názve sú písané malými písmenami s výnimkou tých písmen, ktoré zodpovedajú prvým písmenám slov vo viacslovných názvoch s výnimkou prvého slova. Príklady: nazovPremennej, premenna, sucetNaParnychPoziciach, platnostPodmienky, rozdielDvochCisel. Začiatočníci s týmto spôsobom pomenovávania premenných majú často ťažkosti. Našim cieľom je preto vytvoriť metódu, ktorá ako parameter dostane referenciu na jedno alebo viacslovný názov, v ktorom sú jednotlivé slová oddelené medzerami (aj viacerými). Metóda vráti správne naformátovaný názov premennej v lower-camel-case konvencii.


1
public String camelFormat(String nazov)

Príklady:

  • camelFormat(" Pomocne Pole korytnaciek "): "pomocnePoleKorytnaciek"
  • camelFormat("Nazov Premennej"): "nazovPremennej"
  • camelFormat(" ROZDIEL DVOCH CISEL "): "rozdielDvochCisel"

Užitočné metódy:

  • Character.toLowerCase – vráti malé písmeno k zadanému znaku
  • Character.toUpperCase – vráti veľké písmeno k zadanému znaku

1
2
3
char znak = 'a';
char velkeA = Character.toUpperCase(znak);
char maleX = Character.toLowerCase('X');

Užitočné metódy objektov triedy String:

  • substring – vráti referenciu na nový reťazec, ktorý obsahuje podreťazec reťazca na zadaných indexoch
  • toLowerCase – vráti referenciu na nový reťazec, ktorý obsahuje reťazec zapísaný malými písmenami
  • toUpperCase – vráti referenciu na nový reťazec, ktorý obsahuje reťazec zapísaný veľkými písmenami

Pre fajnšmekrov

  • Do objektov triedy StringTurtle, ktorá rozširuje triedu Turtle, pridajte metódu validateParenthesis vráti či reťazec obsahujúci len znaky '{' a '}' predstavuje správne ozátvorkovaný výraz. Postupnosť "{{}}{}" je správne ozátvorkovaný, kým "{{{}" a "{}}{}{" nie.

1
public boolean validateParenthesis(String s)
  • Pomocou operácie Double.parseDouble() dokážeme zistiť, aké reálne číslo je zapísané v reťazci:

1
double cislo = Double.parseDouble("3.14");

S využitím tejto operácie vytvorte metódu hybSaPodla, ktorá ako parameter dostane referenciu na reťazec obsahujúci príkazy, podľa ktorých sa má korytnačka hýbať. Reťazec je postupnosť dvojíc reálnych čísel oddelených medzerami. Vždy prvé číslo hovorí, aký dlhý krok má korytnačka spraviť, a druhé číslo, o koľko sa má korytnačka otočiť. Napríklad reťazec "100.5 120 30.8 90" zodpovedá príkazom step(100.5), turn(120), step(30.8), turn(90).

  • Pozrite si poslednú úlohu z bloku „Znakové reťazce“, kde sa porovnávali dva fragmenty kódu. Porovnajte ich s nasledujúcim fragmentom a rozhodnite aký výpis sa vykoná. Svoje rozhodnutie zdôvodnite a následne vyskúšajte aký výpis naozaj prebehne.
    • Vyhľadajte, čo je java string pool a načo slúži.

1
2
3
4
5
6
7
String s = "Ahoj";
String t = "Ahoj";
if (s == t) {
    System.out.println("rovnake");
} else {
    System.out.println("rozne");
}