목차
실제 매개변수를 배열로 래핑
실제 매개변수의 가변 개수를 정의하는 방법
실제 매개변수의 개수가 가변적인 메소드 호출
4. 处理个数可变的实参
5. 转发个数可变的实参
6. 是数组?不是数组?
7. 当个数可变的实参遇到泛型
8. 重载中的选择问题
9. 归纳总结
Java java지도 시간 Java의 Varargs 메커니즘 이해

Java의 Varargs 메커니즘 이해

Feb 23, 2017 am 10:23 AM
java varargs

J2SE 1.5는 "Varargs" 메커니즘을 제공합니다. 이 메커니즘을 사용하면 여러 실제 매개변수와 일치할 수 있는 형식 매개변수를 정의할 수 있습니다. 따라서 다양한 수의 실제 매개변수를 더 간단한 방법으로 전달할 수 있습니다. 이 문서에서는 이 메커니즘을 사용하는 방법과 이 메커니즘이 배열, 제네릭 및 오버로드와 상호 작용할 때 발생하는 몇 가지 문제를 소개합니다.

J2SE 1.4까지는 Java 프로그램에서 가변 개수의 실제 매개변수로 메소드를 정의하는 것이 불가능했습니다. 왜냐하면 Java에서는 실제 매개변수(Arguments)와 형식 매개변수(Parameters)의 수와 유형을 요구하기 때문입니다. )은 하나씩 일치해야 하며 형식 매개변수의 수는 메소드 정의 시 고정됩니다. 오버로딩 메커니즘을 사용하여 동일한 메소드에 대해 다양한 수의 형식 매개변수가 있는 버전을 제공할 수 있지만, 이는 여전히 실제 매개변수 양을 임의로 변경하도록 허용하는 목적을 달성할 수 없습니다.

그러나 일부 메서드의 의미론에서는 가변 개수의 실제 매개변수를 허용할 수 있어야 합니다. 예를 들어 유명한 기본 메서드는 모든 명령줄 매개변수를 실제 매개변수로 허용할 수 있어야 하며 명령줄 매개변수 개수는 미리 결정할 수 없습니다.

이 문제에 대해서는 전통적으로 "배열을 사용하여 전달할 실제 매개변수를 래핑하는" 접근 방식을 사용하여 문제를 해결합니다.

실제 매개변수를 배열로 래핑

"실제 매개변수를 배열로 래핑"하는 방법은 세 단계로 나눌 수 있습니다. 먼저 이 방법에 대한 배열 매개변수를 정의합니다. 그런 다음 호출 시 전달할 실제 매개변수를 모두 포함하는 배열을 생성하고, 마지막으로 이 배열을 실제 매개변수로 전달합니다.

이 접근 방식은 "메서드가 다양한 수의 매개 변수를 허용하도록 허용"한다는 목적을 효과적으로 달성할 수 있지만 호출 형식은 충분히 간단하지 않습니다.

J2SE 1.5는 Varargs 메커니즘을 제공하므로 여러 실제 매개변수와 일치할 수 있는 형식 매개변수를 직접 정의할 수 있습니다. 따라서 다양한 수의 실제 매개변수를 더 간단한 방법으로 전달할 수 있습니다.

Varargs의 의미

일반적으로 "Varargs"는 "변수의 수"를 의미합니다. 단순히 "가변인수"라고도 부르는데, 이 용어는 가변이 무엇인지 설명하지 않기 때문에 의미가 조금 모호합니다.

실제 매개변수의 가변 개수를 정의하는 방법

형식 매개변수의 "유형"과 "매개변수 이름" 사이에 ""를 연속으로 3개만 추가하면 됩니다. (즉, 영어 문장의 줄임표 "...") 불확실한 개수의 실제 매개변수와 일치시킬 수 있습니다. 이러한 형식 매개변수를 사용하는 방법은 가변 개수의 실제 매개변수를 사용하는 방법입니다.

목록 1: 가변 개수의 실제 매개변수가 있는 메소드

private static int sumUp(int... values) {    
}
로그인 후 복사


마지막 형식 매개변수만 정의할 수 있습니다. "불확실한 수의 실제 매개변수를 일치시킬 수 있게" 됩니다. 따라서 메소드에는 그러한 매개변수가 하나만 있을 수 있습니다. 또한, 이 메소드에 다른 형식 매개변수가 있는 경우 이를 맨 앞에 배치하십시오.

컴파일러는 비밀리에 이 마지막 형식 매개변수를 배열 매개변수로 변환하고, 컴파일된 클래스 파일에 표시를 넣어 이것이 실제 매개변수의 개수가 가변적인 메소드임을 나타냅니다.

목록 2: 가변 개수의 실제 매개변수를 갖는 메소드의 비밀 형식

private static int sumUp(int[] values) {    
}
로그인 후 복사


이러한 변환으로 인해 다음을 수행할 수 있습니다. 더 이상 변환된 메서드와 동일한 시그니처를 사용하여 이 클래스에 대한 메서드를 정의하지 않습니다.

목록 3: 컴파일 오류를 일으키는 조합

private static int sumUp(int... values) {   
}   

private static int sumUp(int[] values) {   
}
로그인 후 복사


빈 생존 문제

J2SE 1.5 구문에 따름 , "..." 앞의 공백 문자는 선택 사항입니다. 이런 식으로 작성하는 방법에는 두 가지가 있습니다. "..." 앞에 공백 문자를 추가하는 것("Object... args"로 형성됨)과 "..." 앞에 공백 문자를 추가하지 않는 것("로 형성됨)입니다. 개체...args"). 현재 J2SE 1.5와 호환되는 Java Code Convention은 공식적으로 발표되지 않았기 때문에 어떤 작성 방식이 더 정통적인지 알 수 없습니다. 그러나 배열 매개 변수는 "Object [] args"와 "Object[] args"의 두 가지 방법으로 작성할 수도 있고 "[]" 앞에 공백 문자를 추가하지 않는 전통적인 작성 방식을 사용하는 것으로 보입니다. 공백이 없는 "Object...args" "작성 방식이 전반적으로 더 조화로워졌습니다.

실제 매개변수의 개수가 가변적인 메소드 호출

전달하려는 실제 매개변수가 해당 위치에 하나씩 쓰여지면 가변적인 수의 실제 매개변수를 갖는 메소드 가변 메소드. 다른 단계는 필요하지 않습니다.

목록 4: 여러 가지 실제 매개변수를 전달할 수 있음

sumUp(1, 3, 5, 7);
로그인 후 복사


컴파일러는 비밀리에 이 호출 프로세스를 다음 형식으로 변환합니다. "실제 매개변수를 배열로 래핑":

목록 5: 비밀리에 나타나는 배열 생성

sumUp(new int[]{1, 2, 3, 4});
로그인 후 복사


또한 여기에 " "불확실한 숫자"에도 0이 포함되므로 이러한 호출은 합리적입니다.

목록 6: 실제 매개변수 0개를 전달할 수도 있습니다

sumUp(); 

这种调用方法被编译器秘密转化之后的效果,则等同于这样:
로그인 후 복사


목록 7: 빈 배열에 해당하는 0개의 인수

sumUp(new int[]{});
로그인 후 복사


이번에는 null이 아닌 빈 배열이 전달됩니다. 이렇게 하면 어떤 상황에 속하는지 파악하지 않고도 통일된 형태로 처리할 수 있습니다.

4. 处理个数可变的实参

处理个数可变的实参的办法,和处理数组实参的办法基本相同。所有的实参,都被保存到一个和形参同名的数组里。根据实际的需要,把这个数组里的元素读出之后,要蒸要煮,就可以随意了。

清单8:处理收到的实参们

private static int sumUp(int... values) {   
    int sum = 0;   
    for (int i = 0; i < values.length; i++) {   
        sum += values[i];   
    }   
    return sum;   
}
로그인 후 복사


5. 转发个数可变的实参

有时候,在接受了一组个数可变的实参之后,还要把它们传递给另一个实参个数可变的方法。因为编码时无法知道接受来的这一组实参的数目,所以“把它们 逐一写到该出现的位置上去”的做法并不可行。不过,这并不意味着这是个不可完成的任务,因为还有另外一种办法,可以用来调用实参个数可变的方法。

在J2SE 1.5的编译器的眼中,实参个数可变的方法是最后带了一个数组形参的方法的特例。因此,事先把整组要传递的实参放到一个数组里,然后把这个数组作为最后一个实参,传递给一个实参个数可变的方法,不会造成任何错误。借助这一特性,就可以顺利的完成转发了。

清单9:转发收到的实参们

public class PrintfSample {   

    public static void main(String[] args) {   
        // 打印出“Pi:3.141593 E:2.718282”   
        printOut("Pi:%f E:%f/n", Math.PI, Math.E);   
    }   

    private static void printOut(String format, Object... args) {   
        // J2SE 1.5里PrintStream新增的printf(String format, Object... args)方法   
        System.out.printf(format, args);   
    }   
}
로그인 후 복사


Java里的“printf”和“sprintf”

C语言里的printf(按一定的格式输出字符串)和sprintf(按一定的格式组合字符串)是十分经典的使用Varargs机制的例子。在 J2SE 1.5中,也分别在java.io.PrintStream类和java.lang.String类中提供了类似的功能。

按一定的格式输出字符串的功能,可以通过调用PrintStream对象的printf(String format, Object… args)方法来实现。

按一定的格式组合字符串的工作,则可以通过调用String类的String format(String format, Object… args)静态方法来进行。

6. 是数组?不是数组?

尽管在背地里,编译器会把能匹配不确定个实参的形参,转化为数组形参;而且也可以用数组包了实参,再传递给实参个数可变的方法;但是,这并不表示“能匹配不确定个实参的形参”和“数组形参”完全没有差异。

一个明显的差异是,如果按照调用实参个数可变的方法的形式,来调用一个最后一个形参是数组形参的方法,只会导致一个“cannot be applied to”的编译错误。

清单10:一个“cannot be applied to”的编译错误

private static void testOverloading(int[] i) {   
    System.out.println("A");   
}   

public static void main(String[] args) {   
    testOverloading(1, 2, 3);// 编译出错   
}
로그인 후 복사


由于这一原因,不能在调用只支持用数组包裹实参的方法的时候(例如在不是专门为J2SE 1.5设计第三方类库中遗留的那些),直接采用这种简明的调用方式。

如果不能修改原来的类,为要调用的方法增加参数个数可变的版本,而又想采用这种简明的调用方式,那么可以借助“引入外加函数(Introduce Foreign Method)”和“引入本地扩展(Intoduce Local Extension)”的重构手法来近似的达到目的。

7. 当个数可变的实参遇到泛型

J2SE 1.5中新增了“泛型”的机制,可以在一定条件下把一个类型参数化。例如,可以在编写一个类的时候,把一个方法的形参的类型用一个标识符(如T)来代表, 至于这个标识符到底表示什么类型,则在生成这个类的实例的时候再行指定。这一机制可以用来提供更充分的代码重用和更严格的编译时类型检查。

不过泛型机制却不能和个数可变的形参配合使用。如果把一个能和不确定个实参相匹配的形参的类型,用一个标识符来代表,那么编译器会给出一个“generic array creation”的错误。

清单11:当Varargs遇上泛型

private static <T> void testVarargs(T... args) {   
    // 编译出错   
}
로그인 후 복사


造成这个现象的原因在于J2SE 1.5中的泛型机制的一个内在约束——不能拿用标识符来代表的类型来创建这一类型的实例。在出现支持没有了这个约束的Java版本之前,对于这个问题,基本没有太好的解决办法。

不过,传统的“用数组包裹”的做法,并不受这个约束的限制。

清单12:可以编译的变通做法

private static <T> void testVarargs(T[] args) {   
    for (int i = 0; i < args.length; i++) {   
        System.out.println(args[i]);   
    }   
}
로그인 후 복사


8. 重载中的选择问题

Java支持“重载”的机制,允许在同一个类拥有许多只有形参列表不同的方法。然后,由编译器根据调用时的实参来选择到底要执行哪一个方法。

传统上的选择,基本是依照“特殊者优先”的原则来进行。一个方法的特殊程度,取决于为了让它顺利运行而需要满足的条件的数目,需要条件越多的越特殊。

在引入Varargs机制之后,这一原则仍然适用,只是要考虑的问题丰富了一些——传统上,一个重载方法的各个版本之中,只有形参数量与实参数量正 好一致的那些有被进一步考虑的资格。但是Varargs机制引入之后,完全可以出现两个版本都能匹配,在其它方面也别无二致,只是一个实参个数固定,而一 个实参个数可变的情况。

遇到这种情况时,所用的判定规则是“实参个数固定的版本优先于实参个数可变的版本”。

清单13:实参个数固定的版本优先

public class OverloadingSampleA {   

    public static void main(String[] args) {   
        testOverloading(1);// 打印出A   
        testOverloading(1, 2);// 打印出B   
        testOverloading(1, 2, 3);// 打印出C   
    }   

    private static void testOverloading(int i) {   
        System.out.println("A");   
    }   

    private static void testOverloading(int i, int j) {   
        System.out.println("B");   
    }   

    private static void testOverloading(int i, int... more) {   
        System.out.println("C");   
    }   
}
로그인 후 복사


如果在编译器看来,同时有多个方法具有相同的优先权,它就会陷入无法就到底调用哪个方法作出一个选择的状态。在这样的时候,它就会产生一个 “reference to 被调用的方法名 is ambiguous”的编译错误,并耐心的等候作了一些修改,足以免除它的迷惑的新源代码的到来。

在引入了Varargs机制之后,这种可能导致迷惑的情况,又增加了一些。例如现在可能会有两个版本都能匹配,在其它方面也如出一辙,而且都是实参个数可变的冲突发生。

清单14:左右都不是,为难了编译器

public class OverloadingSampleB {   

    public static void main(String[] args) {   
        testOverloading(1, 2, 3);// 编译出错   
    }   

    private static void testOverloading(Object... args) {   
    }   

    private static void testOverloading(Object o, Object... args) {   
    }   
}
로그인 후 복사


另外,因为J2SE 1.5中有“Autoboxing/Auto-Unboxing”机制的存在,所以还可能发生两个版本都能匹配,而且都是实参个数可变,其它方面也一模一样,只是一个能接受的实参是基本类型,而另一个能接受的实参是包裹类的冲突发生。

清单15:Autoboxing/Auto-Unboxing带来的新问题

public class OverloadingSampleC {   

    public static void main(String[] args) { /* 编译出错 */  
        testOverloading(1, 2); /* 还是编译出错 */  
        testOverloading(new Integer(1), new Integer(2));   
    }   

    private static void testOverloading(int... args) {   
    }   

    private static void testOverloading(Integer... args) {   
    }   
}
로그인 후 복사


9. 归纳总结

和“用数组包裹”的做法相比,真正的实参个数可变的方法,在调用时传递参数的操作更为简单,含义也更为清楚。不过,这一机制也有它自身的局限,并不是一个完美无缺的解决方案。

 以上就是Java 中 Varargs 机制的理解 的内容,更多相关内容请关注PHP中文网(www.php.cn)!

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

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

<gum> : Bubble Gum Simulator Infinity- 로얄 키를 얻고 사용하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
Nordhold : Fusion System, 설명
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora : 마녀 트리의 속삭임 - Grappling Hook 잠금 해제 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Feb 07, 2025 pm 12:09 PM

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

PHP : 웹 개발의 핵심 언어 PHP : 웹 개발의 핵심 언어 Apr 13, 2025 am 12:08 AM

PHP는 서버 측에서 널리 사용되는 스크립팅 언어이며 특히 웹 개발에 적합합니다. 1.PHP는 HTML을 포함하고 HTTP 요청 및 응답을 처리 할 수 ​​있으며 다양한 데이터베이스를 지원할 수 있습니다. 2.PHP는 강력한 커뮤니티 지원 및 오픈 소스 리소스를 통해 동적 웹 컨텐츠, 프로세스 양식 데이터, 액세스 데이터베이스 등을 생성하는 데 사용됩니다. 3. PHP는 해석 된 언어이며, 실행 프로세스에는 어휘 분석, 문법 분석, 편집 및 실행이 포함됩니다. 4. PHP는 사용자 등록 시스템과 같은 고급 응용 프로그램을 위해 MySQL과 결합 할 수 있습니다. 5. PHP를 디버깅 할 때 error_reporting () 및 var_dump ()와 같은 함수를 사용할 수 있습니다. 6. 캐싱 메커니즘을 사용하여 PHP 코드를 최적화하고 데이터베이스 쿼리를 최적화하며 내장 기능을 사용하십시오. 7

PHP vs. Python : 차이점 이해 PHP vs. Python : 차이점 이해 Apr 11, 2025 am 12:15 AM

PHP와 Python은 각각 고유 한 장점이 있으며 선택은 프로젝트 요구 사항을 기반으로해야합니다. 1.PHP는 간단한 구문과 높은 실행 효율로 웹 개발에 적합합니다. 2. Python은 간결한 구문 및 풍부한 라이브러리를 갖춘 데이터 과학 및 기계 학습에 적합합니다.

PHP 대 기타 언어 : 비교 PHP 대 기타 언어 : 비교 Apr 13, 2025 am 12:19 AM

PHP는 특히 빠른 개발 및 동적 컨텐츠를 처리하는 데 웹 개발에 적합하지만 데이터 과학 및 엔터프라이즈 수준의 애플리케이션에는 적합하지 않습니다. Python과 비교할 때 PHP는 웹 개발에 더 많은 장점이 있지만 데이터 과학 분야에서는 Python만큼 좋지 않습니다. Java와 비교할 때 PHP는 엔터프라이즈 레벨 애플리케이션에서 더 나빠지지만 웹 개발에서는 더 유연합니다. JavaScript와 비교할 때 PHP는 백엔드 개발에서 더 간결하지만 프론트 엔드 개발에서는 JavaScript만큼 좋지 않습니다.

PHP vs. Python : 핵심 기능 및 기능 PHP vs. Python : 핵심 기능 및 기능 Apr 13, 2025 am 12:16 AM

PHP와 Python은 각각 고유 한 장점이 있으며 다양한 시나리오에 적합합니다. 1.PHP는 웹 개발에 적합하며 내장 웹 서버 및 풍부한 기능 라이브러리를 제공합니다. 2. Python은 간결한 구문과 강력한 표준 라이브러리가있는 데이터 과학 및 기계 학습에 적합합니다. 선택할 때 프로젝트 요구 사항에 따라 결정해야합니다.

PHP의 영향 : 웹 개발 및 그 이상 PHP의 영향 : 웹 개발 및 그 이상 Apr 18, 2025 am 12:10 AM

phphassignificallyimpactedwebdevelopmentandextendsbeyondit

PHP : 많은 웹 사이트의 기초 PHP : 많은 웹 사이트의 기초 Apr 13, 2025 am 12:07 AM

PHP가 많은 웹 사이트에서 선호되는 기술 스택 인 이유에는 사용 편의성, 강력한 커뮤니티 지원 및 광범위한 사용이 포함됩니다. 1) 배우고 사용하기 쉽고 초보자에게 적합합니다. 2) 거대한 개발자 커뮤니티와 풍부한 자원이 있습니다. 3) WordPress, Drupal 및 기타 플랫폼에서 널리 사용됩니다. 4) 웹 서버와 밀접하게 통합하여 개발 배포를 단순화합니다.

PHP vs. Python : 사용 사례 및 응용 프로그램 PHP vs. Python : 사용 사례 및 응용 프로그램 Apr 17, 2025 am 12:23 AM

PHP는 웹 개발 및 컨텐츠 관리 시스템에 적합하며 Python은 데이터 과학, 기계 학습 및 자동화 스크립트에 적합합니다. 1.PHP는 빠르고 확장 가능한 웹 사이트 및 응용 프로그램을 구축하는 데 잘 작동하며 WordPress와 같은 CMS에서 일반적으로 사용됩니다. 2. Python은 Numpy 및 Tensorflow와 같은 풍부한 라이브러리를 통해 데이터 과학 및 기계 학습 분야에서 뛰어난 공연을했습니다.

See all articles