제네릭 타입 Generic Types
generic type은 타입에 걸쳐 패러미터화된 generic class 또는 interface입니다.
다음의 Box 클래스는 개념을 데모하기 위하여 수정됩니다.
A generic type is a generic class or interface that is parameterized over types. The following Box class will be modified to demonstrate the concept.
간단한 Box클래스 A Simple Box Class
어떤 타입의 객체에서도 연산되는 non-generic Box class를 검토하는 것으로 시작합니다.
이 클래스는 메소드 상자에 객체를 추가하는 set와 상자를 검색하는 get의 메소드 2개를 제공하면 됩니다.
Begin by examining a non-generic Box class that operates on objects of any type. It needs only to provide two methods: set, which adds an object to the box, and get, which retrieves it:
public class Box {
private Object object;
public void set(Object object) { this.object = object; }
public Object get() { return object; }
}
위의 메소드들은 Object를 수용하거나 리턴하기 때문에, 기본 타입 만 아니면, 그 안에 어떤 것이든 전달할 수 있습니다.
컴파일 시점에 클래스가 사용된 방법을 검증할 수 없습니다.
코드 중 일부는 상자 안에 Integer를 넣어서 그것으로부터 Integers 를 얻고자 할 수도 있고, 한편으로 코드의 다른 부분에서는 실수로 String을 전달하여 런타임 에러가 생길 수도 있습니다.
Since its methods accept or return an Object, you are free to pass in whatever you want, provided that it is not one of the primitive types. There is no way to verify, at compile time, how the class is used. One part of the code may place an Integer in the box and expect to get Integers out of it, while another part of the code may mistakenly pass in a String, resulting in a runtime error.
제네릭 버전의 Box클래스 A Generic Version of the Box Class
generic class 는 아래와 같은 포맷으로 정의됩니다:
A generic class is defined with the following format:
class name<T1, T2, ..., Tn> { /* ... */ }
클래스 이름의 뒤에 angle brackets (<>)로 구분된 타입 패러미터 섹션이 옵니다.
타입 패러미터 섹션은 type 패러미터(type 변수라고도 함) T1, T2, ..., Tn를 규정합니다.
제네릭을 사용하기 위하여 Box class를 업데이트하려면, "public class Box"를 "public class Box<T>"로 바꾸는 generic type 선언을 합니다.
이 선언은 클래스 안의 모든 곳에서 사용될 수 있는 type 변수인 T를 도입합니다.
이렇게 변경하면 Box class는 아래와 같이 됩니다:
The type parameter section, delimited by angle brackets (<>), follows the class name. It specifies the type parameters (also called type variables) T1, T2, ..., and Tn.
To update the Box class to use generics, you create a generic type declaration by changing the code "public class Box" to "public class Box<T>". This introduces the type variable, T, that can be used anywhere inside the class.
With this change, the Box class becomes:
/**
* Generic version of the Box class.
* @param <T> the type of the value being boxed
*/
public class Box<T> {
// T stands for "Type"
private T t;
public void set(T t) { this.t = t; }
public T get() { return t; }
}
As you can see, all occurrences of Object are replaced by T. A type variable can be any non-primitive type you specify: any class type, any interface type, any array type, or even another type variable.
This same technique can be applied to create generic interfaces.
Type Parameter Naming Conventions
By convention, type parameter names are single, uppercase letters. This stands in sharp contrast to the variable naming conventions that you already know about, and with good reason: Without this convention, it would be difficult to tell the difference between a type variable and an ordinary class or interface name.
The most commonly used type parameter names are:
- E - Element (used extensively by the Java Collections Framework)
- K - Key
- N - Number
- T - Type
- V - Value
- S,U,V etc. - 2nd, 3rd, 4th types
You'll see these names used throughout the Java SE API and the rest of this lesson.
Invoking and Instantiating a Generic Type
To reference the generic Box class from within your code, you must perform a generic type invocation, which replaces T with some concrete value, such as Integer:
Box<Integer> integerBox;
You can think of a generic type invocation as being similar to an ordinary method invocation, but instead of passing an argument to a method, you are passing a type argument — Integer in this case — to the Box class itself.
Type Parameter and Type Argument Terminology: Many developers use the terms "type parameter" and "type argument" interchangeably, but these terms are not the same. When coding, one provides type arguments in order to create a parameterized type. Therefore, the T in Foo<T> is a type parameter and the String in Foo<String> f is a type argument. This lesson observes this definition when using these terms.
Like any other variable declaration, this code does not actually create a new Box object. It simply declares that integerBox will hold a reference to a "Box of Integer", which is how Box<Integer> is read.
An invocation of a generic type is generally known as a parameterized type.
To instantiate this class, use the new keyword, as usual, but place <Integer> between the class name and the parenthesis:
Box<Integer> integerBox = new Box<Integer>();
The Diamond
In Java SE 7 and later, you can replace the type arguments required to invoke the constructor of a generic class with an empty set of type arguments (<>) as long as the compiler can determine, or infer, the type arguments from the context. This pair of angle brackets, <>, is informally called the diamond. For example, you can create an instance of Box<Integer> with the following statement:
Box<Integer> integerBox = new Box<>();
For more information on diamond notation and type inference, see Type Inference.
Multiple Type Parameters
As mentioned previously, a generic class can have multiple type parameters. For example, the generic OrderedPair class, which implements the generic Pair interface:
public interface Pair<K, V> { public K getKey(); public V getValue(); } public class OrderedPair<K, V> implements Pair<K, V> { private K key; private V value; public OrderedPair(K key, V value) { this.key = key; this.value = value; } public K getKey() { return key; } public V getValue() { return value; } }
The following statements create two instantiations of the OrderedPair class:
Pair<String, Integer> p1 = new OrderedPair<String, Integer>("Even", 8); Pair<String, String> p2 = new OrderedPair<String, String>("hello", "world");
The code, new OrderedPair<String, Integer>, instantiates K as a String and V as an Integer. Therefore, the parameter types of OrderedPair's constructor are String and Integer, respectively. Due to autoboxing, it is valid to pass a String and an int to the class.
As mentioned in The Diamond, because a Java compiler can infer the K and V types from the declaration OrderedPair<String, Integer>, these statements can be shortened using diamond notation:
OrderedPair<String, Integer> p1 = new OrderedPair<>("Even", 8); OrderedPair<String, String> p2 = new OrderedPair<>("hello", "world");
To create a generic interface, follow the same conventions as for creating a generic class.
Parameterized Types
You can also substitute a type parameter (i.e., K or V) with a parameterized type (i.e., List<String>). For example, using the OrderedPair<K, V> example:
OrderedPair<String, Box<Integer>> p = new OrderedPair<>("primes", new Box<Integer>(...));
'Java 배우기' 카테고리의 다른 글
Generic Methods (0) | 2016.01.24 |
---|---|
Raw Types (0) | 2016.01.24 |
Why Use Generics? 제네릭스를 사용하는 이유 (0) | 2016.01.24 |
Lesson: Generics (Updated) 단원: 제네릭스 (업데이트된) (0) | 2016.01.05 |
Questions and Exercises: Characters and Strings (0) | 2016.01.05 |
댓글