An online markdown blog and knowledge repository.
23-May through 27-May 2022
Inheritance: Super and Extends => Create sub-classes.
Abstraction: Abstract Classes => Stepping "out one layer" from doing the actual thing.
DRY: Code modularization => Don't Repeat Yourself => Why keep writing the same code over and over?
Object hierarchy: Built-in methods get inherited down the tree.
Everything is Java is either a Class aka Object aka REF type, or a Primative.
Type casting: Upcasting (free in Java) and Downcasting (manual)
Instance of: IS-A => Is this Type type an instance of AnotherType.
Arguments: Have actual value and are set when a method is called.
Parameters: Are placeholders for Arguments that are expected as method inputs.
Primatives: Long, Double, Short, Char, Integer, Decimal, Float, Byte, Boolean.
Keyword 'this': Use to limit the scope, i.e. within a Constructor or inside a specific Class and not it's base or Parent Class.
Objects do NOT have a 'this' keyword, but CLASSES need this.
Dot Notation is equivalent to Class Notation => ClassName.memberName
Encapsulation: Data Hiding => Do not show Properties to other Classes.
Getters & Setters => Methods that enable reading from, and writing to, Properties.
Use methods (getters and setters) to retrieve and update Properties.
Not a primative? Then it is a Reference Type.
Keyword 'new' => Results in creating a new Object (instance).
70/30 Time Split: Devs spent more time planning their project (70%) over writing it (30%).
Passing by Value vs Passing by Reference => High level, parameter values are copied or a reference to the memory location is copied.
Sub Class == Child Class == Inheriting Class.
List out all of the things and their properties into boxes.
Look for overlaps between the items and push any common members 'up the hierarchy'.
Identify member types including Primatives or Classes (Built-in or Custom).
Once the Member details and Hierarchical arrangement are drawn out, THEN open your IDE and start coding the Classes and inheritance.
Use inheritance in Class definitions while coding, e.g.:
public class Artist {
protected String name;
String genre;
public void writeMusic() {
System.out.println("boopy-dooby");
}
}
public class Musician extends Artist {
...
}
public class Composer extends Artist {
...
}
Write Tests! Use the IntelliJ generator to help you write test methods.
Constructors of child classes must be created too!
We can create Constructors that are rooted in the Class's Parent Class.
public class Musician extends Artist {
public int totalSongs;
public Musician(String name, String genre, int totalSongs) {
super(name, genre); // this is calling the parent constructor grabbing its properties
this.totalSongs = totalSongs;
}
}
// private fields in the parent will NOT show via the super() constructor call
Packages vs Sub Classes: A sub class might be imported via another Package, so Access Modifiers are important to understand.
While domain modeling, ask yourself: Which Class should be responsible?
Inheritance introduces the 'instanceOf()' method. It is used to determine inherited hierarchy of your current (or custom) class at runtime.
You cannot instantiate an Abstract Class => NO NEW KEYWORD!
Abstract Classes: Write your "big logic" code here so all inheriting classes implement it without rewriting the same code.
Keyword 'abstract', e.g.:
public abstract class Artist {
protected String name;
String genre;
}
Abstraction: Hide complexities of an implementation. Only expose an API that includes implementations that you want inheritors to use.
Abstract Members: Use the 'abstract' keyword to define methods that tells inheritors they must implement it.
Override: Used to implement abstract methods.
Overload: Create multiple methods using the same name but changing the number or input parameters and possibly the return type.
MVP: Pare this down more than you want to for your final! Avoids.
Passing by value: This is the default operation in Java, primatives and classes.
Java passes REFERENCES by value by default.
Passing by reference: MUST BE SPECIFIED with a keyword.
Call Stack: Where temporary memory gets stored and methods are stored during execution.
Heap: Longer-term memory storage of object instances and their data.
Reference Addresses: Stored in the Stack, point to memory locations in the Heap e.g. to an Object.
GSon: Stringifying objects.
JSON: Similar to javascript objects (braces surrounding KVPs), using string-based keys.
Java cannot natively read JSON:
You will need to spend ~70% of your time whiteboarding a problem and only ~30% coding the solution!
The order in which you reassign REFs to Nodes is critical.
Always reassign the Head pointer LAST.
Recall primitives and references are core aspects of Java's Type system.
Two types:
For example, an int fits inside a double (lossless conversion) but a double does not fit inside an integer (lossy conversion, requires implementing a conversion process).
Casting sub-class to super-class => UPCASTING.
Notes about Upcasting:
Downcasting is the process of casting "from a superclass to a subclass".
instanceOf(subclass)
in an IF statement before downcasting!Supplemental comments were added from these resources:
Use MyClass.class.cast(superClassRef) and MyClass.class.isInstance(superClassRef).
cast()
and isInstance()
methods together.Supplemental comments were added from an article at Baeldung.com on java type casting
What are Generics?
public class Node<T> {
public T value; // value of node
public Node<T> next; // generic Node with type T
public Node(T value) {
this.value = value; // works within constructors and methods
}
}
public class LinkedList<T> {
public Node<T> head = null;
public Node<T> tail;
public void insert(T val) {
Node<T> newNode = new Node(val);
if (head != null) {
newNode.next = head;
}
if (head == null) {
tail = newNode;
}
head = newNode;
}
...
}
Class Number is the parent to int, long, float, double, etc.
Restricts the Types to certain super and child types, rather than the entire Java Libraries types.
Similar to C# "Constraints".
public class Node<T extends MyClass> {...}
Check out this article at Baeldung.com on java type casting for information about casting within generic methods to safely handle generic types.
At the bottom of the article is a demonstration unit test that verifies only the correct type is handled in the generic method under test.
How do we know how the user is going to interact with our App/Packages? The WRRC and API modeling can help answer these questions.
We will also need to consider validating input, etc.
All of this will get revisited in a future class.
For now just keep in mind user-input and output are impotant aspects with code design.
One to One: 1:1 => Each pair of items are directly related to each other and only each other.
One to Many: 1:Infinite => The single has many, many links to other items but each item has only one link to the one.
Many to Many: Infinite:Infinite => Many Items have many links to many other items.
What does good coding practices say about a class that is used A LOT? Modularize it!
Consider whether releated objects have a relationship that indicates whether inheritance with sub-classing or abstract classes etc are necessary.
Why are these necessary? An interface is like a contract, that guarantees a class or Type will have certain members.
Interfaces are NOT Classes but they are instantiated similarly: public interface iMyInterface {...}
Only implement logic and methods as required.
To attach an interface to a Class, use the keyword "implements": public class myClass implements iMyInterface {...}
Notes:
Create a Logger interface that all of your classes implement to ensure logging in your App!
public class Zork extends Doggo implements Feeding, logger {
// now implement the members required by Feeding and logger interfaces
}
More linked lists!
KEY CODE TO REMEMBER FOR LINKED LISTS: while(current != null) {...}
Keywords:
Push as many members UP the hierarchy to a parent class as possible to simplify domain model design.
Use GETTERS so that other Objects have access to information in parent, sibling, and other Objects.
super. vs super():
super.
: References the parent class using object (dot) notation.super()
: invokes the parent Constructor, based on parameter list matching.Implicit Upcasting: A shortcut, affirms that a type IS-A, which is similar to how Generics are implemented.
DRY: Don't repeat yourself.
YAGNI: Don't implement what is not needed right now. Future-Proofing can end up violating this rule.
RuleOf3: If you're writing the same code for the 3rd+ time, consider modularizing it into a common or helper function.
MVP: Minimum Viable Product => An experiment that early, visionaries get to use and provide feedback of an App while it is a mimumally usable (but functional) state.
KISS: Keep is simple. Don't over-complicate the code, just solve the problem at hand, as directly as possible.
SOLID: A set of principles.
Robert C. Martin and Michael Feathers are credited with building these principles while developing software design patterns.
Single Responsibility: A class should have one job. There should only be one reason to change.
Open-Closed: Open for extension, Closed for modification.
Liskov Substitution: Related to polymorphism.
Interface Segregation: Larger interfaces should be split into smaller ones.
Dependency Inversion: Decouple software modules, and depend on abstractions.
This section was completed by taking notes from Baeldung.com article on solid principles
PM: Product or Project Manager
TL: Team Lead
Senior Dev: Assists with managing the developer processes and workload.
Clients/customers: Will have a set of requirements, wants/needs that need to get turned into User Stories.
Take customer/client requirements and develop user stories.
The Dev (You): Implement these User Stories in code.
How do you name your variables and members? Snake_Case? Skewer-Case?
How do you write tests?
Where do you throw exceptions?
How do you name your Classes, Interfaces, etc.
Single-line or multi-line conditionals?
How many tab spaces are standard? 2? 4? Some other number?
How is the directory architecture set up in your project(s)?
My team / employer might have a code style guide and you will need to follow it.
If a guide doesn't exist MAKE ONE! It will help others create consistent code.
Code Linters: The Linter enforces (or warns) about code style violations.
IDs and UUIDs: One is for SQL, the other is for you (the software or API).
Data that we are storing within classes today, will actually be stored in SQL, in real life.
Relational checking a-la: 'Where foreignKey = itemId'
GSON is an external package: Google Script Object Notation.
Lab Goal today: Read-in a JSON file using GSON.
When reading-in JSON, you will need to have a Class instance to push the imported data into (members etc).
Design:
implementation 'com.google.code.gson:n.n.n'
=> add this (and more) to build.gradle.import com.google.gson.Gson;
Read-in a file:
File myFile = new File("./app/src/test/resources/JSON.json");
FileReader fileReader = new FileReader(myFile);
MyClass myClass = new FileReader(fileReader, Class.class);
Write-out a file:
File zorkFile2 = new File(path_to_file_output_file.json);
try(FileWriter zorkFileWriter = new FileWriter(zorkFile2)) {...}
gson.toJson(newZork, zorkFileWriter);
Zip-up 2 linked lists!
Manage your pointers and references.
Use a TEMP variable!
MAINTAIN refs to the nodes with access to the rest of your Linked Lists.
The only time you need to use the TypeToken type, is when getting a collection of JSON objects using the Gson package.
Whenever you make an API call that returns a JSON Array, it will be enclosed in [ ]
and the TypeToken code will be required.
With GSON, using a Class to schema the data, a constructor is not required.
Create fun and interactive websites, not single-page static sites.
UI vs UX: User Interface is one thing, User Experience design interactive, intuitive user interfaces, especially websites.
UX Designers study user psychology and design patterns for useability and interactiveness.
HttpUrlConnection: Class we will use, based on advice from Baeldug.com, but we'll include Try-With-Resources.
URL pokeURL = new URL(stringURL);
HttpURLConnection myConnection = (HttpURLConnection) myURL.openConnection();
Wrapper Classes:
Two primery methods of getting data from APIs:
GET: Query, no payload (data) is sent to the API but response is required.
PUT, PATCH, and DELTE: Action methods that require data to be sent TO the API.
try (BufferedReader responseReader = new BufferedReader(args)) {...}
: This auto-closes (garbage-collects) the enclosed resources automatically.ArrayList<T>
and nested Classes that define T in order to fully model the JSON data.There are two methods of traversal:
A function that calls itself is a Recursive Function.
Remember: The Call Stack is a LIFO system and recursive functions take advantage of that.
Note: At every single iteration the base-case MUST BE TESTED.
Return to Parent Readme.md