forked from ChrisMayfield/ThinkJava2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ch13.tex
814 lines (580 loc) · 28.7 KB
/
ch13.tex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
\chapter{Objects of Arrays}
\index{array!of cards}
In the previous chapter, we defined a class to represent cards and used an array of \java{Card} objects to represent a deck.
In this chapter, we take additional steps toward object-oriented programming.
First we define a class to represent a deck of cards.
Then we present algorithms for shuffling and sorting decks.
Finally, we introduce \java{ArrayList} from the Java library and use it to represent collections of cards.
%While reading the following sections, we recommend that you create a {\it Deck.java} file and paste in all the examples.
%You will need {\it Card.java} from the previous chapter for it to compile.
%The code for this chapter is in {\it Card.java} and {\it Deck.java}, which are in the directory {\it ch13} in the repository for this book.
%Instructions for downloading this code are on page~\pageref{code}.
\section{Decks of Cards}
\label{deck}
\index{Deck}
Here is the beginning of a \java{Deck} class that encapsulates an array of \java{Card} objects:
\begin{code}
public class Deck {
private Card[] cards;
public Deck(int n) {
this.cards = new Card[n];
}
public Card[] getCards() {
return this.cards;
}
}
\end{code}
\index{constructor}
\index{memory diagram}
\index{diagram!memory}
The constructor initializes the instance variable with an array of \java{n} cards, but it doesn't create any \java{Card} objects.
Figure~\ref{fig.deckobject} shows what a \java{Deck} looks like with no cards.
\begin{figure}[!ht]
\begin{center}
\includegraphics{figs/deckobject.pdf}
\caption{Memory diagram of an unpopulated \java{Deck} object.}
\label{fig.deckobject}
\end{center}
\end{figure}
We'll add another constructor that creates a standard 52-card array and populates it with \java{Card} objects:
\begin{code}
public Deck() {
this.cards = new Card[52];
int index = 0;
for (int suit = 0; suit <= 3; suit++) {
for (int rank = 1; rank <= 13; rank++) {
this.cards[index] = new Card(rank, suit);
index++;
}
}
}
\end{code}
This method is similar to the example in Section~\ref{cardarray}; we just turned it into a constructor.
We can use it to create a complete \java{Deck} like this:
\begin{code}
Deck deck = new Deck();
\end{code}
\index{printDeck}
Now that we have a \java{Deck} class, we have a logical place to put methods that pertain to decks.
Looking at the methods we have written so far, one obvious candidate is \java{printDeck} from Section~\ref{cardarray}.
Here's how it looks, rewritten as an instance method of \java{Deck}:
\begin{code}
public void print() {
for (Card card : this.cards) {
System.out.println(card);
}
}
\end{code}
%\begin{code}
%public void print() {
% for (int i = 0; i < this.cards.length; i++) {
% System.out.println(this.cards[i]);
% }
%}
%\end{code}
Notice that when we transform a static method into an instance method, the code is shorter.
Here's how we invoke it:
\begin{code}
deck.print();
\end{code}
\section{Shuffling Decks}
\label{shuffle}
\index{shuffle}
For most card games, you have to shuffle the deck; that is, put the cards in a random order.
In Section~\ref{random} you saw how to generate random numbers, but it is not obvious how to use them to shuffle a deck.
One possibility is to model the way humans shuffle; for example, we could divide the deck in two halves and then choose alternately from each one.
Humans usually don't shuffle perfectly, so after about seven iterations, the order of the deck is pretty well randomized.
But a computer program would have the annoying property of doing a perfect shuffle every time, which is not very random.
In fact, after eight perfect shuffles, you would find the deck back in the order you started in!
For more on this, see \url{https://en.wikipedia.org/wiki/Faro_shuffle}.
\index{pseudocode}
A better shuffling algorithm is to traverse the deck one card at a time, and at each iteration, choose two cards and swap them.
To outline this algorithm, we'll use a combination of Java statements and English comments.
This technique is sometimes called {\bf pseudocode}:
\index{shuffle}
\begin{code}
public void shuffle() {
for each index i {
// choose a random number between i and length - 1
// swap the ith card and the randomly-chosen card
}
}
\end{code}
\index{helper method}
\index{method!helper}
The nice thing about pseudocode is that it often makes clear what other methods you are going to need.
In this case, we need a method that chooses a random integer in a given range and a method that takes two indexes and swaps the cards at those positions:
\begin{code}
private static int randomInt(int low, int high) {
// return a random number between low and high,
// including both
}
private void swapCards(int i, int j) {
// swap the ith and the jth cards in the array
}
\end{code}
\index{randomInt}
\index{swapCards}
Methods like \java{randomInt} and \java{swapCards} are called {\bf helper methods}, because they help you solve parts of the problem.
Helper methods are often \java{private}, because they are used only by methods in the class and are not needed by methods in other classes.
\index{top-down design}
\index{design process}
The process of writing pseudocode first and then writing helper methods to make it work is a kind of {\bf top-down design} (see \url{https://en.wikipedia.org/wiki/Top-down_and_bottom-up_design}).
It is an alternative to ``incremental development'' and ``encapsulation and generalization'', the other design processes you have seen in this book.
One of the exercises at the end of the chapter asks you to write the helper methods \java{randomInt} and \java{swapCards}, and use them to implement \java{shuffle}.
When you do the exercise, notice that \java{randomInt} is a class method and \java{swapCards} is an instance method.
Do you understand why?
\section{Selection Sort}
\label{sorting}
\index{selection sort}
\index{sort!selection}
Now that we have shuffled the deck, we need a way to put it back in order.
There is an algorithm for sorting that is ironically similar to the algorithm for shuffling.
It's called {\bf selection sort}, because it works by traversing the array repeatedly and selecting the lowest (or highest) remaining card each time.
During the first iteration, we find the lowest card and swap it with the card in the zeroth position.
During the $i$th iteration, we find the lowest card to the right of $i$ and swap it with the $i$th card.
Here is pseudocode for selection sort:
\begin{code}
public void selectionSort() {
for each index i {
// find the lowest card at or to the right of i
// swap the ith card and the lowest card found
}
}
\end{code}
Again, the pseudocode helps with the design of the helper methods.
For this algorithm, we can reuse \java{swapCards} from the previous section, so we need only a method to find the lowest card; we'll call it \java{indexLowest}:
\begin{code}
private int indexLowest(int low, int high) {
// find the lowest card between low and high
}
\end{code}
One of the exercises at the end of the chapter asks you to write \java{indexLowest}, and then use it and \java{swapCards} to implement \java{selectionSort}.
\section{Merge Sort}
\label{mergesort}
\index{efficiency}
Selection sort is a simple algorithm, but it is not very efficient.
To sort $n$ items, it has to traverse the array $n-1$ times.
Each traversal takes an amount of time proportional to $n$.
The total time, therefore, is proportional to $n^2$.
\index{merge sort}
\index{sort!merge}
We will develop a more efficient algorithm called {\bf merge sort}.
To sort $n$ items, merge sort takes time proportional to $n \log_2 n$.
That may not seem impressive, but as $n$ gets big, the difference between $n^2$ and $n \log_2 n$ can be enormous.
For example, $\log_2$ of one million is around 20.
So if you had to sort a million numbers, merge sort would require 20 million steps.
But selection sort would require one trillion steps!
The idea behind merge sort is this: if you have two decks, each of which has already been sorted, you can quickly merge them into a single, sorted deck.
Try this out with a deck of cards:
\begin{enumerate}
\item Form two decks with about 10 cards each, and sort them so they are face up with the lowest cards on top.
Place the decks in front of you.
\item Compare the top card from each deck and choose the lower one.
Flip it over and add it to the merged deck.
\item Repeat step 2 until one of the decks is empty.
Then take the remaining cards and add them to the merged deck.
\end{enumerate}
The result should be a single sorted deck.
In the next few sections, we'll explain how to implement this algorithm in Java.
\section{Subdecks}
\index{subdeck}
\label{subdeck}
The first step of merge sort is to split the deck into two ``subdecks'', each with about half of the cards.
So we need a method that takes a deck, and a range of indexes, and returns a new deck that contains the specified subset of cards:
\begin{code}
public Deck subdeck(int low, int high) {
Deck sub = new Deck(high - low + 1);
for (int i = 0; i < sub.cards.length; i++) {
sub.cards[i] = this.cards[low + i];
}
return sub;
}
\end{code}
The first line creates an unpopulated \java{Deck} object that contains an array of \java{null} references.
Inside the \java{for} loop, the subdeck gets populated with references to \java{Card} objects.
\index{off-by-one}
The length of the subdeck is \java{high - low + 1}, because both the low card and the high card are included.
This sort of computation can be confusing, and forgetting the ``\java{+ 1}'' often leads to {\bf off-by-one} errors.
Drawing a picture is usually the best way to avoid them.
%For example, to select the middle three of five values in an array, we need \java{3 - 1 + 1} values.
%
%\begin{center}
%\begin{tabular}{ccccc}
%\hline
%\multicolumn{1}{|c|}{} & \multicolumn{1}{c|}{X} & \multicolumn{1}{c|}{X} & \multicolumn{1}{c|}{X} & \multicolumn{1}{c|}{} \\
%\hline
%0 & 1 & 2 & 3 & 4 \\
%\end{tabular}
%\end{center}
\index{constructor}
\index{overload}
Figure~\ref{fig.subdeck} is a memory diagram of a subdeck with \java{low = 0} and \java{high = 4}.
The result is a hand with five cards that are {\em shared} with the original deck; that is, they are aliased.
\begin{figure}[!ht]
\begin{center}
\includegraphics{figs/subdeck.pdf}
\caption{Memory diagram showing the effect of \java{subdeck}.}
\label{fig.subdeck}
\end{center}
\end{figure}
\index{aliasing}
\index{reference}
Aliasing might not be a good idea, because changes to shared cards would be reflected in multiple decks.
But since \java{Card} objects are immutable, this kind of aliasing is not a problem.
And it saves some memory because we don't create duplicate \java{Card} objects.
\section{Merging Decks}
\index{merge}
The next helper method we need is \java{merge}, which takes two sorted subdecks and returns a new deck containing all cards from both decks, in order.
Here's what the algorithm looks like in pseudocode, assuming the subdecks are named \java{d1} and \java{d2}:
\begin{code}
private static Deck merge(Deck d1, Deck d2) {
// create a new deck, d3, big enough for all the cards
// use the index i to keep track of where we are at in
// the first deck, and the index j for the second deck
int i = 0;
int j = 0;
// the index k traverses the result deck
for (int k = 0; k < d3.length; k++) {
// if d1 is empty, use top card from d2
// if d2 is empty, use top card from d1
// otherwise, compare the top two cards
// add lowest card to the new deck at k
// increment i or j (depending on card)
}
// return the new deck
}
\end{code}
An exercise at the end of the chapter asks you to implement \java{merge}.
It's a little tricky, so be sure to test it with different subdecks.
Once your \java{merge} method is working, you can use it to write a simplified version of merge sort:
\begin{code}
public Deck almostMergeSort() {
// divide the deck into two subdecks
// sort the subdecks using selectionSort
// merge the subdecks, return the result
}
\end{code}
If you have working versions of \java{subdeck}, \java{selectionSort}, and \java{merge}, you should have no trouble getting this method working.
But it is still not very efficient, because it uses \java{selectionSort} to sort the subdecks.
We can make it more efficient if we use \java{mergeSort} instead, but that means we have to make it recursive!
\section{Adding Recursion}
To make \java{mergeSort} work recursively, you have to add a base case; otherwise, it repeats forever.
The simplest base case is a subdeck with one card.
If there is only one card, it can't be out of order, so we consider it sorted.
And if it is already sorted, we can just return it.
And it will turn out to be convenient if we handle another base case, a subdeck with zero cards.
By the same logic, if there are no cards, they can't be out of order.
So we consider an empty deck to be sorted, and return it.
With these base cases, a recursive version of \java{mergeSort} looks like this:
\begin{code}
public Deck mergeSort() {
// if the deck has 0 or 1 cards, return it
// otherwise, divide the deck into two subdecks
// sort the subdecks using mergeSort
// merge the subdecks
// return the result
}
\end{code}
\index{leap of faith}
As usual, there are two ways to think about recursive programs: you can follow the flow of execution, or you can make the ``leap of faith'' (see Section~\ref{leap_of_faith}).
This example should encourage you to make the leap of faith.
When you use \java{selectionSort} to sort the subdecks, you don't feel compelled to follow the flow of execution.
You assume it works because you already debugged it.
When you make \java{mergeSort} recursive, you just replace one sorting algorithm with another.
There is no reason to read the program differently.
Well, almost.
You have to think about the base cases and make sure that you reach them.
But other than that, writing the recursive version should be no problem.
%The most difficult part of merge sort is the \java{merge} method, and that part is not recursive.
As an exercise at the end of this chapter, you'll have a chance to finish off this example.
\section{Static Context}
Figure~\ref{fig.deck} shows a UML class diagram for \java{Deck}, including the instance variable, \java{cards}, and the methods we have so far.
In UML diagrams, \java{private} attributes and methods begin with a minus sign (\java{-}) and \java{static} methods are underlined.
\index{UML}
\index{class diagram}
\index{diagram!class}
\begin{figure}[!ht]
\begin{center}
\includegraphics{figs/deck.pdf}
\caption{UML diagram for the \java{Deck} class.}
\label{fig.deck}
\end{center}
\end{figure}
The helper methods \java{randomInt} and \java{merge} are \java{static}, because they do not read or write any instance variables.
All other methods are instance methods, because they access the instance variable, \java{cards}.
When you have static methods and instance methods in the same class, it is easy to get them confused.
To invoke an instance method, you need an instance:
\begin{code}
Deck deck = new Deck();
deck.print(); // correct
\end{code}
\java{Deck} with a capital \java{D} is a class, and \java{deck} with a lowercase \java{d} is an object.
Say you try to invoke \java{print} like this:
\begin{code}
Deck.print(); // wrong!
\end{code}
\index{static context}
\index{this}
You get a compiler error like this:
\begin{stdout}
Non-static method print() cannot be referenced from a
static context.
\end{stdout}
By ``static context'', the compiler means you are trying to invoke a method in a context that requires a static method.
On the other hand, if you have a \java{Deck} object, you can use it to invoke a static method:
\begin{code}
Deck deck = new Deck();
int i = deck.randomInt(0, 51); // legal, but not good style
\end{code}
This is legal, but it is not considered good style, because someone reading this code would expect \java{randomInt} to be an instance method.
Another common error is to use \java{this} in a static method.
For example, say you write something like this:
\begin{code}
private static Deck merge(Deck d1, Deck d2) {
return this.cards; // wrong!
}
\end{code}
You get a compiler error like this:
\begin{stdout}
Non-static variable this cannot be referenced from a
static context.
\end{stdout}
The problem is that \java{cards} is an instance variable, so it is {\em non-static}; therefore, you can't access it from a static method.
In general, you can't use \java{this} in a static method, because a static method is not invoked on an object.
For beginners, error messages about non-static context can be confusing and frustrating.
We hope this section helps.
%\index{sort!Arrays}
%\index{array!sorting}
%Normally we wouldn't implement two different sorting algorithms in the same class.
%Our goal with \java{Deck} was to demonstrate static methods and different ways of solving the same problem.
%In practice, we could just write a single \java{sort} method that uses \java{java.util.Arrays}.
%
%\begin{code}
%public void sort() {
% Arrays.sort(this.cards);
%}
%\end{code}
\section{Piles of Cards}
\index{War (card game)}
Now that we have classes that represent cards and decks, let's use them to make a game.
One of the simplest card games that children play is called ``War'' (see \url{https://en.wikipedia.org/wiki/War_(card_game)}).
Initially, the deck is divided evenly into two piles, one for each player.
During each round, each player takes the top card from their pile and places it, face up, in the center.
Whoever has the highest-ranking card, ignoring suit, takes the two cards and adds them to the bottom of their pile.
The game continues until one player has won the entire deck.
We could use the \java{Deck} class to represent the individual piles.
However, our implementation of \java{Deck} uses a \java{Card} array, and the length of an array can't change.
As the game progresses, we need to be able to add and remove cards from the piles.
\index{ArrayList}
\index{collection}
We can solve this problem with an \java{ArrayList}, which is in the \java{java.util} package.
An \java{ArrayList} is a {\bf collection}, which is an object that contains other objects.
It provides methods to add and remove elements, and it grows and shrinks automatically.
%The Java library includes many other collections (see \url{https://thinkjava.org/collections}).
%For our purposes, \java{ArrayList} is a good choice because it provides methods to add and remove elements, and it grows and shrinks automatically.
\index{Pile}
We define a new class named \java{Pile} to represent a pile of cards.
It uses an \java{ArrayList} to store \java{Card} objects:
\begin{code}
public class Pile {
private ArrayList<Card> cards;
public Pile() {
this.cards = new ArrayList<Card>();
}
}
\end{code}
\index{angle brackets}
\index{brackets!angle}
\index{\textless\textgreater\ angle brackets}
When you declare an \java{ArrayList}, you specify the type it contains in angle brackets (\java{<>}).
This declaration says that \java{cards} is not just an \java{ArrayList}; it's an \java{ArrayList} of \java{Card} objects.
The constructor initializes \java{this.cards} with an empty \java{ArrayList}.
%Java collections can only store objects, not primitives like \java{int}.
%But you can use wrapper classes, for example \java{ArrayList<Integer>}.
Now let's think about the methods we need to play the game.
At the beginning of each round, each player draws a card from the top of their pile.
So we define a method to do that:
% ABD: ML suggests renaming this drawCard. I am inclined to agree.
% CSM: we might want to avoid the word "draw" meaning "paint"
\begin{code}
public Card popCard() {
return this.cards.remove(0); // from the top of the pile
}
\end{code}
\java{popCard} removes the \java{Card} at the beginning of the \java{ArrayList}, which we think of as the top of the pile.
Because we use \java{ArrayList.remove}, it automatically shifts the remaining cards to fill the gap.
At the end of each round, the winner adds cards to the bottom of their pile.
So we define a method to do that:
\begin{code}
public void addCard(Card card) {
this.cards.add(card); // to the bottom of the pile
}
\end{code}
\java{ArrayList} provides a method, \java{add}, that adds an element to the end of the collection, which we think of as the bottom of the pile.
To know when to stop the game, we have to check if one of the piles is empty.
Here's a method to do that:
\begin{code}
public boolean isEmpty() {
return this.cards.isEmpty();
}
\end{code}
So far, these methods don't do very much; they just invoke methods on the instance variable, \java{cards}.
Methods like these are called {\bf wrapper methods} because they wrap one method with another.
Finally, to start the game, we need to divide the deck into two equal parts.
We can do that with \java{subdeck} from Section~\ref{subdeck} and a new method, \java{addDeck}:
\begin{code}
public void addDeck(Deck deck) {
for (Card card : deck.getCards()) {
this.cards.add(card);
}
}
\end{code}
\java{addDeck} takes a \java{Deck} object, loops through the cards, and adds them to the \java{Pile}.
Notice that it does not remove the cards from the \java{Deck}, so the \java{Deck} and the \java{Pile} share cards.
But that won't be a problem because cards are immutable.
\section{Playing War}
Now we can use \java{Deck} and \java{Pile} to implement the game.
We'll start by creating a deck and shuffling:
\begin{code}
Deck deck = new Deck();
deck.shuffle();
\end{code}
Then we divide the \java{Deck} into two piles:
\begin{code}
Pile p1 = new Pile();
p1.addDeck(deck.subdeck(0, 25));
Pile p2 = new Pile();
p2.addDeck(deck.subdeck(26, 51));
\end{code}
The game itself is a loop that repeats until one of the piles is empty.
At each iteration, we draw a card from each pile and compare their ranks:
\begin{code}
while (!p1.isEmpty() && !p2.isEmpty()) {
// pop a card from each pile
Card c1 = p1.popCard();
Card c2 = p2.popCard();
// compare the cards
int diff = c1.getRank() - c2.getRank();
if (diff > 0) {
p1.addCard(c1);
p1.addCard(c2);
} else if (diff < 0) {
p2.addCard(c1);
p2.addCard(c2);
} else {
// it's a tie
}
\end{code}
If the two cards have the same rank, it's a tie.
In that case, each player draws four more cards.
Whoever has the higher fourth card takes all cards in play.
If there's another tie, they draw another four cards, and so on.
One of the exercises at the end of this chapter asks you to implement the \java{else} block when there's a tie.
After the \java{while} loop ends, we display the winner based on which pile is not empty:
\begin{code}
if (p2.isEmpty()) {
System.out.println("Player 1 wins!");
} else {
System.out.println("Player 2 wins!");
}
\end{code}
\java{ArrayList} provides many other methods that we didn't use for this example.
Take a minute to read the documentation, which you can find by doing a web search for ``Java ArrayList''.
\section{Vocabulary}
\begin{description}
\term{pseudocode}
A way of designing programs by writing rough drafts in a combination of English and Java.
\term{helper method}
A method that implements part of a more complex algorithm; often it is not particularly useful on its own.
\term{top-down design}
Breaking down a problem into subproblems, and solving each subproblem one at a time.
\term{selection sort}
A simple sorting algorithm that searches for the smallest or largest element $n$ times.
\term{merge sort}
A recursive sorting algorithm that divides an array into two parts, sorts each part (using merge sort), and merges the results.
\term{off-by-one}
A common programming mistake that results in iterating one time too many, or too few.
\term{static context}
The parts of a class that run without reference to a specific instance of the class.
\term{collection}
A Java library class, like \java{ArrayList}, that represents a group of objects.
\term{wrapper method}
A method that calls another method without doing much additional work.
%\term{insertion sort}
%Another sorting algorithm that inserts elements into place, one at a time.
\end{description}
\section{Exercises}
The code for this chapter is in the {\it ch13} directory of {\it ThinkJavaCode2}.
See page~\pageref{code} for instructions on how to download the repository.
Before you start the exercises, we recommend that you compile and run the examples.
\begin{exercise} %%V6 Ex13.5
Write a \java{toString} method for the \java{Deck} class.
It should return a single string that represents the cards in the deck.
When it's printed, this string should display the same results as the \java{print} method in Section~\ref{deck}.
\index{StringBuilder}
\index{efficiency}
{\em Hint:} You can use the \java{+} operator to concatenate strings, but that is not very efficient.
Consider using \java{StringBuilder} instead; see Section~\ref{stringbuilder}.
\end{exercise}
\begin{exercise} %%V6 Ex13.2
\label{ex.shuffle}
The goal of this exercise is to implement the shuffling algorithm from this chapter.
\begin{enumerate}
\item In the repository for this book, you should find the file named {\it Deck.java}.
Check that you can compile it in your environment.
\item Implement the \java{randomInt} method.
You can use the \java{nextInt} method provided by \java{java.util.Random}, which you saw in Section~\ref{random}.
{\em Hint:} To avoid creating a \java{Random} object every time \java{randomInt} is invoked, consider defining a class variable.
\item Write a \java{swapCards} method that takes two indexes and swaps the cards at the given locations.
\item Fill in the \java{shuffle} method by using the algorithm in Section~\ref{shuffle}.
\end{enumerate}
\end{exercise}
\begin{exercise} %%V6 Ex13.3
The goal of this exercise is to implement the sorting algorithms from this chapter.
Use the {\it Deck.java} file from the previous exercise or create a new one from scratch.
\begin{enumerate}
\item Implement the \java{indexLowest} method.
Use the \java{Card.compareTo} method to find the lowest card in a given range of the deck, from \java{lowIndex} to \java{highIndex}, including both.
\item Fill in \java{selectionSort} by using the algorithm in Section~\ref{sorting}.
\item Using the pseudocode in Section~\ref{mergesort}, implement the \java{merge} method.
The best way to test it is to build and shuffle a deck.
Then use \java{subdeck} to form two small subdecks, and use selection sort to sort them.
Finally, pass the two halves to \java{merge} and see if it works.
\index{testing}
\item Fill in \java{almostMergeSort}, which divides the deck in half, then uses \java{selectionSort} to sort the two halves, and uses \java{merge} to create a new, sorted deck.
You should be able to reuse code from the previous step.
\item Implement \java{mergeSort} recursively.
Remember that \java{selectionSort} is \java{void} and \java{mergeSort} returns a new \java{Deck}, which means that they get invoked differently:
\begin{code}
deck.selectionSort(); // modifies an existing deck
deck = deck.mergeSort(); // replaces old deck with new
\end{code}
\end{enumerate}
\end{exercise}
\begin{exercise}
%%V6 Ex13.1
You can learn more about the sorting algorithms presented in this chapter at \url{https://www.toptal.com/developers/sorting-algorithms}.
This site provides explanations of the algorithms, along with animations that show how they work.
It also includes an analysis of their efficiency.
%%V6 Ex13.4
For example, ``insertion sort'' is an algorithm that inserts elements into place, one at a time.
Read about it on the website and play the animations.
Then write a method named \java{insertionSort} that implements this algorithm.
One goal of this exercise is to practice top-down design.
Your solution should use a helper method, named \java{insert}, that implements the inner loop of the algorithm.
\java{insertionSort} should invoke this method $n-1$ times.
\end{exercise}
\begin{exercise} %%V6.5 NEW
Find and open the file {\it War.java} in the repository.
The \java{main} method contains all the code from the last section of this chapter.
Check that you can compile and run this code before proceeding.
The program is incomplete; it does not handle the case when two cards have the same rank.
Finish implementing the \java{main} method, beginning at the line that says: \java{// it's a tie}.
When there's a tie, draw three cards from each pile and store them in a collection, along with the original two.
Then draw one more card from each pile and compare them.
Whoever wins the tie takes all ten of these cards.
If one pile does not have at least four cards, the game ends immediately.
If a tie ends with a tie, draw three more cards, and so on.
Notice that this program depends on \java{Deck.shuffle}, so you might have to do Exercise~\ref{ex.shuffle} first.
\end{exercise}