최근에 자바 @SafeVarargs
주석을 보았습니다 . Java의 가변 기능을 안전하지 않게 만드는 것에 대한 인터넷 검색으로 인해 혼란스러워졌습니다 (힙 중독? 지워진 유형?). 몇 가지 사항을 알고 싶습니다.
-
가변성 Java 기능이 안전하지 않은 이유는 무엇입니까
@SafeVarargs
(바람직한 예의 형태로 설명하는 것이 바람직 함)? -
왜이 주석이 프로그래머의 재량에 달려 있습니까? 컴파일러가 확인할 수있는 것이 아닌가?
-
그의 기능이 실제로 Varag를 안전하게 지키기 위해 준수해야 할 표준이 있습니까? 그렇지 않은 경우이를 보장하기위한 모범 사례는 무엇입니까?
답변
1) 인터넷과 StackOverflow에는 제네릭 및 varargs의 특정 문제에 대한 많은 예가 있습니다. 기본적으로 유형 매개 변수 유형의 가변 개수의 인수가있는 경우입니다.
<T> void foo(T... args);
Java에서 varargs는 컴파일 타임에 간단한 “다시 쓰기”를 수행하는 구문 설탕입니다. type의 varargs 매개 변수는 type X...
의 매개 변수로 변환됩니다 X[]
. 이 varargs 메소드를 호출 할 때마다 컴파일러는 varargs 매개 변수에 포함 된 모든 “변수 인수”를 수집하고 다음과 같은 배열을 작성합니다 new X[] { ...(arguments go here)... }
.
이것은 varargs 유형이 콘크리트와 같은 경우에 효과적 String...
입니다. 와 같은 유형 변수 인 T...
경우 T
해당 호출에 대한 구체적인 유형으로 알려진 경우에도 작동 합니다. 방법은 위의 클래스의 일부 예를 들어 경우 Foo<T>
, 당신은이 Foo<String>
참조, 다음 호출 foo
우리가 알고 있기 때문에 괜찮을 것에 T
인 String
코드에서 그 시점에서.
그러나의 “값” T
이 다른 유형 매개 변수 인 경우에는 작동하지 않습니다 . Java에서는 유형 매개 변수 구성 요소 유형 ( new T[] { ... }
) 의 배열을 작성할 수 없습니다 . 따라서 Java는 대신 new Object[] { ... }
(여기 Object
의 상한이 있습니다 T
. 상한이 다르면 대신 대신 사용됩니다 Object
) 컴파일러 경고를 제공합니다.
그렇다면 무엇을 만드는 new Object[]
대신에 new T[]
무엇이 잘못 되었나요 ? Java의 배열은 런타임에 구성 요소 유형을 알고 있습니다. 따라서 전달 된 배열 객체는 런타임에 잘못된 구성 요소 유형을 갖습니다.
varargs를 가장 일반적으로 사용하는 경우 요소를 반복하기 만하면 문제가되지 않으므로 (배열의 런타임 유형에 신경 쓰지 않아도 됨) 안전합니다.
@SafeVarargs
final <T> void foo(T... args) {
for (T x : args) {
// do stuff with x
}
}
그러나 전달 된 배열의 런타임 구성 요소 유형에 의존하는 것은 안전하지 않습니다. 안전하지 않고 충돌하는 간단한 예는 다음과 같습니다.
class UnSafeVarargs
{
static <T> T[] asArray(T... args) {
return args;
}
static <T> T[] arrayOfTwo(T a, T b) {
return asArray(a, b);
}
public static void main(String[] args) {
String[] bar = arrayOfTwo("hi", "mom");
}
}
문제는 여기에 우리의 유형에 따라 달라집니다 것입니다 args
될 수 T[]
로 복귀하기 위해 T[]
. 그러나 실제로 런타임에 인수의 유형은의 인스턴스가 아닙니다 T[]
.
3) 메소드에 type 인수가있는 경우 T...
(여기서 T는 모든 유형 매개 변수 임)
- 안전 : 메소드가 배열의 요소가 인스턴스라는 사실에만 의존하는 경우
T
- 안전하지 않은 경우 : 어레이가 인스턴스라는 사실에 의존하는 경우
T[]
어레이의 런타임 타입에 의존하는 상황이 포함 형으로 리턴 T[]
형의 매개 인자로 전달 T[]
이용 어레이 형 점점 .getClass()
처럼 배열의 실행 종류에 따라 메소드에 전달 List.toArray()
하고 Arrays.copyOf()
등
2) 위에서 언급 한 구별이 너무 복잡하여 자동으로 쉽게 구별 할 수 없습니다.
답변
모범 사례를 위해 이것을 고려하십시오.
이것이 있다면 :
public <T> void doSomething(A a, B b, T... manyTs) {
// Your code here
}
이것을 다음과 같이 변경하십시오.
public <T> void doSomething(A a, B b, T... manyTs) {
doSomething(a, b, Arrays.asList(manyTs));
}
private <T> void doSomething(A a, B b, List<T> manyTs) {
// Your code here
}
나는 일반적으로 발신자에게 더 편리하게하기 위해 varargs 만 추가한다는 것을 알았습니다. 내부 구현에서을 사용하는 것이 거의 항상 더 편리합니다 List<>
. 따라서 피기 백 (piggy-back) Arrays.asList()
하고 힙 오염 (Hap Pollution)을 도입 할 수있는 방법이 없도록하기 위해 이것이 바로 제가하는 일입니다.
나는 이것이 당신의 # 3에만 대답한다는 것을 알고 있습니다. newacct는 위의 # 1 및 # 2에 대한 훌륭한 답변을 주었으며, 이것을 의견으로 남길만한 명성은 충분하지 않습니다. :피