forked from nus-cs2103-AY1819S1/addressbook-level1
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
docs: refactor out LearningOutcomes into a separate file
Do this for consistency with the other AddressBook levels.
- Loading branch information
Showing
2 changed files
with
269 additions
and
264 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -13,272 +13,9 @@ endif::[] | |
the main programming language. | ||
* It provides a *reasonably well-written* code example that is *significantly bigger* than what students | ||
usually write in data structure modules. | ||
* It can be used to achieve a number of beginner-level <<learning-outcomes>> without | ||
* It can be used to achieve a number of beginner-level <<LearningOutcomes#, Learning Outcomes>> without | ||
running into the complications of OOP or GUI programmings. | ||
== Learning Outcomes | ||
|
||
_Learning Outcomes_ are the things you should be able to do after studying this code and completing the | ||
corresponding exercises. | ||
|
||
=== Set up a project in an IDE `[LO-IdeSetup]` | ||
|
||
* Learn https://se-edu.github.io/se-book/intellij/projectSetup/[how to set up a project in IntelliJ]. | ||
|
||
==== Exercise: Setup project in IntelliJ | ||
|
||
Part A: | ||
|
||
* Create a new project in IntelliJ and write a small HelloWorld program. | ||
|
||
Part B: | ||
|
||
* Download the source code for this project: Click on the `clone or download` link above and either, | ||
. download as a zip file and unzip content. | ||
. clone the repo (if you know how to use Git) to your Computer. | ||
* <<DeveloperGuide#SettingUp, Set up>> the project in IntelliJ. | ||
* <<UserGuide#StartingTheProgram, Run the program>> from within IntelliJ, and try the features described in | ||
the <<UserGuide#, User Guide>> section. | ||
|
||
=== Navigate code efficiently `[LO-CodeNavigation]` | ||
|
||
The `AddressBook.java` code is rather long, which makes it cumbersome to navigate by scrolling alone. | ||
Navigating code using IDE shortcuts is a more efficient option. | ||
For example, CTRL+B will navigate to the definition of the method/field at the cursor. | ||
|
||
Learn https://se-edu.github.io/se-book/intellij/codeNavigation/[basic IntelliJ code navigation features]. | ||
|
||
==== Exercise: Learn to navigate code using shortcuts | ||
|
||
* Use Intellij basic code navigation features to navigate code of this project. | ||
|
||
=== Use a debugger `[LO-Debugging]` | ||
|
||
Learn https://se-edu.github.io/se-book/intellij/debuggingBasic/[basic IntelliJ debugging features]. | ||
|
||
==== Exercise: Learn to step through code using the debugger | ||
|
||
Prerequisite: `[LO-IdeSetup]` | ||
|
||
Demonstrate your debugging skills using the AddressBook code. | ||
|
||
Here are some things you can do in your demonstration: | ||
|
||
. Set a 'break point' | ||
. Run the program in debug mode | ||
. 'Step through' a few lines of code while examining variable values | ||
. 'Step into', and 'step out of', methods as you step through the code | ||
. ... | ||
|
||
=== Automate CLI testing `[LO-AutomatedCliTesting]` | ||
|
||
Learn https://se-edu.github.io/se-book/testing/testAutomation/testingTextUis/[how to automate testing of CLIs]. | ||
|
||
==== Exercise: Practice automated CLI testing | ||
|
||
* Run the tests as explained in the <<DeveloperGuide#Testing, Testing>> section. | ||
* Examine the test script to understand how the script works. | ||
* Add a few more tests to the `input.txt`. Run the tests. It should fail. | ||
Modify `expected.txt` to make the tests pass again. | ||
* Edit the `AddressBook.java` to modify the behavior slightly and modify tests to match. | ||
|
||
=== Use Collections `[LO-Collections]` | ||
|
||
Note how the `AddressBook` class uses `ArrayList<>` class (from the Java `Collections` library) to store a list of `String` or `String[]` objects. | ||
|
||
Learn https://se-edu.github.io/se-book/javaTools/collections/[how to use some Java `Collections` classes, such as `ArrayList` and `HashMap`] | ||
|
||
==== Exercise: Use `HashMap` | ||
|
||
Currently, a person's details are stored as a `String[]`. Modify the code to use a `HashMap<String, String>` instead. | ||
A sample code snippet is given below. | ||
|
||
[source,java] | ||
---- | ||
private static final String PERSON_PROPERTY_NAME = "name"; | ||
private static final String PERSON_PROPERTY_EMAIL = "email"; | ||
... | ||
HashMap<String,String> john = new HashMap<>(); | ||
john.put(PERSON_PROPERTY_NAME, "John Doe"); | ||
john.put(PERSON_PROPERTY_EMAIL, "[email protected]"); | ||
//etc. | ||
---- | ||
|
||
=== Use Enums `[LO-Enums]` | ||
|
||
==== Exercise: Use `HashMap` + `Enum` | ||
|
||
Similar to the exercise in the `LO-Collections` section, but also bring in Java `enum` feature. | ||
|
||
[source,java] | ||
---- | ||
private enum PersonProperty {NAME, EMAIL, PHONE}; | ||
... | ||
HashMap<PersonProperty,String> john = new HashMap<>(); | ||
john.put(PersonProperty.NAME, "John Doe"); | ||
john.put(PersonProperty.EMAIL, "[email protected]"); | ||
//etc. | ||
---- | ||
|
||
=== Use Varargs `[LO-Varargs]` | ||
|
||
Note how the `showToUser` method uses https://se-edu.github.io/se-book/javaTools/varargs/[Java Varargs feature]. | ||
|
||
==== Exercise: Use Varargs | ||
|
||
Modify the code to remove the use of the Varargs feature. | ||
Compare the code with and without the varargs feature. | ||
|
||
=== Follow a coding standard `[LO-CodingStandard]` | ||
|
||
The given code follows the https://github.com/oss-generic/process/blob/master/codingStandards/CodingStandard-Java.md[coding standard] | ||
for the most part. | ||
|
||
This learning outcome is covered by the exercise in `[LO-Refactor]`. | ||
|
||
=== Apply coding best practices `[LO-CodingBestPractices]` | ||
|
||
Most of the given code follows the best practices mentioned | ||
https://se-edu.github.io/se-book/codeQuality/[here]. | ||
|
||
This learning outcome is covered by the exercise in `[LO-Refactor]` | ||
|
||
=== Refactor code `[LO-Refactor]` | ||
|
||
*Resources*: | ||
|
||
* https://se-edu.github.io/se-book/refactoring/[se-edu/se-book: Refactoring] | ||
* https://se-edu.github.io/se-book/intellij/refactoring/[se-edu/se-book: Refactoring in Intellij] | ||
|
||
==== Exercise: Refactor the code to make it better | ||
|
||
Note: this exercise covers two other Learning Outcomes: `[LO-CodingStandard]`, `[LO-CodingBestPractices]` | ||
|
||
* Improve the code in the following ways, | ||
** Fix https://github.com/oss-generic/process/blob/master/codingStandards/CodingStandard-Java.md[coding standard] | ||
violations. | ||
** Fix violations of the best practices given in https://se-edu.github.io/se-book/codeQuality/[in this document]. | ||
** Any other change that you think will improve the quality of the code. | ||
* Try to do the modifications as a combination of standard refactorings given in this | ||
http://refactoring.com/catalog/[catalog] | ||
* As far as possible, use automated refactoring features in IntelliJ. | ||
* If you know how to use Git, commit code after each refactoring. | ||
In the commit message, mention which refactoring you applied. | ||
Example commit messages: `Extract variable isValidPerson`, `Inline method isValidPerson()` | ||
* Remember to run the test script after each refactoring to prevent https://se-edu.github.io/se-book/testing/testingTypes/regressionTesting[regressions]. | ||
|
||
=== Abstract methods well `[LO-MethodAbstraction]` | ||
|
||
Notice how most of the methods in `AddressBook` are short and focused (does only one thing and does it well). | ||
|
||
*Case 1*. Consider the following three lines in the `main` method. | ||
|
||
[source,java] | ||
---- | ||
String userCommand = getUserInput(); | ||
echoUserCommand(userCommand); | ||
String feedback = executeCommand(userCommand); | ||
---- | ||
|
||
If we include the code of `echoUserCommand(String)` method inside the `getUserInput()` | ||
(resulting in the code given below), the behavior of AddressBook remains as before. | ||
However, that is a not a good approach because now the `getUserInput()` is doing two distinct things. | ||
A well-abstracted method should do only one thing. | ||
|
||
[source,java] | ||
---- | ||
String userCommand = getUserInput(); //also echos the command back to the user | ||
String feedback = executeCommand(userCommand); | ||
---- | ||
|
||
*Case 2*. Consider the method `removePrefixSign(String s, String sign)`. | ||
While it is short, there are some problems with how it has been abstracted. | ||
|
||
. It contains the term `sign` which is not a term used by the AddressBook vocabulary. | ||
+ | ||
[NOTE] | ||
==== | ||
*A method adds a new term to the vocabulary used to express the solution*. | ||
Therefore, it is not good when a method name contains terms that are not strictly necessary to express the | ||
solution (e.g. there is another term already used to express the same thing) or not in tune with the solution | ||
(e.g. it does not go well with the other terms already used). | ||
==== | ||
|
||
. Its implementation is not doing exactly what is advertised by the method name and the header comment. | ||
For example, the code does not remove only prefixes; it removes `sign` from anywhere in the `s`. | ||
. The method can be _more general_ and _more independent_ from the rest of the code. For example, | ||
the method below can do the same job, but is more general (works for any string, not just parameters) | ||
and is more independent from the rest of the code (not specific to AddressBook) | ||
+ | ||
[source,java] | ||
---- | ||
/** | ||
* Removes prefix from the given fullString if prefix occurs at the start of the string. | ||
*/ | ||
private static String removePrefix(String fullString, String prefix) { ... } | ||
---- | ||
+ | ||
If needed, a more AddressBook-specific method that works on parameter strings only can be defined. | ||
In that case, that method can make use of the more general method suggested above. | ||
|
||
==== Exercise: Improve abstraction of method | ||
|
||
Refactor the method `removePrefixSign` as suggested above. | ||
|
||
=== Follow SLAP `[LO-SLAP]` | ||
|
||
Notice how most of the methods in `AddressBook` are written at a single | ||
level of abstraction (_cf_ https://se-edu.github.io/se-book/codeQuality/practices/slapHard/[se-edu/se-book:SLAP]) | ||
|
||
Here is an example: | ||
|
||
[source,java] | ||
---- | ||
public static void main(String[] args) { | ||
showWelcomeMessage(); | ||
processProgramArgs(args); | ||
loadDataFromStorage(); | ||
while (true) { | ||
userCommand = getUserInput(); | ||
echoUserCommand(userCommand); | ||
String feedback = executeCommand(userCommand); | ||
showResultToUser(feedback); | ||
} | ||
} | ||
---- | ||
|
||
==== Exercise 1: Reduce SLAP of method | ||
|
||
In the `main` method, replace the `processProgramArgs(args)` call with the actual code of that method. | ||
The `main` method no longer has SLAP. Notice how mixing low level code with high level code reduces | ||
readability. | ||
|
||
==== Exercise 2: Refactor the code to make it worse! | ||
|
||
Sometimes, going in the wrong direction can be a good learning experience too. | ||
In this exercise, we explore how low code qualities can go. | ||
|
||
* Refactor the code to make the code as bad as possible. | ||
i.e. How bad can you make it without breaking the functionality while still making it look like it was written by a | ||
programmer (but a very bad programmer :-)). | ||
* In particular, inlining methods can worsen the code quality fast. | ||
|
||
=== Work in a 1kLoC code base `[LO-1KLoC]` | ||
|
||
==== Exercise: Enhance the code | ||
|
||
Enhance the AddressBook to prove that you can work in a codebase of 1KLoC. | ||
Remember to change code in small steps, update/run tests after each change, and commit after each significant change. | ||
|
||
Some suggested enhancements: | ||
|
||
* Make the `find` command case insensitive e.g. `find john` should match `John` | ||
* Add a `sort` command that can list the persons in alphabetical order | ||
* Add an `edit` command that can edit properties of a specific person | ||
* Add an additional field (like date of birth) to the person record | ||
|
||
''''' | ||
|
||
== Contributors | ||
|
||
The full list of contributors for se-edu can be found https://se-edu.github.io/Team.html[here]. | ||
|
Oops, something went wrong.