>Java >java지도 시간 >Java의 제네릭 자세히 살펴보기

Java의 제네릭 자세히 살펴보기

零下一度
零下一度원래의
2017-07-18 09:47:091536검색

​​​​​​ 제네릭 정의: 제네릭은 JDK 1.5의 새로운 기능입니다. 그 본질은 매개변수화된 유형(Parameterized Type)을 적용하는 것입니다. 즉, 작동되는 데이터 유형은 다음과 같이 지정됩니다. 매개변수를 사용하는 경우 특정 유형을 지정합니다. 이 매개변수 유형은 각각 일반 클래스, 일반 인터페이스 및 일반 메소드라고 불리는 클래스, 인터페이스 및 메소드 생성에 사용될 수 있습니다.

제네릭의 개념은 일찍이 C++ 언어의 템플릿에서 뿌리를 내리기 시작했습니다. Java 언어가 제네릭이 아직 등장하지 않은 버전에 있었을 때 유일한 두 가지 메소드는 Object의 상위 클래스였습니다. 모든 유형 및 유형 캐스팅 기능을 결합하여 유형 일반화를 달성합니다. 예를 들어, JDK 1.5 이전에는 해시 테이블에 접근할 때 HashMap의 get() 메소드를 사용했고, 반환 값은 Object 객체였습니다. Java 언어의 모든 유형은 java.lang.Object를 상속하기 때문입니다. 청두는 어떤 물체로도 변신이 가능합니다. 그러나 무한한 가능성이 있기 때문에 프로그래머와 런타임 가상 머신만이 이 객체가 어떤 유형의 객체인지 알 수 있습니다. 컴파일하는 동안 컴파일러는 이 개체의 강제 변환이 성공했는지 여부를 확인할 수 없습니다. 프로그래머가 이 작업의 정확성을 보장하는 데에만 의존하는 경우 ClassCastException의 많은 위험이 프로그램 런타임으로 이전됩니다. C#과 Java에서 제네릭 기술을 사용하는 것은 동일해 보이지만 프로그램 소스 코드, 중간 언어, 중간 언어, 중간 언어에서 C#의 제네릭은 구현에 근본적인 차이가 있습니다. , 중간 언어에서는 제네릭이 자리 표시자이거나 실제로 런타임에 CLR에 존재합니다. List와 List은 시스템 런타임 중에 생성되며 다음과 같은 두 가지 유형을 갖습니다. 자신의 가상 메서드 테이블과 형식 데이터를 사용하는 경우 이러한 구현을 형식 인플레이션이라고 하며 이 메서드를 기반으로 구현된 제네릭을 실제 제네릭이라고 합니다.

Java 언어의 제네릭은 프로그램 소스 코드에만 존재합니다. 컴파일된 바이트코드 파일에서는 원래 원시 유형(Raw Type, Naked 유형이라고도 함)으로 대체되었습니다. 따라서 런타임 시 Java 언어의 경우 ArrayList와 ArrayList은 동일한 클래스입니다. 따라서 일반 기술은 실제로 Java 언어의 구문 설탕입니다. Java 언어의 일반 구현 방법을 유형 삭제라고 하며 이 방법을 기반으로 구현된 제네릭을 의사 제네릭이라고 합니다. (유형 삭제에 대해서는 나중에 연구하겠습니다.)


제네릭 메커니즘을 사용하여 작성된 프로그램 코드는 Object 변수를 지저분하게 사용한 다음 강제 유형 변환을 수행하는 코드보다 더 안전하고 읽기 쉽습니다. 제네릭은 컬렉션 클래스에 특히 유용합니다.

1. 제네릭을 사용하는 이유

여기서 코드 조각을 살펴보겠습니다.

List list = new ArrayList();  
list.add("CSDN_SEU_Cavin");  
list.add(100);  
for (int i = 0; i < list.size(); i++) {  
  String name = (String) list.get(i); //取出Integer时,运行时出现异常  
System.out.println("name:" + name);  
}

이 예에서는 목록 유형 컬렉션에 문자열 유형 값과 정수 유형 값을 추가합니다. (목록의 기본 유형이 객체이기 때문에 이것은 합법적입니다.) 후속 루프에서는 이전에 또는 다른 이유로 목록에 Integer 유형 값을 추가하는 것을 잊어버렸기 때문에 런타임 시 java.lang.ClassCastException 예외가 발생합니다. 이 문제를 해결하기 위해 제네릭이 등장했습니다.

2. 제네릭 사용 제네릭을 사용하면 프로그래머는 일반적으로 컬렉션에 사용되는 유형 추상화를 사용할 수 있습니다.

List<String> list = new ArrayList<String>();

  이렇게 작성하면 위의 루프 값 획득 방법은 오류를 보고하지 않으며 유형 변환을 수행할 필요가 없습니다. List을 통해 목록 컬렉션이 문자열 유형의 요소만 포함할 수 있도록 직접적으로 제한됩니다.

3. 제네릭은 컴파일 중에만 유효합니다. 제네릭을 사용할 때는 제네릭이 컴파일될 때 어떤 일이 발생하는지 이해해야 하므로 여기서는 다음 사항에 특별한 주의를 기울여야 합니다. 제네릭 유형은 다음과 같습니다. 코드가 클래스 파일로 컴파일될 때 유효합니다

AyyayList<String> a = new ArrayList<String>();  
ArrayList b = new ArrayList();  
Class c1 = a.getClass();  
Class c2 = b.getClass();  
System.out.println(a == b);

 위 프로그램의 출력 결과는 true입니다. 이는 모든 반영 작업이 런타임에 수행되기 때문입니다. 이는 컴파일 후에 프로그램이 비제네릭화 조치를 취한다는 것을 증명합니다.

 즉, Java의 제네릭은 컴파일 단계에서만 유효합니다. 컴파일 과정에서 제네릭 결과가 올바르게 검증된 후 제네릭의 관련 정보가 지워지고 객체의 진입 및 종료 메소드 경계에 유형 검사 및 유형 변환 메소드가 추가됩니다. 즉,

성공적으로 컴파일된 클래스 파일에는 일반적인 정보가 포함되어 있지 않습니다. 일반 정보는 런타임 단계

에 포함되지 않습니다. 다음 코드는 제네릭이 컴파일 중에만 유효하다는 것을 Java의 리플렉션 메커니즘을 통해 잘 설명합니다

ArrayList<String> a = new ArrayList<String>();  
a.add("CSDN_SEU_Cavin");  
Class c = a.getClass();  
try{  
    Method method = c.getMethod("add",Object.class);  
    method.invoke(a,100);  
    System.out.println(a);  //[CSDN_SEU_Cavin, }catch(Exception e){  
    e.printStackTrace();

 

4 제네릭 클래스와 제네릭 메서드

public static class FX<T> {  
    private T ob; // 定义泛型成员变量  
  
    public FX(T ob) {  
        this.ob = ob;  
    }  
  
    public T getOb() {  
        return ob;  
    }  
  
    public void showTyep() {  
        System.out.println("T的实际类型是: " + ob.getClass().getName());  
    }  
}  
    public static void main(String[] args) {  
        FX<Integer> intOb = new FX<Integer>(100);  
        intOb.showTyep();  
        System.out.println("value= " + intOb.getOb());  //java.lang.Integer  System.out.println("----------------------------------");  
  
        FX<String> strOb = new FX<String>("CSDN_SEU_Calvin");  
        strOb.showTyep();  
        System.out.println("value= " + strOb.getOb());  //value= 100 } 
  

 5. 제네릭의 장점

    (1) 타입 안전성.

  제네릭을 사용하여 정의된 변수의 유형 제한을 알면 컴파일러는 Java 프로그램의 유형 안전성을 보다 효과적으로 향상시킬 수 있습니다.

   (2) 강제형 변환을 제거합니다.

    소스 코드에서 많은 캐스트를 제거합니다. 이렇게 하면 코드를 더 쉽게 읽을 수 있고 오류 가능성이 줄어듭니다. 모든 캐스트는 자동이며 암시적입니다.

   (3) 성능 향상

위 내용은 Java의 제네릭 자세히 살펴보기의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.