Java Generics - 유형 삭제
1. 개요
List
2. 컴파일러는 제네릭을 어떻게 처리하나요?
일반적으로 컴파일러는 두 가지 방법으로 제네릭을 처리합니다.
1.코드 전문화. 제네릭 클래스나 제네릭 메서드를 인스턴스화할 때 새로운 개체 코드(바이트코드 또는 바이너리 코드)가 생성됩니다. 예를 들어 일반 목록의 경우 문자열, 정수 및 부동 소수점에 대한 세 가지 대상 코드를 생성해야 할 수 있습니다.
2.코드 공유. 각 일반 클래스에 대해 대상 코드의 고유한 복사본만 생성됩니다. 일반 클래스의 모든 인스턴스는 이 대상 코드에 매핑되고 필요한 경우 유형 검사 및 유형 변환이 수행됩니다.
C++의 템플릿은 일반적인 코드 전문화 구현입니다. C++ 컴파일러는 각 일반 클래스 인스턴스에 대한 실행 코드를 생성합니다. 정수 목록과 문자열 목록은 실행 코드에서 서로 다른 두 가지 유형입니다. 이로 인해 코드가 팽창하게 되지만 숙련된 C++ 프로그래머는 능숙하게 코드 팽창을 피할 수 있습니다.
코드 전문화의 또 다른 단점은 참조 유형 컬렉션의 요소가 본질적으로 포인터이기 때문에 참조 유형 시스템의 공간 낭비입니다. 각 유형에 대한 실행 코드를 생성할 필요는 없습니다. 이는 Java 컴파일러가 코드 공유를 사용하여 제네릭을 처리하는 주된 이유이기도 합니다.
Java 컴파일러는 코드 공유를 통해 각 일반 유형에 대한 고유한 바이트코드 표현을 생성하고 일반 유형의 인스턴스를 이 고유한 바이트코드 표현에 매핑합니다. 여러 일반 유형 인스턴스를 고유한 바이트코드 표현으로 매핑하는 것은 유형 삭제를 통해 이루어집니다.
3. 유형 삭제란 무엇인가요?
유형 삭제는 유형 매개변수 병합을 통해 일반 유형 인스턴스를 동일한 바이트코드에 연결하는 것을 의미합니다. 컴파일러는 일반 유형에 대해 하나의 바이트코드만 생성하고 해당 인스턴스를 이 바이트코드와 연결합니다. 유형 삭제의 핵심은 제네릭 유형에서 유형 매개변수에 대한 정보를 지우고 필요한 경우 유형 검사 및 유형 변환 방법을 추가하는 것입니다.
유형 삭제는 단순히 일반 Java 코드를 일반 Java 코드로 변환하는 것으로 이해될 수 있지만, 컴파일러는 일반 Java 코드를 일반 Java 바이트코드로 직접 변환하는 보다 직접적입니다.
유형 삭제의 주요 프로세스는 다음과 같습니다.
1. 모든 일반 매개변수를 가장 왼쪽 경계(최상위 상위 유형) 유형으로 바꿉니다.
2. 모든 유형 매개변수를 제거합니다.
예를 들어, 유형 삭제 후
interface Comparable <A> { public int compareTo( A that); } final class NumericValue implements Comparable <NumericValue> { priva te byte value; public NumericValue (byte value) { this.value = value; } public byte getValue() { return value; } public int compareTo( NumericValue t hat) { return this.value - that.value; } } ----------------- class Collections { public static <A extends Comparable<A>>A max(Collection <A> xs) { Iterator <A> xi = xs.iterator(); A w = xi.next(); while (xi.hasNext()) { A x = xi.next(); if (w.compareTo(x) < 0) w = x; } return w; } } final class Test { public static void main (String[ ] args) { LinkedList <NumericValue> numberList = new LinkedList <NumericValue> (); numberList .add(new NumericValue((byte)0)); numberList .add(new NumericValue((byte)1)); NumericValue y = Collections.max( numberList ); } }<span style="color: #333333; font-family: Arial; font-size: 14px;"> </span>
의 유형은
인터페이스 Comparable {
public int compareTo( Object that); } final class NumericValue implements Comparable { priva te byte value; public NumericValue (byte value) { this.value = value; } public byte getValue() { return value; } public int compareTo( NumericValue t hat) { return this.value - that.value; } public int compareTo(Object that) { return this.compareTo((NumericValue)that); } } ------------- class Collections { public static Comparable max(Collection xs) { Iterator xi = xs.iterator(); Comparable w = (Comparable) xi.next(); while (xi.hasNext()) { Comparable x = (Comparable) xi.next(); if (w.compareTo(x) < 0) w = x; } return w; } } final class Test { public static void main (String[ ] args) { LinkedList numberList = new LinkedList(); numberList .add(new NumericValue((byte)0)); , numberList .add(new NumericValue((byte)1)); NumericValue y = (NumericValue) Collections.max( numberList ); } }<span style="color: #333333; font-family: Arial; font-size: 14px;"> </span>
삭제 A 후 첫 번째 일반 클래스 Comparable 가장 왼쪽 경계 객체로 대체되었습니다. Comparable
두 번째 예는 유형 매개변수 >A를 확장합니다. A는 Comparable의 하위 클래스여야 합니다. 모든 유형 매개변수가 먼저 논의됩니다. ti를 가장 왼쪽 경계인 Comparable로 대체한 다음 매개변수 유형 A를 제거하여 최종 삭제된 결과를 얻습니다.
4. 유형 삭제로 인한 문제
유형 삭제가 많은 일반적인 초자연적 문제로 직접적으로 이어지는 것은 바로 유형 삭제의 존재 때문입니다.
Q1. 동일한 일반 클래스의 인스턴스를 사용하여 메서드 시그니처를 구별하나요? --아니요!
import java.util.*;
public class Erasure{ public void test(List<String> ls){ System.out.println("Sting"); } public void test(List<Integer> li){ System.out.println("Integer"); } }<span style="color: #333333; font-family: Arial; font-size: 14px;"> </span>
이 클래스를 컴파일하세요.
参数类型明明不一样啊,一个List
Q2. 同时catch同一个泛型异常类的多个实例?——NO!
同理,如果定义了一个泛型一场类GenericException
Q3.泛型类的静态变量是共享的?——Yes!
猜猜这段代码的输出是什么?
import java.util.*; public class StaticTest{ public static void main(String[] args){ GT<Integer> gti = new GT<Integer>(); gti.var=1; GT<String> gts = new GT<String>(); gts.var=2; System.out.println(gti.var); } } class GT<T>{ public static int var=0; public void nothing(T x){} }<span style="color: #333333; font-family: Arial; font-size: 14px;"> </span>
答案是——2!由于经过类型擦除,所有的泛型类实例都关联到同一份字节码上,泛型类的所有静态变量是共享的。
五、Just remember
1.虚拟机中没有泛型,只有普通类和普通方法
2.所有泛型类的类型参数在编译时都会被擦除
3.创建泛型对象时请指明类型,让编译器尽早的做参数检查(Effective Java,第23条:请不要在新代码中使用原生态类型)
4.不要忽略编译器的警告信息,那意味着潜在的ClassCastException等着你。
更多Java泛型-类型擦除相关文章请关注PHP中文网!

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











Go의 일반 함수는 가변 유형의 문제를 해결합니다. 일반 함수를 사용하면 런타임에 유형 매개변수를 지정할 수 있습니다. 이를 통해 다양한 유형의 인수를 처리할 수 있는 함수를 작성할 수 있습니다. 예를 들어 Max 함수는 두 개의 비교 가능한 매개변수를 허용하고 더 큰 값을 반환하는 일반 함수입니다. 일반 함수를 사용하면 다양한 유형의 매개변수를 처리할 수 있는 보다 유연하고 일반적인 코드를 작성할 수 있습니다.

Go의 제네릭 적용 시나리오: 컬렉션 작업: 필터링과 같은 모든 유형에 적합한 컬렉션 작업을 만듭니다. 데이터 구조: 큐, 스택, 맵과 같은 범용 데이터 구조를 작성하여 다양한 유형의 데이터를 저장하고 조작합니다. 알고리즘: 다양한 유형의 데이터를 처리할 수 있는 정렬, 검색, 축소 등의 범용 알고리즘을 작성합니다.

Java 함수 제네릭을 사용하면 상한 및 하한을 설정할 수 있습니다. 확장은 함수에서 허용하거나 반환하는 데이터 유형이 지정된 유형의 하위 유형이어야 함을 지정합니다. 하한(슈퍼)은 함수에서 허용하거나 반환하는 데이터 유형이 지정된 유형의 슈퍼 유형이어야 함을 지정합니다. 제네릭을 사용하면 코드 재사용성과 보안이 향상됩니다.

답변: Golang 제네릭은 코드 재사용성, 유연성, 유형 안전성 및 확장성을 개선하기 위한 강력한 도구입니다. 자세한 설명: 장점: 코드 재사용성: 공통 알고리즘 및 데이터 구조 유연성: 특정 유형의 인스턴스의 런타임 생성 유형 안전성: 컴파일 시간 유형 확인 확장성: 손쉬운 확장 및 사용자 정의 목적: 공통 기능: 정렬, 비교 목록과 같은 공통 데이터 구조 , 맵, 스택 등 유형 별칭: 유형 선언 단순화 제한된 제네릭: 유형 안전성 보장

Android 개발에 제네릭을 적용하면 코드 재사용성, 보안 및 유연성이 향상됩니다. 구문은 유형 매개변수화된 데이터를 조작하는 데 사용할 수 있는 유형 변수 T를 선언하는 것으로 구성됩니다. 작동 중인 일반 항목에는 사용자 정의 데이터 어댑터가 포함되어 있어 어댑터가 모든 유형의 사용자 정의 데이터 개체에 적응할 수 있습니다. Android는 또한 다양한 유형의 매개변수를 조작할 수 있는 일반 목록 클래스(예: ArrayList)와 일반 메서드를 제공합니다. 제네릭 사용의 이점에는 코드 재사용성, 보안 및 유연성이 포함되지만, 코드 가독성을 보장하기 위해 올바른 경계를 지정하고 이를 적당히 사용하도록 주의를 기울여야 합니다.

Go 일반 함수의 제한 사항: 유형 매개변수만 지원되고 값 매개변수는 지원되지 않습니다. 함수 재귀는 지원되지 않습니다. 유형 매개변수는 명시적으로 지정할 수 없으며 컴파일러에 의해 유추됩니다.

Go 함수 서명 및 매개변수에 대한 제네릭의 영향은 다음과 같습니다. 유형 매개변수: 함수 서명에는 함수가 사용할 수 있는 유형을 지정하는 유형 매개변수가 포함될 수 있습니다. 유형 제약 조건: 유형 매개 변수에는 충족해야 하는 조건을 지정하는 제약 조건이 있을 수 있습니다. 매개변수 유형 유추: 컴파일러는 지정되지 않은 유형 매개변수의 유형을 유추할 수 있습니다. 유형 지정: 일반 함수를 호출하기 위해 매개변수 유형을 명시적으로 지정할 수 있습니다. 이를 통해 코드 재사용성과 유연성이 향상되어 여러 유형과 함께 사용할 수 있는 함수 및 유형을 작성할 수 있습니다.

Java에서 열거형 유형과 제네릭의 조합: 제네릭으로 열거형을 선언할 때 꺾쇠 괄호를 추가해야 하며 T는 유형 매개변수입니다. 일반 클래스를 생성할 때 꺾쇠 괄호도 추가해야 합니다. T는 모든 유형을 저장할 수 있는 유형 매개변수입니다. 이 조합은 코드 유연성, 유형 안전성을 향상하고 코드를 단순화합니다.
