Java

메소드

지금까지 배운 것으로도 얼마든지 프로그램을 만들 수 있다. 반복문, 조건문, 변수, 상수와 같은 것들은 사실상 프로그램을 만드는 가장 중요한 도구들이라고 할 수 있다. 지금부터 배우게 될 메소드나 객체지향과 같은 개념들은 웅장하고, 결함이 없고, 유지보수가 쉬운 애플리케이션을 만들기 위한 기법들이라고 할 수 있다. 이것들 없이도 프로그램을 만들 수는 있지만, 이것들 없이 규모있는 애플리케이션을 만든다는 것은 현실적으로 어려운 일이다. 지금까지 만드는 방법을 배웠다면 이제부터는 잘 만드는 방법을 익히는 것이라고 해도 과언이 아니다.

메소드

메소드(method)는 코드를 재사용할 수 있게 해준다. 이번 시간에는 경제적으로 로직을 작성하는 방법에 대해서 알아볼 것이다.

메소드의 형식

여러분은 이미 메소드를 만들고 사용했다. 아래 그림을 보자.

 

이것이 메소드다. 지금까지 수없이 만들었던 저 main이라고 하는 것이 바로 메소드인 것이다. 위의 그림을 자세히 보면 핑크색으로 된 부분과 검은색으로 된 부분이 있다. 이 중에서 검은색으로 된 부분은 이번 토픽에서 다루지 않을 개념이다. 이것들은 객체지향과 함께 언급되어야 본질을 이해할 수 있기 때문에 여기서는 언급하지 않겠다. 메소드를 만들 때 public static이라고 적어야 한다고 일단은 기계적으로 이해하자. 여러분이 주목할 것은 핑크색으로 강조한 부분이다.

 

메소드의 정의와 호출

직접 메소드를 만드는 것을 정의라고 하고, 만들어진 메소드를 실행하는 것을 호출이라고 한다.

아래의 예제를 보자. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo1 {
    public static void numbering() {
		int i = 0;
		while (i < 10) {
			System.out.println(i);
			i++;
		}
	}

	public static void main(String[] args) {
		numbering();
	}
}

결과는 아래와 같다.

0
1
2
3
4
5
6
7
8
9

 아래 그림을 보자.

위의 예제는 numbering이라는 이름의 메소드를 정의하고 있다. 이 메소드는 main이라는 이름의 메소드 안에서 호출되고 있다. 위의 코드는 아래의 코드와 정확하게 동일한 의미를 갖는다.

핑크색으로 표시한 부분의 코드를 numbering이라는 이름의 메소드로 묶어서 외부로 분리한 것이다. 그리고 메소드 numbering의 로직이 필요할 때 numbering();이라고하면 메소드 numbering의 로직이 실행된다.

main

자 그럼 main 메소드가 무엇인지 알아보자. main 메소드는 규칙이다. 여러분이 만들고 싶은 프로그램이 있다면 여러분은 반드시 public static void main(String[] args)가 이끄는 중괄호 안에 실행되기를 기대하는 로직을 위치시켜야 한다. 이것은 약속이기 때문에 여러분은 약속을 지켜야 한다. 그렇게 코드를 작성하면 자바를 실행할 때 자바는 여러분이 작성한 main 메소드를 실행하게 되는 것이다. 여러분은 main 메소드를 작성하고, 자바는 main 메소드를 실행하는 관계라고 할 수 있다.

메소드가 없다면

아래 예제를 보자. 반복문 수업에서 0부터 9까지 출력하는 애플리케이션을 만들었다. 그런데 0부터 9까지를 5번 출력해야 한다면 어떻게 해야 할까? 아래와 같이 해야 할 것이다. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo2 {
    
    public static void main(String[] args) {
		int i = 0;
		while(i<10){
			System.out.println(i);
			i++;
		}
		
		i = 0;
		while(i<10){
			System.out.println(i);
			i++;
		}
		
		i = 0;
		while(i<10){
			System.out.println(i);
			i++;
		}
		
		i = 0;
		while(i<10){
			System.out.println(i);
			i++;
		}
		
		i = 0;
		while(i<10){
			System.out.println(i);
			i++;
		}
	}

}

이 정도는 copy&paste로 해볼 만하다. 하지만 만약 이것을 1000번 해야 한다면? 각각의 로직이 1000 줄에 육박한다면? 그리고 그 내용을 수정해야 한다면? 서서히 암담한 느낌이 들지 않는가? 메소드를 사용한다면 이러한 문제를 현저히 줄일 수 있다. 아래의 예제를 보자. 결과는 같지만 로직은 단 한 번만 등장한다. 이러한 것을 재활용성이라고 한다. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo3 {
    public static void numbering() {
		int i = 0;
		while (i < 10) {
			System.out.println(i);
			i++;
		}
	}

	public static void main(String[] args) {
		numbering();
		numbering();
		numbering();
		numbering();
		numbering();
	}
}

입력과 출력

지금까지 메소드의 첫 번째 면모인 재활용성에 대해서 알아봤다. 자주 사용하는 로직을 메소드로 만들어두면 호출하는 것을 통해서 간편하게 로직을 재활용할 수 있다. 이제 메소드의 두 번째 면모를 살펴볼 것인데 입력과 출력이다.

살아있는 것들은 외부의 자극에 따라서 반응한다. 외부의 자극이 입력이라면 반응은 출력이라고 할 수 있다. 우리가 아는 쓸모있는 대부분의 프로그램이 사용자의 입력에 따라서 다른 결과를 출력한다. 메소드는 프로그램 안에서 동작하는 하나의 작은 프로그램이라고 할 수 있다. 위에서 살펴본 numbering이라는 메소드는 항상 똑같은 동작만을 반복한다. 이것도 재활용이라는 측면에서는 장점이 있지만, 입력 값에 따라서 출력 값을 달리 제공한다면 더욱 쓸모 있는 프로그램이 될 수 있을 것이다.

매개변수와 인자

메소드의 입력 값은 매개변수(parameter)를 통해서 이루어진다. 위의 예제를 조금 개선해보자. 이전 예제는 0부터 9까지의 숫자를 화면에 출력했다. 만약 필요에 따라서 0부터 4까지 출력하고 싶거나 0부터 8까지 출력하고 싶다면 어떻게 해야 할까? 각각에 맞는 메소드를 새로 정의해야 할까? 그렇게 해도 되지만 더 좋은 방법이 있다. 입력 값에 따라서 다른 출력 값을 갖도록 메소드를 정의하면 된다. 즉, 입력을 고민할 때가 된 것이다. 아래의 예제를 보자. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo4 {
    public static void numbering(int limit) {
		int i = 0;
		while (i < limit) {
			System.out.println(i);
			i++;
		}
	}

	public static void main(String[] args) {
		numbering(5);
	}
}

결과는 0부터 4까지 출력한다. 메소드를 호출할 때 괄호에 값을 주고 있는데 저 값을 다른 값으로 바꿔보자. 값에 따라서 다른 결과가 출력되고 있다. 입력을 통해서 메소드의 동작을 제어하고 있다. 아래 그림을 보자.

메소드 numbering의 괄호 안에 위치한 숫자 5는 이 메소드가 호출될 때 limit이라는 변수의 값이 된다. 이 값은 메소드 numbering의 중괄호 안에서만 사용할 수 있다. 위의 코드는 아래의 코드와 동일한 의미를 갖는다.

public static void numbering() {
	int limit = 5;
	int i = 0;
	while (i < limit) {
		System.out.println(i);
		i++;
	}
}

여기서 limit이라는 변수는 메소드 numbering의 정의 부에 있는 로직들에게 5라는 값을 전달하고 있다. 호출에서 입력한 값을 로직으로 매개 한다는 의미에서 이러한 변수를 매개변수라고 부른다. 영어로는 parameter다. 그리고 메소드를 호출할 때 전달된 값인 5를 '인자' 영어로는 argument라고 한다. 관습적으로는 매개변수와 인자를 구분하지 않고 부르는 경우도 많다.

복수의 인자

만약 메소드로 여러개의 입력값을 전달하고 싶다면 어떻게 해야 할까? 다음 예제는 위의 예제를 개선해서 출력할 숫자의 시작 값과 마지막 값을 입력값으로 전달하는 예제다. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo5 {

    public static void numbering(int init, int limit) {
		int i = init;
		while (i < limit) {
			System.out.println(i);
			i++;
		}
	}

	public static void main(String[] args) {
		numbering(1, 5);
	}

}

 결과는 1부터 4까지가 출력된다. 위와 같이 입력 값을 복수로 받고 싶다면 콤마 뒤에 매개변수를 정의해주면 된다. 또 이 메소스를 호출할 때는 매개변수의 순서대로 인자를 배치하면 된다.

return

위의 예제는 화면에 숫자를 출력한다. 물론 이것도 출력이지만 좀 더 활용도가 높은 출력 방법이 있다. 아래 예제를 보자. (실행)

package org.opentutorials.javatutorials.method;

public class MethodDemo6 {
    public static String numbering(int init, int limit) {
		int i = init;
		// 만들어지는 숫자들을 output이라는 변수에 담기 위해서 변수에 빈 값을 주었다.
		String output = "";
		while (i < limit) {
			// 숫자를 화면에 출력하는 대신 변수 output에 담았다.
			output += i;
			i++;
		}
		// 중요!!! output에 담겨 있는 문자열을 메소드 외부로 반환하려면 아래와 같이 return 키워드 뒤에 반환하려는 값을
		// 배치하면 된다.
		return output;
	}

	public static void main(String[] args) {
		// 메소드 numbering이 리턴한 값이 변수 result에 담긴다.
		String result = numbering(1, 5);
		// 변수 result의 값을 화면에 출력한다.
		System.out.println(result);
	}
}

메소드 내에서 사용한 return은 return 뒤에 따라오는 값을 메소드의 결과로 반환한다. 동시에 메소드를 종료시킨다. 한가지 잊지 말아야 할 점은 return을 통해서 반환할 값의 데이터 형식을 메소드의 이름 옆에 명시해주어야 한다는 것이다.

이것은 프로그래밍을 처음 시작하거나, JavaScript나 PHP와 같은 언어들에 익숙한 개발자에게는 까다롭고 귀찮게 느껴질 수 있는 부분이다. 하지만 메소드가 리턴 할 값을 명시함으로서 numbering이라는 메소드는 반드시 문자열의 값을 리턴한다는 것을 보장할 수 있는 장점이 있다. 모든 일에는 장점과 단점이 있다. 장단의 다면성을 충실하게 응시할 때 적합함을 얻을 수 있다. (필자도 노력중이다)

만약 반환 값이 없다면 아래와 같이 void를 적어준다.

굳이 이렇게 복잡하게 데이터를 리턴하는 이유는 무엇일까? 내용을 화면에 출력하는 것은 동일하지 않은가? 결론적으로 말하면 부품으로서의 가치를 높이기 위해서라고 할 수 있다. 만약 여러분이 이 메소드가 출력한 값을 화면에 출력하는 것이 아니라 파일에 기록하고 싶다면 어떻게 해야 할까? 또는 이메일로 보내고 싶다면 어떻게 해야 할까? 3개의 메소드를 만들고 용도에 따라서 코드를 재작성하는 것도 좋은 방법이다. 하지만 더 좋은 방법은 숫자를 출력하고, 숫자를 파일에 기록하고, 숫자로 이메일을 보내는 작업으로부터 숫자를 계산하는 로직을 분리하는 것이다. numbering은 자신이 어떻게 사용될지 모른다. 누구든지 numbering이라는 메소드를 호출할 때 초기값과 마지막 값을 입력하면 numbering은 숫자를 문자열의 형태로 반환하면 되는 것이다. 코드를 보자.

package org.opentutorials.javatutorials.method;

import java.io.*; // 무시

public class MethodDemo7 {
    public static String numbering(int init, int limit) {
		int i = init;
		String output = "";
		while (i < limit) {
			output += i;
			i++;
		}
		return output;
	}

	public static void main(String[] args) {
		String result = numbering(1, 5);
		System.out.println(result);
		try { // 무시
			// 다음 행은 out.txt 라는 파일에 numbering이라는 메소드가 반환한 값을 저장합니다.
			BufferedWriter out = new BufferedWriter(new FileWriter("out.txt"));
			out.write(result);
			out.close();
		} catch (IOException e) {
		} // 무시
	}
}

코드에서 무시라고 표시된 부분은 지금 단계에서는 이해하기 어려운 것이다. 무시하자. 중요한 것은 numbering이라는 메소드로부터 화면에 출력이라는 구체적인 행위를 제거하고 대신에 처리 결과를 반환하고 있다는 사실이다.

return의 특성에 대해서 조금 더 알아보자. return은 값을 반환하는 동작을 한다. 그런데 이것은 return에 대한 반쪽짜리 설명이다. return은 메소드를 중단시키는 역할도 한다. 코드를 보자. (실행)

package org.opentutorials.javatutorials.method;

public class ReturnDemo {
    public static int one() {
		return 1;
		return 2;
		return 3;
	}

	public static void main(String[] args) {
		System.out.println(one());
	}
}

위의 코드는 컴파일조차 되지 않는다. 왜냐하면, return 은 메소드를 종료시키는 역할을 하므로 return이 처음 등장한 이후의 구문은 실행되지 않기 때문이다. 하지만 아래의 예제는 문제가 전혀 없다.

package org.opentutorials.javatutorials.method;

public class ReturnDemo2 {
    public static String num(int i) {
    	if(i==0){
			return "zero";
		} else if(i==1){
			return "one";
		} else if(i==2){
			return "two";
		}
		return "none";
	}

	public static void main(String[] args) {
		System.out.println(num(1));
	}
}

return이 여러 번 등장하지만 return이 중복적으로 실행될 가능성이 없기 때문이다. return "none";를 제거하면 컴파일이 되지 않을 것이다. 

복수의 리턴

메소드는 여러 개의 입력 값을 가질 수 있다. 그렇다면 여러 개의 값을 출력하고 싶다면? 자바는 문법적으로 그런 기능을 제공하지 않는다. 하나의 변수에 여러개의 값을 담아서 출력하면 된다. 아래의 코드를 보자. (실행)

package org.opentutorials.javatutorials.method;

public class ReturnDemo3 {
    public static String getMember1() {
		return "최진혁";
	}

	public static String getMember2() {
		return "최유빈";
	}

	public static String getMember3() {
		return "한이람";
	}

	public static void main(String[] args) {
		System.out.println(getMember1());
		System.out.println(getMember2());
		System.out.println(getMember3());
	}
}

하나의 메소드는 하나의 값만을 반환할 수 있기 때문에 위와 같이 각각의 회원정보를 반환하는 메소드를 만들었다. 무언가 비정상적이지 않은가? 이번엔 배열을 이용한 아래의 코드를 보자. 맴버를 담고 있는 배열을 반환하고 있다. 간단하지 않은가? 메소드 getMembers가 리턴한 배열을 members 변수에 담았다. 이 변수를 이용해서 여러 개의 데이터를 처리 할 수 있게 된다. (실행)

package org.opentutorials.javatutorials.method;

public class ReturnDemo4 {

    public static String[] getMembers() {
    	String[] members = { "최진혁", "최유빈", "한이람" };
		return members;
	}

	public static void main(String[] args) {
		String[] members = getMembers();
	}

}

 

댓글

댓글 본문
작성자
비밀번호
  1. 엘비
    20191114 완료
  2. 이재우
  3. 다야
    06/11/19 완료
  4. Jack
    2019/10/31 수강완료
  5. 진방
    19/10/30 완료
  6. gwonran
    19/10/20 복습완료
  7. 허공
    감사합니다!
  8. 워나
    2019/10/12 완료
  9. 2019-09-23 수강완료
  10. PassionOfStudy
    190916 Note (추석연휴 끝나고 다시 시작)
    * 메소드(Method)
    메소드는 함수라고도 하며 코드를 재사용할 수 있게 해준다.
  11. 홍주호
    20190904 완료
  12. 섭이
    190901
  13. doevery
    수강완료
  14. 다나가
    190821 - 학습 완료!!!
  15. 코딩을시작
    check!!!
  16. 김혜민
    메서드 생성 방법, 호출(사용)방법 숙지하세요.

    리턴타입 메서드명 (매개변수) {
    매서드가 할일 기술
    }
    리턴 타입은 기본 변수 및 사용자가 생성한 클래스 등으로 할 수 있습니다
    void면 리턴하지 않습니다.

    예)
    리턴 이 있는 메서드
    String getStudentName() -- 학생의 이름이 리턴되는 메서드의 경우

    리턴이 없는 메서드
    void sendSms() -- sms를 보내고 결과는 안받아도 되는 경우


    좋은 강의 감사합니다.
  17. 왕만두
    네 자바에서는 + 덧셈 연산시 연산자와 피연산자 중 하나라도 스트링이면 스트링으로 변환되어 연산되어집니다

    숫자+숫자 = 숫자
    문자열 + 숫자 = 문자열
    숫자 + 문자열 = 문자열
    문자열 + 문자열 = 문자열

    원래 StringBuilder 클래스 이용하는데 비교적 간단한 연산을 직관적이고 빠르게 하기 위해서 사용합니다
    대화보기
    • 김만기
      i 가 int 형으로 선언 되었는데 어떻게 String 변수인 output에 연산이 되는거죠?

      output += i 는 output = output+i 라고 하셨는데
      문자열 = 문자열 + 정수 인데 그럼 i 는 문자열로 자동으로 형변환이 되는건가요?
    • 행인
      string 타입으로 선언한 뒤 i += 하면 덧셈이 되는게 아니라
      그 숫자가 그대로 뒤에 붙습니다.
      예를들어 i 가 3, 4, 5, 6 순으로 들어온다면
      ouput = 1 + 2 + 3 + 4 = 10 이 되는게 아니라
      ouput = 1234 가 되는거죠
      아마 굳이 스트링으로 한 것은
      각각 값이 뭐가 들어왔나 확인용이 아닐까 싶습니다.
      대화보기
      • 그런데 numbering 메소드에서 output이 String 타입인데 int인 i의 += 연산을 해도 output 객체의 숫자가 증가 할 수있나요?

        output과 return value를 String으로 두신 이유가 있을까요?
      • Hyeon-Jun Ha
        완료
      • 진한
        잘봤습니다. 휴 ~~
      • 얼그레이
        return과 다양한 데이터 타입이 지정된 메소드를 만드는 게 어렵네요! 190706 완료!
      • 무슨소리신지
        Return 나중에 다시보기
      • 6/17시작
        6/20 완료
        점점 어려워 지네요. 화이팅!
      • 자바공부
        리턴은 말 그대로 그 자리에서 다시 돌아간다는 의미입니다.
        어디로? 메소드를 호출한 곳으로요
        리턴뒤에 메소드 리턴타입이 있으면(string, int) 그 값을 반환하는 것이구요
        대화보기
        • 에듀케이티드키드
          ㅠㅠ 어렵다
        • 조혜령
          동영상 5번에서 왜 구지 output을 String으로 쓰나요?
        • 지미츄
          감사합니다.
        • 김동현
          감사합니다
        • silver94
          강의 잘들엇지만 다시 들으러 갑니다
        • hayoon
          맨 마지막 예제 코드 10번 줄은 다음과 같이 바꾸시면 이해가 쉽네요.

          public static void main(String[] args) {
          String[] members = getMembers();
          for ( int i = 0 ; i < members.length ; i++ ) {
          System.out.println(members[i]);
          }
          }
        • lolands01
          Method가 함수와 같은 의미라는것을 깨달았습니다, 그동안 별개의 개념인줄 알았거든요. 정말 감사합니다 ㅠ
        • 가쥬앙
          감사합니당
        • YukiHo
          감사합니다~
        • 라또마니
          아~ 메소드부터 어려워지고 있네요~ 특히 return 을 다시 들어봐야겠어요
        • C_K_PARK
          20190225
        • 닷미
          목소리도 좋으시고 딱적당한 속도로 이해하기 쉽게 잘설명해주시는것 같아요.
          착각인줄 몰라도 설명해주신것 다 이해한것 같아요. 이해한게 맞겠죠? ㅋㅋ
          동영상 보다가 정지해두고 설명에 없던 내용을 응용해서 깨작깨작해보며 스스로 이해도를 높이고 있긴해요.
          아직은 백지에서 배운걸 코딩하기는 어렵지만 그래도 설명해주시는 한번 다 보고 혼자서 코딩할수 있도록 연습하겠습니다.
          정말 감사합니다. 왕추천입니다.
        • 호두
          20190215
        • 호두
          고맙습니다
        • j-graphy
          2019. 2. 2. 학습 완료
        • andrew
          말투도 예시도 딱 수준 보이네요 ㅎㅎ
          대화보기
          • 오닉스
            완전히 이해는 가지 않지만 일단 진도를 쭉 나가고 2회차때 다시 보러 오겠습니다
          • andrewjk
            입력값을 콤마로 구분하지 않고 배열로 여러개의 변수를 넣을 수 있습니다.

            public static void getMembers(String[] members){
            ...
            }

            이런식으로 인자값에 배열을 받는다라고 정의하시면 됩니다. 하지만 예를 들어 정수인 나이와 문자열인 이름을 입력하려면 서로 다른 타입의 변수는 한가지 타입의 배열에 담을 수 없습니다. 편법으로 정수인 나이를 문자열로 인식해서 넣을 수 있지만 매소드 안에서 계산이 이뤄진다면 문자열을 다시 정수로 바꿔주는 작업을 해줘야 하는데 이게 단순히 형변환으로 바뀌지 않습니다. 스캐너를 이용해서 바꾸는 방법과 또 다른 방법이 있지만 굳이 배열로 묶지않고 콤마로 분리해서 넣는게 훨씬 직관적이고 간단한 방법이라고 할 수 있습니다.
            대화보기
            • (질문)
              안녕하세요! 강의 정말 잘 듣고 있습니다!
              책과 함께 수강하고 있습니다.

              마지막 예제에 대해서(ReturnDemo4)
              243p 하단부에 작성되어 있는
              ` 메소드와 출력, 그리고 메소드와 배열의 관계는 굉장히 중요한 관계라고 할 수 있습니다.
              물론 return에 한정해서 설명했지만 메소드가 인자값을 주면 그것은 입력값이 됩니다. 그 입력값을
              배열로 받으면 여러 개의 인자를 콤마(,)로 구분해서 배치하지 않아도 하나의 값에 여러 개의 데이터를 담아서 메소드로 전달할 수도 있습니다. 이처럼 메소드를 배우니까 배열이라는 것에 대해 더 잘 이해할 수 있었습니다. 배열을 사용하는 이유가 여기서 또 한 번 드러나기 때문입니다.`
              이부분에 통째로 이해가 되지 않습니다............

              어떻게 콤마로 구분해서 배치하지 않고 입력값을 배열로 받을 수 있는지 모르겠습니다.

              좋은 강의 제공해 주셔서 정말 감사합니다:)
            • ㅇㅇ
              c 사용자함수 같은...
            • wnajsldkf@naver.com
              메소드. 어렵군요 다시 돌아오겠습니다.
            • 노답
              그렇게 예시달거면 예시달지마세요 ㅋㅋ
              좋은강의에 뭔짓거리
              대화보기
              • 진짜 재미도없고 감동도없고 실속도없고 기분만 나빠지는 예시임
                대화보기
                • 이민욱
                  댓글 처음 다는데 갑자기 어려워 졌네여 ㅎㅎ 열심히 반복하겠습니다 return부터 너무 어려워져서 ㅠㅠ
                버전 관리
                egoing
                현재 버전
                선택 버전
                graphittie 자세히 보기