Flashcards for topic Classes and Interfaces
What is the primary rule of thumb for accessibility in Java class design, and why is it considered fundamental?
Rule: Make each class or member as inaccessible as possible.
This means using the lowest possible access level consistent with proper functionality.
Why fundamental:
Implementation: Use private, package-private, protected, and public modifiers in order of increasing accessibility.
When implementing a Map's internal Entry objects, what is the correct design approach and why?
The correct approach is to implement Entry as a private static member class because:
public class HashMap<K,V> { // Other implementation details... // Correct implementation private static class Entry<K,V> { final K key; V value; // Methods don't need Map instance access public K getKey() { return key; } public V getValue() { return value; } public V setValue(V newValue) { V oldValue = value; value = newValue; return oldValue; } } }
What are the two implicit access levels introduced in Java 9's module system, and why might they be of limited utility to typical Java programmers?
Two implicit module-based access levels:
Limitations of utility:
Practical implication: Most Java programmers should focus on the four traditional access levels (private, package-private, protected, public) unless there's a compelling need for module encapsulation.
Current recommendation: Avoid relying on module-based access levels unless specifically needed for large systems with clear module boundaries.
What are the potential issues with public mutable fields in Java classes and why are they particularly problematic for thread-safety?
Issues with public mutable fields:
Thread-safety problems:
Example of vulnerability:
public class UnsafeCounter { public int count; // Thread-unsafe // Multiple threads calling increment() will lose updates public void increment() { count++; // Not atomic, can result in lost updates } }
Better approach: Use private fields with synchronized accessor methods or thread-safe data structures.
What fundamental problem occurs when inheritance is used with the HashSet's addAll method in a custom counting implementation, and how does it manifest?
The problem is self-use in the parent class, which violates encapsulation and creates fragile code:
add()
and addAll()
in a subclass of HashSet:
HashSet.addAll()
internally calls add()
for each elementadd()
in the subclass is called by the parent's implementation// When executing this code: InstrumentedHashSet<String> s = new InstrumentedHashSet<>(); s.addAll(List.of("Snap", "Crackle", "Pop")); // Expected count: 3 // Actual count: 6 (each element is counted twice)
This demonstrates why implementation inheritance breaks encapsulation - the subclass depends on implementation details of the superclass that aren't part of its contract.
Why are immutable objects inherently thread-safe, and what specific advantages does this provide in concurrent programming?
Immutable objects are inherently thread-safe because:
Specific advantages in concurrent programming:
Example: String and BigInteger are thread-safe with no synchronization overhead, while their mutable counterparts (StringBuilder and BitSet) require careful synchronization in concurrent contexts.
What are the three common problems that can arise when extending a class that wasn't designed for inheritance, and how does each problem manifest?
Three common problems:
Dependency on self-use patterns:
addAll
calling add
)InstrumentedHashSet
counting elements added via addAll
twiceVulnerability to superclass evolution:
Exposure to implementation details:
These problems stem from inheritance violating encapsulation by exposing implementation details that would normally be hidden.
What permanent commitments are you making when designing a class for inheritance, and why does this impose strict constraints on future evolution of the class?
When designing a class for inheritance, you permanently commit to:
Self-use patterns of overridable methods:
Implementation decisions exposed through protected methods:
Implementation decisions exposed through protected fields:
These commitments severely constrain future evolution because:
This is why classes must be tested for inheritance before release, and why inheritance is appropriate only for genuine subtype relationships.
What are the key advantages of interfaces over abstract classes for defining types?
Advantages of interfaces over abstract classes:
Multiple implementations:
Retrofitting existing classes:
Defining mixins:
Non-hierarchical type frameworks:
Enable safer functionality enhancements:
Example showing combination capabilities:
// Base capability interfaces public interface Swimmer { void swim(); } public interface Flyer { void fly(); } public interface Runner { void run(); } // Combined capabilities - doesn't force a rigid hierarchy public interface FlyingFish extends Swimmer, Flyer { } public interface Duck extends Swimmer, Flyer, Runner { } // Any class can implement these as needed public class Penguin implements Swimmer, Runner { /*...*/ } public class Eagle implements Flyer { /*...*/ } public class Platypus implements Swimmer, Runner { /*...*/ }
With abstract classes, this flexibility would be impossible without massive code duplication.
What is an Adapter pattern in the context of nonstatic member classes, and when would you use it?
An Adapter pattern using nonstatic member classes:
Common examples:
public class MyList<E> implements List<E> { // List implementation details... @Override public Iterator<E> iterator() { return new MyIterator(); // Returns adapter } // Adapter pattern using nonstatic member class private class MyIterator implements Iterator<E> { private int position = 0; @Override public boolean hasNext() { // Accesses enclosing instance's state return position < MyList.this.size(); } @Override public E next() { // Accesses enclosing instance's methods return MyList.this.get(position++); } } }
Showing 10 of 68 cards. Add this deck to your collection to see all cards.