2017-09-11 14 views
0

Lambda와 함께 SwingBuilder를 네이티브 Java로 빌드하는 것이 가능할 수도 있지만, 가능하다면 지금까지 완료되었을 것입니다.Java Lambdas를 사용하여 Groovy의 SwingBuilder와 같은 것을 구현할 수 있습니까?

이 작업을 수행 할 수없는 이유가 있습니까?

나는 이것이 SwingBuilder like GUI syntax for Java?과 비슷하다는 것을 알았지 만, 람다를 믹스에 추가하기를 바랬습니다. SwingBuilder는 대부분 Lambdas를 기반으로합니다. 이것은 Java 8 이전에는 불가능했습니다. 또한 동적 프로그래밍을 사용하여 분명히 사용할 수 없으므로 깨끗하지는 않지만 가능한 일이라고 생각합니다.

+0

가까워 질 수 있어야합니다, 당신은 missingProperty stuff의 niceness를 얻지 못합니다.하지만 이름으로 컨트롤을 얻습니다. Swing이 문 밖으로 나가기 때문에 JavaFX를 대상으로합니다. –

답변

1

먼저 해결해야 할 문제를 파악해야합니다. 가장 큰 문제는 이전 Java 버전에서 내부 클래스를 사용해야하는 간단한 이벤트 바인딩을 만드는 것입니다. 대부분의 경우 싱글 메소드 리스너 인터페이스의 람다 식으로 대체 할 수 있습니다. 멀티 메소드 인터페이스의 경우 this 또는 that Q & A와 같은 어댑터가 필요하지만 한 번만 수행해야합니다.

다른 문제는 초기화 코드의 구조입니다. 원칙적으로 명령형 코드는 정상적으로 작동하지만 컨테이너에 추가 할 구성 요소의 일부 속성을 수정하려면 container.add(new ComponentType());을 변경하여 후속 명령문에서 사용할 새 로컬 변수를 도입해야합니다. 또한 컨테이너 자체에 추가하는 것은 변수에 유지해야합니다. 자바는 중괄호로 지역 변수의 범위를 제한 할 수 있지만 결과는 여전히 서툴 릅니다.

이것은 최고의 출발점입니다. 우리가 사용한다면.

public class SwingBuilder { 
    public static <T> T build(T instance, Consumer<T> prepare) { 
     prepare.accept(instance); 
     return instance; 
    } 
    public static <T extends Container> T build(
             T instance, Consumer<T> prepare, Component... ch) { 
     return build(build(instance, prepare), ch); 
    } 
    public static <T extends Container> T build(T instance, Component... ch) { 
     for(Component c: ch) instance.add(c); 
     return instance; 
    } 
} 

이러한 간단한 일반 방법은 결합 할 수 있기 때문에 이미 조용합니다. import static 사용하여, 사용하는 사이트는 그루비에 비해

JFrame frame = build(new JFrame("Example"), 
    f -> { 
     f.getContentPane().setLayout(
      new BoxLayout(f.getContentPane(), BoxLayout.LINE_AXIS)); 
     f.setResizable(false); 
     f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); 
    }, 
    build(new JLabel("\u263A"), l -> l.setFont(l.getFont().deriveFont(36f))), 
    Box.createHorizontalStrut(16), 
    build(new JPanel(new GridLayout(0, 1, 0, 5)), 
     new JLabel("Hello World"), 
     build(new JButton("Close"), b -> { 
      b.addActionListener(ev -> System.exit(0)); 
     }) 
    ) 
); 
frame.pack(); 
frame.setVisible(true); 

처럼 보일 수 있습니다, 우리는 여전히 속성을 변경 메소드 호출의 대상을 표현하는 변수를 사용해야하지만, 이러한 변수 유형을 사용하여 name ->로 간단하게 선언 할 수 있습니다 lambda 표현을 통해 Consumer을 구현할 때 추론. 또한 변수의 범위는 자동으로 초기화 기간으로 제한됩니다.

이 시작점을 사용하면 자주 사용되는 구성 요소 및 자주 사용되는 속성에 대한 특수 메서드는 물론 다중 메서드 수신기의 이미 언급 한 팩터 리 메서드를 추가 할 수 있습니다. 예 :

public static JFrame frame(String title, Consumer<WindowEvent> closingAction, 
          Consumer<? super JFrame> prepare, Component... contents) { 
    JFrame f = new JFrame(title); 
    if(closingAction!=null) f.addWindowListener(new WindowAdapter() { 
     @Override public void windowClosing(WindowEvent e) { 
      closingAction.accept(e); 
     } 
    }); 
    if(prepare!=null) prepare.accept(f); 
    final Container target = f.getContentPane(); 
    if(contents.length==1) target.add(contents[0], BorderLayout.CENTER); 
    else { 
     target.setLayout(new BoxLayout(target, BoxLayout.PAGE_AXIS)); 
     for(Component c: contents) target.add(c); 
    } 
    return f; 
} 

그러나 나는 생각합니다. 사진은 분명합니다.