>Java >java지도 시간 >Java8에서 람다 표현식을 사용하는 방법

Java8에서 람다 표현식을 사용하는 방법

黄舟
黄舟원래의
2017-09-19 10:34:581338검색

Java8의 가장 배울 만한 기능은 람다 표현식입니다. 다음 글에서는 주로 Java8 학습 튜토리얼에서 람다 표현식 사용에 대한 관련 정보를 샘플 코드를 통해 자세히 소개하고 있어 매우 유용합니다. 모든 사람의 공부나 업무를 위해 특정 참고 자료와 학습 가치가 있습니다. 필요한 친구는 편집자를 따라 함께 배울 수 있습니다.

머리말

이전 글에서는 람다 표현식의 구문을 소개하고 람다 표현식의 사용 시나리오와 람다 표현식 사용의 이점을 소개했습니다. 이 기사에서는 람다 식을 정의하고 사용하는 방법과 다른 언어와 비교하여 Java에서 람다 식의 특별한 사양을 예제와 함께 설명합니다.

익명 내부 클래스 사용의 예

우선 Java 8이 등장하기 전에는 람다 표현식이 내부 클래스를 사용하여 수행할 수 있었던 작업을 프로그래밍을 단순화할 수 있다는 점을 분명히 하겠습니다.
아래 예에서는 조건에 따라 목록에서 독자를 선택합니다.

TantanitReader 정의:


public class TantanitReader {
 private int age;
 private String loginName;
 private String realName;
 private String career;

 public TantanitReader() {
 }

 public TantanitReader(int age, String loginName, String realName, String career) {
 this.age = age;
 this.loginName = loginName;
 this.realName = realName;
 this.career = career;
 }

 public int getAge() {
 return age;
 }

 public void setAge(int age) {
 this.age = age;
 }

 public String getLoginName() {
 return loginName;
 }

 public void setLoginName(String loginName) {
 this.loginName = loginName;
 }

 public String getRealName() {
 return realName;
 }

 public void setRealName(String realName) {
 this.realName = realName;
 }

 public String getCareer() {
 return career;
 }

 public void setCareer(String career) {
 this.career = career;
 }

 @Override
 public String toString() {
 return "age:"+this.getAge()+",loginName:"+this.loginName
 +",realName:"+this.getRealName()+",career:"+this.getCareer();
 }
}

판단 인터페이스 정의:


public interface Predicate<T> {
 boolean test(T t);
}

선택 기능 정의:


public class SelectService<T> {
 public List<T> select(Collection<T> source, Predicate<T> predicate){
 List result = new LinkedList();
 for(T element:source){
  if (predicate.test(element)) {
  result.add(element);
  }
 }
 return result;
 }

}

테스트용 예제 작성, 성인 독자 및 청소년 선택(포함) 10세 ) reader:


public class TantanitReaderPredicateTest {


 public static void main(String[] args) {
 SelectService tantanitReaderSelectSerive
 =new SelectService<TantanitReader>();
 List<TantanitReader> source = new LinkedList<>();
 source.add(new TantanitReader(10,"jack","张三","学生"));
 source.add(new TantanitReader(18,"rose","李四","学生"));
 source.add(new TantanitReader(19,"mike","王五","程序员"));
 source.add(new TantanitReader(20,"jack","赵六","作家"));

 List<TantanitReader> audultReaders
 =tantanitReaderSelectSerive.select(source, new Predicate() {
  @Override
  public boolean test(Object o) {
  TantanitReader tantanitReader=(TantanitReader)o;
  return tantanitReader.getAge()>=18;
  }
 });
 System.out.println("tantanit.com 成年读者名单如下:");
 printTantanitReaders(audultReaders);

 System.out.println("tantanit.com 十多岁(包含 10 岁)成员如下:");
 List<TantanitReader> teenReaders
 =tantanitReaderSelectSerive.select(source, new Predicate() {
  @Override
  public boolean test(Object o) {
  TantanitReader tantanitReader=(TantanitReader)o;
  return tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19;
  }
 });
 printTantanitReaders(teenReaders);
 }


 public static void printTantanitReaders(List<TantanitReader> tantanitReaders) {
 for (TantanitReader tantanitReader : tantanitReaders) {
  System.out.println(tantanitReader.toString());
 }
 }


}

실행 후 인쇄된 결과는 다음과 같습니다.


tantanit.com 成员读者名单如下:
age:18,loginName:rose,realName: 李四,career: 学生 
age:19,loginName:mike,realName: 王五,career: 程序员
age:20,loginName:jack,realName: 赵六,career: 作家
tantanit.com 十多岁(包含10 岁)成员如下:
age:10,loginName:jack,realName: 张三,career: 学生
age:18,loginName:rose,realName: 李四,career: 学生
age:19,loginName:mike,realName: 王五,career: 程序员

reader를 두 번 선택하면 new Predicate()가 필요하고 테스트 메서드가 재정의되는 것을 볼 수 있습니다. 실제 차이점은 실제로 판단문에만 있습니다:


tantanitReader.getAge()>=18


tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19

하지만 Java8 이전에는 람다 표현식이 없었기 때문에 이러한 중복성만 허용할 수 있었습니다. 람다 표현식을 사용하여 코드를 단순화하는 방법은 무엇입니까?

다른 언어와 달리 Java 개발자의 기존 프로그래밍 습관을 관리하기 위해 Java8에서는 람다 식을 사용하는 메커니즘을 설계할 때 인터페이스를 계속 사용해야 하며 사용되는 인터페이스는 기능적 인터페이스여야 한다고 규정합니다. 이 예에서는 다음을 계속 사용할 수 있습니다.


public interface Predicate<T> {
 boolean test(T t);
}

이 인터페이스에는 추상 메서드가 하나만 있으므로(java8에서는 기본 메서드를 도입했으며 기본 메서드에는 특정 구현이 있으며 추상 메서드가 아님) 기능적 인터페이스입니다. . 기능적 인터페이스는 @FunctionalInterface를 사용하여 선언할 수도 있고 그렇지 않을 수도 있습니다. 그러나 이를 추가한 후 컴파일러는 컴파일 단계에서 인터페이스가 기능적 인터페이스의 정의를 준수하는지 확인하므로 여기서는 새 인터페이스를 정의하고 @FunctionalInterface 선언을 추가합니다:


@FunctionalInterface
public interface PredicateFunction<T> {
 boolean test(T t);
}

및 SelectService 메서드를 다음과 함께 추가합니다. PredicateFunction을 매개변수로 사용:


public List<T> select(Collection<T> source, PredicateFunction<T> predicate){
 List result = new LinkedList();
 for(T element:source){
 if (predicate.test(element)) {
  result.add(element);
 }
 }
 return result;
}

테스트 예제를 다시 수정합니다:


public class TantanitReaderPredicateFunctionTest {

 public static void main(String[] args) {
 SelectService tantanitReaderSelectSerive
 =new SelectService();
 List source = new LinkedList<>();
 source.add(new TantanitReader(10,"jack","张三","学生"));
 source.add(new TantanitReader(18,"rose","李四","学生"));
 source.add(new TantanitReader(19,"mike","王五","程序员"));
 source.add(new TantanitReader(20,"jack","赵六","作家"));

 PredicateFunction predicateFunction
  = (TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18;
 List audultReaders
 =tantanitReaderSelectSerive.select(source,predicateFunction);

 System.out.println("tantanit.com 成员读者名单如下:");
 printTantanitReaders(audultReaders);

 System.out.println("tantanit.com 十多岁(包含 10 岁)成员如下:");
 PredicateFunction predicateFunction2
 = (TantanitReader tantanitReader)
 -> tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19;
 List teenReaders
 =tantanitReaderSelectSerive.select(source,predicateFunction2);
 printTantanitReaders(teenReaders);
 }


 public static void printTantanitReaders(List tantanitReaders) {
 for (TantanitReader tantanitReader : tantanitReaders) {
  System.out.println(tantanitReader.toString());
 }
 }

}

이 코드가 어떻게 적용되는지 분석해 보겠습니다.


PredicateFunction<TantanitReader> predicateFunction
 = (TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18;
List<TantanitReader> audultReaders
=tantanitReaderSelectSerive.select(source,predicateFunction);

이 코드는 PredicateFunction 유형의 인스턴스를 생성하고 tantanitReaderSelectServive의 select 메소드에 대한 매개변수로 인스턴스를 참조하고 select 메소드를 실행합니다. select를 실행하는 동안 predicateFunction의 테스트 메서드가 호출되고 테스트 메서드의 내용은 우리가 전달한 람다 식입니다. 마지막으로 람다 식에 따라 리더가 선택됩니다.

한 단계 더 나아가면 일반적으로 predicateFunction 변수를 정의할 수 없지만 다음과 같이 람다 표현식을 tantanitReaderSelectSerive의 select 메소드에 매개변수로 직접 전달할 수 있습니다.


List<TantanitReader> audultReaders
=tantanitReaderSelectSerive.select(
 source,(TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18
);

그러나 이 예에서는 컴파일 오류가 발생합니다. 실제로 TantanitReader select 메소드가 generics를 사용하기 때문에 tantanitReaderSelectServive의 select 메소드 정의와 일치하지 않는다고 보고됩니다. Java 8의 문서에는 제네릭을 사용할 때 람다 표현식을 매개변수로 직접 사용할 수 없다고 규정되어 있습니다. 이는 매우 당혹스러운 일입니다. 제네릭을 사용하지 않으면 이 문제가 발생하지 않습니다.

요약

다음은 람다 표현식 사용법을 요약한 것입니다

  • 먼저 기능적 인터페이스(Functional Interface)를 정의하고, 인터페이스에서 사용해야 하는 추상 메서드를 정의합니다.

  • 비즈니스 메소드를 작성하고, 기능적 인터페이스를 매개변수로 취하고, 인터페이스에서 정의한 메소드를 호출하여 비즈니스 로직을 완성합니다.

  • 비즈니스 메서드를 호출하고 람다 식을 매개변수로 전달합니다.

제네릭을 사용하는 경우 마지막 단계는 먼저 기능적 인터페이스의 인스턴스에 대한 참조를 정의한 다음 이를 비즈니스 메서드에 매개변수로 전달하는 것입니다.

또한 람다 식은 계속해서 함수 참조로 단순화될 수 있으며 이에 대해서는 이후 문서에서 설명하겠습니다.

요약

위 내용은 Java8에서 람다 표현식을 사용하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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