▶ Graphical User Interface (GUI) - 그래픽 사용자 인터페이스


- GUI 는 사용자와 상호작용하는 프로그램을 쉽게 작성할 수 있다.


현재 자바에서 사용되어지는 GUI 객체는 Abatract Windows Toolkit(AWT) 와 스윙(Swing)으로 두 가지가 있다.

  

- AWT -> java.awt.*;

1. 운영 체제가 제공하는 자원을 이용하여 컴포넌트를 생성한다

2. 다양한 플랫폼에서 쉽게 컴포넌트를 제공할 수 있다.

3. 컴포넌트가 플랫폼에 종속되므로 실행되는 플랫폼에 따라서 컴포넌트의 모습이 달라진다.

, 일관된 화면을 제공하는 것이 어렵다.

 

플랫폼은 소프트웨어 응용 프로그램들을 돌리는 데 쓰이는 하드웨어와 소프트웨어의 결합이다. 플랫폼은 하나의 운영 체제 또는 컴퓨터 아키텍처라고 단순히 말할 수 있으며 그 두 가지를 통칭해서 말할 수도 있다일반적으로 플랫폼은 컴퓨터의 아키텍처, 운영 체제(OS), 프로그램 언어, 그리고 관련 런타임 라이브러리 또는 GUI를 포함한다. - [출처] 위키백과

 

- Swing -> javax.swing.*; / javax.swing.event*;

1. 컴포넌트가 자바로 작성되어 있기 때문에 플랫폼에 상관없이 일관된 화면을 보여준다.

2. 스윙은 AWT에서 없는 다양한 컴포넌트들을 제공한다. (현재는 스윙의 사용이 권장되고 있다.)

3. 스윙은 AWT를 대체하기 위하여 AWT의 컴포넌트 이름 앞에 J를 붙인 JButton, JLabel 등으로 나타내고, 동일한 기능을 제공한다.

4. 다양한 룩앤필(Look and feel)을 선택할 수 있다. 룩앤필이란 각 운영 체제가 제공하는 GUI의 모습을 뜻한다. 룩앤필에 따라서 외관이 조금씩 달라진다.




▶ Swing의 기본 구조

 


1. Component(컴포넌트) : 레이블, 버튼, 텍스트 필드와 같은 GUI를 작성하는 기본 빌딩 블록을 의미한다. 윈도우 시스템(ex- javaFX)에서는 컨트롤(control)이라고 한다. 컴포넌트 클래스는 화면에 표시되어서 사용자와 상호 작용하는 시각적인 객체를 나타낸다. 이 클래스는 모든 스윙 클래스들이 가지고 있는 공통적인 기본 메소드를 정의한다.

 

- 기본 컴포넌트 : JButton, JLabel, JCheckBox, JList, JMenu, JTextField, JTextArea

- 컨테이너 컴포넌트 : JFrame, JDialog, JApplet, JPanel, JScrollPane

 

2. Container(컨테이너) : 다른 컴포넌트들을 내부에 놓을 수 있는 컴포넌트


- 최상위 컨테이너 : 다른 컨테이너 안에 포함될 수 없는 컨테이너

ex) JFrame, JDialog, JApplet

* JFrame : 윈도우와 메뉴를 가지는 일반적인 데스크탑 애플리케이션에 적합하다.

* JDialog : 메뉴가 없는 대화 상자 형식의 간단한 애플리케이션에 사용된다.

* JApplet : 애플릿을 작성하는데 사용된다.

 

- 일반 컨테이너 : 다른 컨테이너 안에 포함될 수 있는 컨테이너

ex) JPanel, JScrollPane




▶ 프레임(Frame)


컴포넌트를 추가할 수 있는 윈도우


생성자

설명 

 JFrame()

 타이틀이 없는 새로운 프레임을 생성

 JFrame(String title)

 매개변수로 받은 문자열을 타이틀로 하여 새로운 프레임을 생성


- JFrame은 "pane"이라는 것들로 분리되어지는데, 이들은 서로가 최상위에 위치해있다. Pane의 종류로는 Root Pane, Layered Pane, Content Pane, Glass Pane이 있고, 각 Pane은 다른 기능을 수행한다. 주로 컴포넌트가 추가되는 pane은 ContentPane이고 여기서는 ContentPane에 대해서만 다룬다.



- Content Pane에는 직접적으로 접근할 수 있는데, 그 방법으로 JPanel을 이용하는 것이다. 프레임의 ContentPane에 생성한 JPanel을 추가할 경우 보다 쉽게 프레임 속의 모든 것들을 다루게 된다.


프레임의 속성 변경 : 프레임도 객체이므로 속성을 변경하려면 set으로 시작하는 설정자 메소드를 사용하여야 한다

(반환형은 모두 void 이다.)


 설정자 메소드

 설명

setLocation(int x, int y)

(컴포넌트도 사용가능)

 모니터에서 프레임(or 컴포넌트)의 위치 설정

setSize(int width, int height)

(컴포넌트도 사용가능)

 프레임(or 컴포넌트)의 크기 설정

setBounds(int x, int y, int w, int h)

(컴포넌트도 사용가능)

 프레임(or 컴포넌트)의 위치와 크기 설정

 setIconImage(ImageIcon icon)

 윈도우 시스템에 타이틀 바태스크 스위처에 표시할 아이콘을 설정

setTitle(String title) 

 타이틀 바의 제목을 설정한다.

setResizable(boolean resizable) 

 사용자가 프레임의 크기를 조절할 수 있는 지 설정한다.

 setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE)

 사용자가 프로그램 닫기를 눌렀을 때 프로그램이 완전히 종료되는

 지의 여부를 설정하며, 기본값은 닫기를 눌러도 프로그램이 종료되

 지 않도록  되어 있다.

setLayout(LayoutManager manager)

(패널도 사용가능)

 레이아웃을 설정하는 메소드로 매개변수는 컴포넌트들을 배치하

 는 배치관리자이다. 디폴트 값은 BorderLayout 이고, 컴포넌트

 위치를 변경하고 싶다면 LayoutManager null로 설정해야 한다.

setVisible(boolean visible)

(컴포넌트도 사용가능)

 프레임(or 컴포넌트)이 눈에 보이게 하는지 설정하는 메소드이다. 

 기본값이 false 이므 true로 설정해야 프레임을 볼 수 있다

 설정안하면 안보인다.

setLocationRelativeTo(Component component) 

 매개변수로 지정받은 컴포넌트 위치에 따라 상대적으로 프레임의

 위치가 결정된다. 그러나 매개변수로 null을 전달할 경우 모니터 

 화면의 중앙을 기준으로 프레임이 배치된다.

 setMenuBar(JMenuBar menubar)

 현재 프레임에 메뉴바를 붙인다.

 setContentPane(JPanel panel)

 컨테이너에 패널을 추가




▶ 패널(Panel)


- 컴포넌트들을 포함하고 있도록 설계된 컨테이너이다. 패널에 컴포넌트들을 추가하고 해당 패널을 프레임에 추가하는 것이 일반적으로 컴포넌트들을 화면에 표시하는 방법이다. 패널을 프레임에 추가하지 않고 프레임에 컴포넌트들을 직접 추가하는 것도 가능하다. 그러나 별도의 패널을 쓰는 것이 유지 보수 및 배치 관리에 좋은 경우가 많다. 쉽게 말해 패널은 스윙의 "위젯"을 성립시키기 위해 이용되어진다.

 

 생성자

 설명

 JPanel()

 새로운 패널 생성

 JPanel(boolean isDoubleBuffered)

 매개변수가 참일 경우 더블 버퍼링을 사용

 JPanel(LayoutManager layouy)

 매개변수로 지정된 배치 관리자를 사용하는 패널 형성


 메소드

 설명

void setLayout(LayoutManager layout)

 설명은 위와 같고, 디폴트는 FlowLayout 이다.

 void setToolTipText(String text)

 사용자가 마우스를 패널(or 컴포넌트)의 빈 곳에 올려놓으면 툴팁을 표시한다.




▶ 그 외 프레임 또는 패널에서 자주 쓰이는 메소드


 메소드

 설명

 void add(Component c)

 지정된 컴포넌트를 추가한다.

void remove(Component c)

 지정된 컴포넌트를 제거한다.

void pack()

 프레임 크기를 추가된 컴포넌트들의 크기에 맞도록 조절한다.




 레이블(Label) 


편집이 불가능한 텍스트를 표시하기 위한 컴포넌트


JLabel

 생성자

 JLabel()

 비어있는 새로운 레이블을 생성한다.

 JLabel(String text)

 매개변수로 받은 텍스트를 표시하는 레이블을 생성한다.

 JLabel(Icon icon)

 매개변수로 받은 아이콘을 표시하는 레이블을 생성한다. 

 주로 ImageIcon을 사용하여 레이블의 이미지를 생성한다.

 메소드

 void setText(String text)

 레이블의 텍스트를 설정한다.

 String getText()

 레이블의 텍스트를 반환한다.

void setIcon(Icon icon)

 레이블의 아이콘을 설정한다. 

  Icon getIcon()

 레이블의 아이콘을 반환한다. 




 버튼(Button)


- 사용자가 클릭했을 때, 이벤트를 발생하여 원하는 행위를 하게 하는 데 이용된다.

 

JButton

 생성자

 JButton()

 레이블이 없는 버튼을 생성한다.

JButton(String label)

 매개변수로 받은 텍스트가 표시되는 버튼을 생성한다.

 JButton(Icon icon)

 매개변수로 받은 아이콘이 표시되는 버튼을 생성한다. 

 주로 ImageIcon을 사용하여 버튼 이미지를 생성한다.

 메소드

 void setText(String text)

 버튼의 텍스트를 설정한다.

 String getText()

 버튼의 텍스트를 반환한다.

void setIcon(Icon icon)

 버튼의 아이콘을 설정한다. 

  Icon getIcon()

 버튼의 아이콘을 반환한다. 

 void doClick()

 사용자가 버튼을 누른 것처럼 이벤트를 발생한다.

void setBorderPainted(boolean value)

 버튼의 경계를 나타내거나 감춘다.

void setContentAreaFilled(boolean value)

 버튼의 배경을 채울 것인지를 설정한다.

 void setEnabled(boolean value)

 버튼을 활성화하거나 비활성화한다.

 void setRollOverEnabled(boolean value)

 마우스가 버튼 위에 있으면 경계를 진하게 하는 롤오버 효과를 

 설정한다.




 텍스트 필드(TextField) 


- 입력이 가능한 한줄의 텍스트 필드를 만드는데 사용된다.

 

JTextField

 생성자

JTextField()

 비어있는 새로운 텍스트 필드를 생성한다.

 JTextField(int num)

 매개변수로 지정된 글자 수를 가지고 있는 텍스트 필드를 생성한다.

 JTextField(String text)

 지정된 문자열로 초기화된 텍스트 필드 생성

 메소드

 void setText(String text)

 텍스트 필드의 문자열을 설정한다.

 String getText()

 텍스트 필드에 입력된 문자열을 반환한다.

void setEditable(boolean value)

 사용자가 텍스트 필드를 수정할지의 여부를 설정한다.

  boolean isEditable()

 사용자가 텍스트 필드를 수정할 수 있으면 true, 없으면 false를 반환한다.








▶ 중첩 클래스 및 중첩 인터페이스를 사용하는 이유


- 자바는 객체 지향 프로그램으로 각 클래스가 관계를 맺으며 상호작용을 한다. 어떤 클래스는 많은 클래스와 상호작용을 하기도 하지만, 다른 어떤 클래스는 특정 클래스하고만 상호작용을 하기도 한다. 여러 클래스와 상호작용을 할 때는 해당 클래스를 독립적으로 생성하는 것이 좋지만, 특정 클래스하고만 상호작용을 할 경우 특정 클래스 내부에 해당 클래스를 생성하는 것이 효율적이다.


- 중첩 인터페이스는 해당 클래스와 깊은 상호작용을 하는 구현 클래스를 생성하기 위해 사용된다.




▶ 중첩 클래스(Nested Class)


- 하나의 클래스 내부에 선언된 다른 클래스를 의미한다. 즉, 포함되어지는 클래스이다.


- 두 클래스의 멤버간에 접근이 용이하다. 중첩 클래스는 자신을 포함하는 클래스의 모든 멤버에 접근할 수 있다. 즉, 접근 제한자가 private인 멤버라 할지라도 중첩 클래스는 접근할 수 있다.


- 불필요한 상호작용을 맺은 클래스들을 외부로부터 감춤으로써 코드가 간결해진다.



class ClassName {

class NestedClassName {

...

}

}





▶ 멤버 클래스


- 클래스의 멤버로서 선언되는 중첩 클래스


- 해당 클래스나 객체가 사용 중일 경우 재사용이 가능하다.


- 멤버 클래스도 하나의 클래스이기 때문에 컴파일하면 바이트 코드 파일(~.class)가 생성된다.


멤버 클래스의 바이트 코드 파일 이름 : 바깥클래스명$ 멤버클래스명.class

로컬 클래스의 경우 : 바깥클래스명$1 로컬클래스명.class


- 인스턴스 멤버 클래스


class A {

class B { ... } // A객체를 생성해야 B 중첩 클래스(인스턴스 멤버 클래스)를 사용할 수 있다.

}


  • 중첩 클래스를 포함한 클래스를 생성해야만 해당 중첩 클래스를 사용할 수 있다.
  • 정적 필드와 메소드는 선언할 수 없다.
  • 외부 클래스에서 인스턴스 멤버 클래스를 생성하려면 바깥 클래스를 먼저 선언(객체 생성)해야 한다.
- 정적 멤버 클래스


class A {

static class B { ... } // A 클래스로 바로 접근 가능한 B 중첩 클래스(정적 멤버 클래스)

}


  • 중첩 클래스를 포함한 클래스로 바로 접근 가능한 중첩 클래스
  • 모든 종류의 필드와 메소드를 선언할 수 있다.
  • 바깥 객체를 생성하지 않아도 바로 정적 멤버 클래스를 이용할 수 있다.



▶ 로컬 클래스

- 메소드 내부에서 선언된 중첩 클래스

- 메소드 실행 시에만 사용되며, 메소드가 종료되면 사라진다.

- 로컬 클래스는 접근 제한자(public, private, protected) 및 static을 붙일 수 없다. 메소드 내부에서 사용되므로 접근을 제한할 필요가 없기 때문이다.

- 정적 필드와 메소드를 선언할 수 없다.

- 로컬 클래스가 선언된 메소드는 new 연산자를 통해 객체를 생성해야 해당 로컬 클래스를 사용할 수 있다.

- 다음과 같이 비동기 처리를 위한 스레드 객체 생성 시 사용된다.

void method() {

class DownloadThread extends Thread { ... } //로컬 클래스

DownloadThread th = new DownloadThread(); // 선언과 초기화 필수

th.start();

}





▶ 중첩 클래스의 접근 제한


- 멤버 클래스

  • 인스턴스 멤버 클래스는 바깥 클래스의 인스턴스 필드를 통해 객체를 생성하거나 인스턴스 메소드 내부에서 객체를 생성할 수 있지만 정적 필드나 메소드는 해당되지 않는다. 즉, 바깥 클래스의 정적 멤버에 포함될 수 없다.
  • 인스턴스 멤버 클래스 내부에서 바깥 클래스로 접근할 때 모든 필드와 메소드가 가능하다. 즉, 바깥 클래스의 정적 멤버에 접근가능하다.
  • 정적 멤버 클래스는 바깥 클래스의 모든 필드의 초기값으로 혹은 모든 메소드에서 객체를 생성할 수 있다.
  • 정적 멤버 클래스 내부에서는 바깥 클래스의 정적 필드와 메소드에만 접근할 수 있다. 즉, 바깥 클래스의 인스턴스 멤버에 접근할 수 없다.

- 로컬 클래스
  • 로컬 클래스 내부에서는 바깥 클래스로 제한 없이 접근할 수 있다.
  • 로컬 클래스에서 사용되는 매개변수와 로컬 변수는 모두 final 특성(최종값)을 갖는다.
☞ 로컬 클래스의 객체는 메소드 실행이 끝나도 힙 메모리에 남아있으므로 사용할 수 있지만, 메소드의 매개변수 및 로컬변수는 메소드 실행이 끝나면 사라지므로 로컬 객체에서 사용할 경우 에러가 발생한다. 해결 방법으로 컴파일 시 로컬 클래스에서 사용하는 매개변수 및 로컬변수는 로컬 클래스 내부의 기억장소에 두고 사용한다. 그러나 값이 수정되면 복사해 둔 기억장소가 변경되므로 이 문제를 해결하고자 변수를 final로 선언하여 수정을 맞는다. 쉽게 말해, final 로 선언된 매개변수와 로컬변수만 로컬 클래스에서 사용가능하다. 자바 8부터는 final 키워드 없이 선언되어도 여전히 값을 수정할 수 없는 final 특성을 갖는다.

☞ final 키워드의 존재 여부의 차이점은 로컬 클래스의 복사 위치이다. final 키워드가 있을 경우 로컬 클래스를 포함한 메소드 내부에서 지역 변수로 복사된다. 없을 경우 로컬 클래스의 필드로 복사된다.


- 중첩 클래스 내부에서 바깥 클래스의 객체 참조를 얻으려면 아래와 같이 선언해야한다.

바깥클래스명.this.필드명;

바깥클래스명.this.메소드명;




▶ 중첩 인터페이스(Nested Interface)


- 클래스의 멤버로 선언된 인터페이스


- 주로 UI(User Interface) 프로그래밍에서 이벤트를 처리할 때 사용된다. 즉 해당 이벤트를 처리하는 구현 클래스를 만드는 것이다.


class ClassName {

interface NestedInterfaceName {

...

}

}







 

 

무명 클래스(Anonymous class)

 

- 말그대로 몸체는 있으나 이름만 없는 클래스를 의미하며, 하나의 객체만 사용할 경우에 이용한다. , 한번만 사용가능하다.


클래스명 변수명 = new 클래스명 { 클래스 몸체 };

 

- 무명 클래스의 선언은 하나의 실행문이므로 끝에 세미콜론(;)이 무조건 붙는다.

 

- 클래스명에 올 수 있는 것은 상속받고자 하는 부모 클래스의 이름이나 구현하고자 하는 인터페이스의 이름을 적어주면 된다.

 

- 무명클래스는 클래스를 정의하면서 동시에 객체를 생성하게 된다.

 

- 코드의 양을 줄일 수 있다는 장점이 있으나 코드가 난해해진다.

 

 

 

익명 구현 객체

 

- 무명클래스를 통해 인터페이스를 구현하고자 하는 일회성의 구현 객체를 말한다.

 

- 대부분 UI(user interface) 프로그래밍에서 이벤트를 처리하거나 임시 작업 스레드를 만들기 위해 익명 구현 객체를 많이 이용한다.

 

- 형식 


인터페이스타입 변수명 = new 인터페이스타입() {


//인터페이스에 선언된 추상메소드의 실체 메소드 선언


}; 



- 추가적으로 필드와 메소드도 선언이 가능하지만, 익명 객체 안에서만 사용할 수 있고, 해당 변수로는 접근할 수는 없다.





 

 

 

인터페이스(Interface)

 

- 추상 메소드들로만 이루어진 형태로, 객체의 사용 방법을 정의한 새로운 타입이다.

 

- 상수 외에 필드를 선언할 수 없다.

 

- 자바에서 인터페이스란 객체와 객체 사이의 상호 작용이 이루어지는 인터페이스이다.

 

- 추상 메소드의 성질을 이용하면, 인터페이스는 여러 객체들과 사용이 가능하므로 어떤 객체를 사용하느냐에 따라 실행 내용과 리턴값이 다를 수 있다.

 

- , 개발 코드를 수정하지 않고 사용하는 객체만 변경하여 메소드를 실행한다.

 

- 객체의 교환성을 높여주기 때문에 다형성을 구현하는 매우 중요한 역할을 한다.

 

상속은 같은 종류의 하위 클래스를 만드는 기술이고, 인터페이스는 사용 방법이 동일한 클래스를 만든다는 점에서 개념적인 차이가 있다. 그러나 둘 다 다형성을 구현하는 기술이다.


- 인터페이스를 구현하는 객체를 구현 객체라고 부르며, 구현 객체를 생성하는 클래스를 구현 클래스라고 부른다. 구현 클래스는 추상 메소드의 몸체를 반드시 구현하여야 한다.

 

- 인터페이스의 모든 메소드는 기본적으로 public 접근 제한자를 갖기 때문에 구현 클래스에서 작성할 때 public 보다 더 강한 접근 제한자(default, private )를 사용할 수 없다. , public을 생략해선 안된다.

 

- 상속과는 달리 구현 클래스는 다수의 인터페이스를 구현할 수 있다.

 

- 인터페이스끼리는 상속이 가능하다.

public interface 인터페이스1 extends 인터페이스2 { ... }

 

- 구현 클래스는 선언부에 implements 키워드를 추가하고 인터페이스 명을 명시해야 한다.

* implements 시행하다, 도구라는 뜻이다.

 

- 인터페이스 타입의 변수는 참조형 변수이기 때문에 구현 객체가 대입될 경우 구현 객체의 번지를 저장한다.


 


인터페이스의 구성 멤버

 

- 상수 필드 : 상수는 인터페이스에 고정된 값이므로 런타임 시에 데이터를 바꿀 수 없다. 상수를 선언할 때에는 반드시 초기값을 대입해야 한다.


public static final 자료형 변수명;


 

- 추상 메소드 : 객체가 사용할 메소드의 형태를 설명한 것으로, 어ᄄᅠᆫ 매개값이 필요하고 리턴 타입이 무엇인지만 알려준다. 실체 실행부는 해당 인터페이스를 구현하는 클래스에서 작성해야 한다.

 

- 디폴트 메소드 : 자바 8버전에서만 허용하는 멤버이며, 기존 인터페이스를 확장해서 사용하기 위함이다. 선언할 때 인터페이스를 구현한 실제 객체가 가지고 있는 메소드라고 생각하면 된다. 따라서 구현 객체에서는 인터페이스의 디폴트 메소드의 실체 메소드를 작성할 필요가 없다.

 

- 정적 메소드 : 자바 8버전에서만 허용하는 멤버이며, 디폴트 메소드와 달리 객체가 없어도 인터페이스만으로 호출이 가능하다. 쉽게 말해, 인터페이스를 선언하지 않고 "인터페이스이름.정적메소드명()"과 같이 바로 호출이 가능함.

 

 

 

인터페이스의 일반적인 형태

 

- 인터페이스의 모든 메소드는 추상 메소드로 간주되므로 abstract키워드는 사용하지 않는다.

 


public interface 인터페이스명 {

//상수

...

 

//추상 메소드

리턴타입 추상메소드1(...);

리턴타입 추상메소드2(...);

...


//디폴트 메소드 -> default를 붙여야한다. 안 붙이면 추상 메소드로 간주됨

default 리턴타입 메소드명(...) { ... }

...


//정적 메소드

static 리턴타입 메소드명(...) { ... }

}




public class 클래스명 implements 인터페이스명 {

 

//FIELDS

...

 

//CONSTRUCTOR

...

 

//METHODS

@Override

리턴타입 추상메소드1(...) {

    ...

}


@Override

리턴타입 추상메소드2(...) {

    ...

}

 

} 


- @Override 란 인터페이스의 추상 메소드에 대한 정확한 실체 메소드인지 컴파일러가 체크하도록 지시하는 어노테이션이다. 생략해도 상관없으나, 실체 메소드가 잘못되지 않도록 해주기 때문에 붙여 주는 것이 좋다.




▶ 인터페이스의 타입 변환


- 구현 객체가 인터페이스 타입으로 변환되는 것은 자동 타입 변환에 해당한다. 이렇게 자동 타입 변환된 구현 객체는 해당 클래스의 필드와 메소드를 사용할 수 없다. 즉, 인터페이스의 상수와 메소드만 사용가능하다.


- 반대로, 인터페이스 타입인 구현 객체를 강제 타입 변환하여 구현 클래스 타입으로 사용할 수 있다.


- 필드와 매개변수의 타입을 인터페이스로 선언하면 여러 구현 객체를 대입하여 다양한 실행 결과를 만들 수 있다.


+ Recent posts