2015-09-07 11:44:36 -06:00
|
|
|
// generics/Fill2.java
|
2015-12-15 11:47:04 -08:00
|
|
|
// (c)2016 MindView LLC: see Copyright.txt
|
2015-11-15 15:51:35 -08:00
|
|
|
// We make no guarantees that this code is fit for any purpose.
|
|
|
|
// Visit http://mindviewinc.com/Books/OnJava/ for more book information.
|
2015-06-15 17:47:35 -07:00
|
|
|
// Using adapters to simulate latent typing.
|
|
|
|
// {main: Fill2Test}
|
|
|
|
import generics.coffee.*;
|
|
|
|
import java.util.*;
|
2015-11-03 12:00:44 -08:00
|
|
|
import java.util.function.*;
|
2015-06-15 17:47:35 -07:00
|
|
|
|
|
|
|
interface Addable<T> { void add(T t); }
|
|
|
|
|
|
|
|
public class Fill2 {
|
|
|
|
// Classtoken version:
|
|
|
|
public static <T> void fill(Addable<T> addable,
|
|
|
|
Class<? extends T> classToken, int size) {
|
|
|
|
for(int i = 0; i < size; i++)
|
|
|
|
try {
|
|
|
|
addable.add(classToken.newInstance());
|
|
|
|
} catch(InstantiationException |
|
|
|
|
IllegalAccessException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
2015-11-03 12:00:44 -08:00
|
|
|
// Supplier version:
|
2015-06-15 17:47:35 -07:00
|
|
|
public static <T> void fill(Addable<T> addable,
|
2015-11-03 12:00:44 -08:00
|
|
|
Supplier<T> generator, int size) {
|
2015-06-15 17:47:35 -07:00
|
|
|
for(int i = 0; i < size; i++)
|
2015-11-03 12:00:44 -08:00
|
|
|
addable.add(generator.get());
|
2015-06-15 17:47:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// To adapt a base type, you must use composition.
|
|
|
|
// Make any Collection Addable using composition:
|
|
|
|
class AddableCollectionAdapter<T> implements Addable<T> {
|
|
|
|
private Collection<T> c;
|
|
|
|
public AddableCollectionAdapter(Collection<T> c) {
|
|
|
|
this.c = c;
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public void add(T item) { c.add(item); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// A Helper to capture the type automatically:
|
|
|
|
class Adapter {
|
|
|
|
public static <T>
|
|
|
|
Addable<T> collectionAdapter(Collection<T> c) {
|
|
|
|
return new AddableCollectionAdapter<>(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// To adapt a specific type, you can use inheritance.
|
|
|
|
// Make a SimpleQueue Addable using inheritance:
|
|
|
|
class AddableSimpleQueue<T>
|
|
|
|
extends SimpleQueue<T> implements Addable<T> {
|
|
|
|
@Override
|
|
|
|
public void add(T item) { super.add(item); }
|
|
|
|
}
|
|
|
|
|
|
|
|
class Fill2Test {
|
|
|
|
public static void main(String[] args) {
|
|
|
|
// Adapt a Collection:
|
|
|
|
List<Coffee> carrier = new ArrayList<>();
|
|
|
|
Fill2.fill(
|
|
|
|
new AddableCollectionAdapter<>(carrier),
|
|
|
|
Coffee.class, 3);
|
|
|
|
// Helper method captures the type:
|
|
|
|
Fill2.fill(Adapter.collectionAdapter(carrier),
|
|
|
|
Latte.class, 2);
|
|
|
|
for(Coffee c: carrier)
|
2015-11-03 12:00:44 -08:00
|
|
|
System.out.println(c);
|
|
|
|
System.out.println("----------------------");
|
2015-06-15 17:47:35 -07:00
|
|
|
// Use an adapted class:
|
|
|
|
AddableSimpleQueue<Coffee> coffeeQueue =
|
|
|
|
new AddableSimpleQueue<>();
|
|
|
|
Fill2.fill(coffeeQueue, Mocha.class, 4);
|
|
|
|
Fill2.fill(coffeeQueue, Latte.class, 1);
|
|
|
|
for(Coffee c: coffeeQueue)
|
2015-11-03 12:00:44 -08:00
|
|
|
System.out.println(c);
|
2015-06-15 17:47:35 -07:00
|
|
|
}
|
2015-09-07 11:44:36 -06:00
|
|
|
}
|
|
|
|
/* Output:
|
2015-06-15 17:47:35 -07:00
|
|
|
Coffee 0
|
|
|
|
Coffee 1
|
|
|
|
Coffee 2
|
|
|
|
Latte 3
|
|
|
|
Latte 4
|
|
|
|
----------------------
|
|
|
|
Mocha 5
|
|
|
|
Mocha 6
|
|
|
|
Mocha 7
|
|
|
|
Mocha 8
|
|
|
|
Latte 9
|
2015-09-07 11:44:36 -06:00
|
|
|
*/
|