Java

유효범위

유효범위

변수와 메소드 같은 것들을 사용할 수 있는 것은 이름이 있기 때문이다. 아래 코드에서 left는 변수의 이름이고, sum은 메소드의 이름이다.

int left;
public void sum(){}

프로그램이 커지면 여러 가지 이유로 이름이 충돌하게 된다. 이를 해결하기 위해서 고안된 것이 유효범위라는 개념이다. 흔히 스코프(Scope)라고도 부른다.

유효범위는 암시적인 기능이라고 할 수 있기 때문에 자연스럽게 알게 되는 부분이다. 또 이해하는 것도 그렇게 어렵지 않기 때문에 긴장하지 말고 편안한 마음으로 나머지 부분을 읽어보자. 어찌 보면 유효범위의 출현배경을 통해서 부품으로서의 로직이라는 가치가 얼마나 중요한 것인가를 환기하는 것이 더욱 중요할 수도 있다.

출현배경

메소드, 클래스와 같은 개념들이 등장한 배경은 프로그램을 만드는 데 사용하는 코드의 양이 기하급수적으로 증가하면서 직면하게 되는 막장을 극복하기 위한 것이었다. 거대해진 코드를 효율적으로 제어하지 못한다면 웅장한 소프트웨어를 만드는 것은 점점 불가능한 일이 될 것이다. 유효범위라는 것도 그러한 맥락에서 등장한 개념이다. 하지만 유효범위는 메소드나 클래스처럼 특별한 문법적인 규칙을 가지고 있는 것은 아니다. 오히려 메소드나 클래스 안에 포함되어서 이러한 기능들의 부품으로서의 가치를 높여주는 역할을 한다고 할 수 있다.

아래의 코드를 보자. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo {

    static void a() {
		int i = 0;
	}

	public static void main(String[] args) {
		for (int i = 0; i < 5; i++) {
			a();
			System.out.println(i);
		}
	}

}

10행에는 변수 i의 값을 기준으로 동작하는 반복문이 있다. 11행에는 메소드 a를 호출하고 있는데 메소드 a의 내부에는 변수 i의 값이 0으로 지정되고 있다. 위의 코드를 실행시킨 결과는 아래와 같다.

0
1
2
3
4

그런데 만약 메소드 a가 실행될 때 메소드 내부의 변수 i의 값이 반복문의 변수 i의 값을 덮어쓰게 된다면 어떻게 될까? 반복문이 호출될 때마다 변수 i의 값이 0이 되기 때문에 이 반복문은 무한 반복에 빠지게 된다. 이런 상황을 해결하기 위해서는 메소드 a의 내부변수 i의 이름이나 반복문의 변수 i의 이름을 다르게 로직을 고쳐야 할 것이다.

만약 로직이 매우 복잡하거나, 메소드 a가 타인이 만든 것을 사용하는 것이라면 이것은 쉽지 않은 일이 된다. 이러한 문제는 부품으로서의 가치를 저하시킨다. 부품이란 조작 방법만 알면 내부의 동작 원리를 모르고도 사용할 수 있어야 한다. 또한, 부품 내부의 상태로 인해서 그 부품을 사용하는 외부의 동작 방법에 영향을 준다면 이 또한 좋은 부품이라고 할 수 없을 것이다.

실행결과를 보면 알겠지만, 내부 변수의 값이 그 외부에 영향을 미치지 않는다는 것을 알 수 있다. 처음 언어를 배우는 입장에서는 이것을 그러려니 하기 쉽겠지만 그렇지 않다. 이러한 동작방법은 수 많은 시행착오를 통해서 조율된 결과라고 할 수 있다. 과거의 프로그래밍 언어는 메소드 내에서의 변수가 외부의 변수에도 영향을 미쳤기 때문에 변수나 메소드의 이름을 사무실 칠판에 적어가면서 코딩을 해야 했던 시절도 있었다. 또는 변수명을 길게 하도록 권장하거나, 심지어 변수명에 프로그래머의 이름을 적는 경우도 있었다!

이런 문제를 해결하기 위해서 다양한 시도들이 있었는데 그 노력의 결과 중의 하나가 유효범위라고 할 수 있다.

다양한 유효범위들

디렉터리를 생각하면 쉬울 것 같다. 처음에는 파일이 있었다. 그런데 파일이 많아지면서 파일의 이름이 충돌하기 시작한다. 파일의 사용자들은 궁여지책으로 파일의 이름에 날짜나 부서 혹은 이름을 적어서 충돌을 피했을 것이다. 이러한 문제로 인한 절망이 충분히 성숙했을 때 운영체제의 개발자들은 이를 해결하기 위한 방법에 대해서 고민을 하게 되었을 것이다. 그래서 고안된 것이 디렉터리라고 할 수 있다. 디렉터리는 파일을 그룹핑해서 그룹별로 파일을 격리한다. 디렉터리 내에서는 파일명이 중복되면 안 되지만 디렉터리 밖의 파일명과는 중복이 돼도 문제 없다. 덕분에 마음 놓고 다른 사람이 만든 파일이 담긴 디렉터리를 자신의 디렉터리로 가져올 수 있게 되었다.

위의 예제를 살펴보자. 6행에서 변수 i를 아래와 같이 선언했다.

static void a(){
	int i = 5;
}

이것은 변수 i가 메소드 a에 소속된 변수라는 의미다. 따라서 이 변수의 값을 어떻게 바꿔도 이 변수의 밖에는 영향을 주지 않는다.

하지만 코드를 아래와 같이 변경한다면 무한반복이 일어날 것이다. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo2 {
    static int i;
	
	static void a() {
		i = 0;
	}

	public static void main(String[] args) {
		for (i = 0; i < 5; i++) {
			a();
			System.out.println(i);
		}
	}

}

변수 i가 4행에서 선언되고 있다. 4행의 변수 i는 위치적으로 어떠한 메소드의 소속도 아니다. 클래스 ScopeDemo2의 직접적인 소속인 클래스 변수다. 클래스 소속의 변수가 되면 모든 메소드에서 접근할 수 있게 된다. 그래서 7행의 변수 i는 클래스 맴버인 4행의 변수 i를 의미하게 된다. 마찬가지로 11행의 for문 안에 변수 i도 4행의 변수 i를 의미하게 된다. 다시 말해서 메소드 a의 변수 i와 for문의 변수 i가 동시에 클래스 변수 i를 사용하게 된다는 의미다. 그래서 반복문을 통해서 변수 i의 값을 아무리 바꿔도 메소드 a에 의해서 클래스 변수 i의 값이 0이 되기 때문에 반복문이 멈추지 않게 되는 것이다.

만약 위의 코드를 아래와 같이 바꾸면 위의 문제가 사라질 것이다. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo3 {
    static int i;
	
	static void a() {
		int i = 0;
	}

	public static void main(String[] args) {
		for (i = 0; i < 5; i++) {
			a();
			System.out.println(i);
		}
	}

}

바뀐 부분은 아래와 같다.

static void a(){
	int i = 0;
}

우선 메소드만 놓고 봤을 때 메소드 안에서 선언한 변수는 그 메소드가 실행될 때 만들어지고, 그 메소드가 종료되면 삭제된다. 만약 클래스 아래의 변수와 메소드 아래의 변수가 같은 이름을 가지고 있다면 메소드 아래의 변수가 우선하게 된다. 메소드 내의 변수가 존재하지 않을 때 클래스 아래의 변수를 사용하게 되는 것이다.

즉 클래스 아래에서 선언된 변수는 클래스 전역에 영향을 미치지만 메소드 내에서 선언된 변수는 클래스 아래에서 선언된 변수보다 우선순위가 높다고 할 수 있다. 지역적인 것이 전역적인 것보다 우선순위가 높다는 원칙은 특수한 것이 전체적인 것보다 우선순위가 높다는 의미로도 해석할 수 있는데 이러한 원리는 공학 전반에서 적용되는 원칙이기 때문에 교양으로서도 유익하다. 전역적으로 기본값을 설정하고, 필요에 따라서 지역 값을 다르게 사용하는 것이 더 효율적이기 때문에 이러한 원칙이 사용된다. 클래스 전역에서 접근 할 수 있는 변수를 전역변수, 메소드 내에서만 접근 할 수 있는 변수를 지역변수라고 한다. 아래 코드는 지역변수가 메소드 내에서만 접근이 가능함을 보여준다. 주석 처리된 9번라인의 주석을 제거하면 오류가 발생할 것이다. title은 메소드 a에서만 유효하기 때문이다. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo4 {
    static void a(){
    	String title = "coding everybody";
	}
	public static void main(String[] args) {
		a();
		//System.out.println(title);
	}

}

반복문에서 정의한 변수도 반복문 밖에서는 유효하지 않다. 주석 처리된 8행의 주석을 제거하면 에러가 발생한다. 반복문에서 선언된 변수 i는 반복문 밖에서는 유효하지 않기 때문이다. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo5 {
    public static void main(String[] args) {
		for (int i = 0; i < 5; i++) {
			System.out.println(i);
		}
		// System.out.println(i);
	}

}

조금 복잡한 예를 보자. (실행)

package org.opentutorials.javatutorials.scope;

public class ScopeDemo6 {
    static int i = 5;

	static void a() {
		int i = 10;
		b();
	}

	static void b() {
		System.out.println(i);
	}

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

}

결과는 5다. 위의 예제는 메소드 a가 메소드 b를 호출하고 있는데 메소드 b에는 변수 i의 값이 존재하지 않는다. 이 상태에서 메소드 a를 호출하면 메소드 b에서 System.out.println(i)를 했을 때 클래스 변수가 사용될까? 메소드 b를 호출한 메소드 a의 지역 변수 i가 사용될까? 클래스 변수를 사용한다. 메소드 내(b)에서 지역변수가 존재하지 않는다면 그 메소드가 소속된 클래스의 전역변수를 사용하게 된다.

이러한 방식을 정적 스코프(static scope) 혹은 렉시컬 스코프(lexical scope)라고도 부른다. 즉 사용되는 시점에서의 유효범위(메소드 a의 i)를 사용하는 것이 아니라 정의된 시점에서의 유효범위(i = 5)를 사용하는 것이다.

동적 스코프라는 것도 있다. 만약 메소드 b의 결과가 10이라면 메소드 b는 메소드 a의 유효범위에 소속된 것이라고 할 수 있다. 하지만 자바는 동적 스코프를 채택하지 않고 있다. 대부분의 현대적인 언어들이 정적 스코프 방식을 선택하고 있다.

인스턴스의 유효범위

지금까지는 클래스 중심으로 유효범위를 알아봤다. 인스턴스에서의 유효범위도 클래스와 거의 동일하지만 결정적인 차이점은 this에 있다고 할 수 있다. 아래 예제를 보자.  (실행)

package org.opentutorials.javatutorials.scope;

class C {
    int v = 10;

	void m() {
		System.out.println(v);
	}
}

public class ScopeDemo7 {

	public static void main(String[] args) {
		C c1 = new C();
		c1.m();
	}

}

결과는 10이다.

아래와 같이 코드를 변경해보자. (실행)

package org.opentutorials.javatutorials.scope;

class C2 {
    int v = 10;

	void m() {
		int v = 20;
		System.out.println(v);
	}
}

public class ScopeDemo8 {

	public static void main(String[] args) {
		C2 c1 = new C2();
		c1.m();
	}

}

7행이 추가되었다. 결과는 20이다. 즉 메소드 안에서 선언된 변수 v가 지역 변수가 되면서 인스턴스 전역에서 유효한 인스턴스 변수 v의 값보다 우선순위가 높아지면서 20이 출력된 것이다.

이런 상황에서 메소드 m에서 인스턴스 변수 v에 접근하려면 어떻게 해야할까? this를 사용하면 된다. 아래 코드를 보자. (실행)

package org.opentutorials.javatutorials.scope;

class C3 {
    int v = 10;

	void m() {
		int v = 20;
		System.out.println(this.v);
	}
}

public class ScopeDemo9 {

	public static void main(String[] args) {
		C3 c1 = new C3();
		c1.m();
	}

}

7행이 this.v로 바뀌었다. 그 결과 메소드 m 안에서 인스턴스 변수 v를 사용할 수 있게 되었다. this는 인스턴스 자신을 의미하는 키워드라고 할 수 있다.

교훈

이렇게해서 유효범위를 알아봤다. 유효범위란 변수를 전역변수, 지역변수 나눠서 좀 더 관리하기 편리하도록 한 것이다. 객체라는 개념이 존재하지 않는 절차지향 프로그래밍에서는 모든 메소드에서 접근이 가능한 변수의 사용을 죄악시하는 경향이 있다. 전역적인 사용의 효용이 분명한 데이터에 한해서 제한적으로 전역변수를 사용하도록 하고 있는 것이다. 객체지향 프로그래밍은 바로 이런 문제를 극복하기 위한 노력이라도고 볼 수 있다. 즉 연관된 변수와 메소드를 그룹핑 할 수 있도록 함으로서 좀 더 마음놓고 객체 안에서 전역변수를 사용할 수 있도록 한 것이다. 전역변수는 더 이상 죄악시할 대상이 아닌 것이 된다. 이렇듯 도구와 사람의 마음은 밀접한 연관이 있다.

부품의 관점에서도 생각해볼 수 있다. 어떤 메소드가 전역변수를 사용하고 있다는 것은 그 메소드는 그 전역변수에 의존한다는 의미다. 전역변수에 의존한다는 것은 이 메소드가 다른 완제품의 부품으로서 사용될 수 없다는 의미다. 객체지향 덕분에 좀 더 안심하고 전역변수를 사용하게 되었지만, 객체도 크기가 커지면 관리의 이슈가 생겨난다. 객체지향 프로그래밍에서도 가급적이면 전역변수의 사용을 자제하는 것이 좋고, 동시에 단일 객체가 너무 비대해지지 않도록 적절하게 규모를 쪼개는 것도 중요하다.

댓글

댓글 본문
작성자
비밀번호
  1. this를 왜쓰냐구? 알려줄라고!
    만약 클래스 아래의 변수와 메소드 아래의 변수가 같은 이름을 가지고 있다면 메소드 아래의 변수가 우선하게 된다. 메소드 내의 변수가 존재하지 않을 때 클래스 아래의 변수를 사용하게 되는 것이다.

    즉 클래스 아래에서 선언된 변수는 클래스 전역에 영향을 미치지만 메소드 내에서 선언된 변수는 클래스 아래에서 선언된 변수보다 우선순위가 높다고 할 수 있다. 지역적인 것이 전역적인 것보다 우선순위가 높다는 원칙은 특수한 것이 전체적인 것보다 우선순위가 높다는 의미로도 해석할 수 있는데 이러한 원리는 공학 전반에서 적용되는 원칙이기 때문에 교양으로서도 유익하다. 전역적으로 기본값을 설정하고, 필요에 따라서 지역 값을 다르게 사용하는 것이 더 효율적이기 때문에 이러한 원칙이 사용된다

    강의 내용에 아주 자세히 나와 있네요. 즉, 메소드안에 동일한 이름의 변수가 없으니 자연스레 클래스 변수로 가게 되는 겁니다
    인스턴스의 유효범위를 다시 보시는걸 추천하고 싶네요
    대화보기
    • 역지사지
      네티켓은 지켜주시죠. 그리고 강의 순서는 올바르다고 생각합니다. 전에 올라온 강의 내용만 잘 기억하고 들으면 "아 이게 그거구나!" 라는 강렬한 인상을 심어줘서 좀 더 기억에 많이 남게 될 수도 있는건데 더러운 말 써가며 비꼬면 강의를 준비한 입장에서는 얼마나 슬플고 힘들겠나요. 진짜로 순서를 바꾸길를 원하신다면 정중한 태도로 이메일을 써도 모자랄판에 자신이 욕을 받는 입장에서 이런 댓글을 보고 어떻게 해줄까요? 말고도 하고 싶은 말은 많지만 여기서 끊고 댓글을 쓰기보다 먼저 강의를 한 번 훑어보고 이해가 안가면 댓글을 달아주세요 토픽 목록에 대목도 떡하니 나와있잖아요
      대화보기
      • 자바보단c#
        ㅇㅈ
        대화보기
        • 씨발
          강의 순서를 바꾸셔야할거같네요
          전역변수 지역변수도 알려주지도 않고
          이전강의예제에서 막 쓰고 전 강의들 다보면서 이게 뭔 상황이지? 이런생각떄문에 아 난 안되는건가 포기할려고하다가 그냥 다음강의 듣자 해서 봤는데
          이제야와서 알려주니까 정말 어이가없네요 ㅋㅋ
          아니면 강의하는 도중에 이건 나중에배우니까 신경쓰지말라 이런말도안하고 ㅋㅋ
        • ubiquitous4g
          네, 할당됨.
          대화보기
          • 5월 2일 유효범위! 이해가 쏚쏚!
          • 열정하나
            알찬 강의였습니다.
            다시 한번 복습해야겠어요
            감사합니다~

            ㅎ.ㅎ
          • 박현모
            감사합니다~ 잘 봤습니다~
          • JeongHo Park
            잘 봤습니다~~~~
          • 김세창
            너무 잘 설명해주셔서 감사합니다!!
            이해가 쏙쏙됫네욤!!^^
          • 앞 예제 인데요

            class Calculator{
            int left, right;

            public void setOprands(int left, int right){
            this.left = left;
            this.right = right;
            }

            public void sum(){
            System.out.println(this.left+this.right);
            }

            public void avg(){
            System.out.println((this.left+this.right)/2);
            }
            }

            에서 sum, avg메소드에 변수를 this.left와 this.right 를 사용하지않고
            left와 right를 사용해도 제대로 값이 출력되는데요. 그 이유가 setoprands메소드에서
            this.left = left;
            this.right = right;
            를 통해 setoprands 메소드의 인자값을 전역변수인 left와 right에 대입하였기때문이고
            sum, avg메소드가 전역변수 left와 right를 사용하기 때문인가요?
          • 초보
            앞 강의 CalculatorDemo1 의 int left,right;가 선언된 위치를 잘보시면 클래스의 메소드가 아니라 클래스 내에서 선언되어 있는걸 알 수 있습니다. 즉, 전역변수로 그 클래스의 모든 곳에서 사용가능한 변수라는겁니다. 그리고 static이 붙은 변수인 클래스 변수뿐만 아니라 인스턴스 변수 또한 전역변수입니다.
            그래서 클래스의 인스턴스 메소드에서 별다른 선언없이 사용가능한거구요. 그럼 왜 this.left를 사용했느냐, 그건 메소드의 매개변수로 설정된 변수 이름 또한 left,right이기 때문입니다. this를 사용하지 않고 그냥 left를 해버리면 전역변수인 left,right가 아니라 가장 가까운 메소드의 매개변수인 public void abc(int left, int right)가 지정이 되기 때문입니다. 마지막으로 static과 인스턴스의 차이를 말씀드리자면 쉽게 말해 값을 공유 하는지의 차이입니다. 자세한 내용은 구글 검색하시면 너무나 상세히 나와있습니다.
            대화보기
            • 배성재
              재밌네요ㅎㅎ
            • 꿈*은이루어진다
              완료!!
            • 하면된다하자
              완료
            • popinbompin
              완료
            • selina
              완료
            • Younghun Liam Youn
              감사합니다!
            • GoldPenguin
              감사합니다.
            • 완료.!
            • 심영
              오늘은 좀 쉬웠습니다

              잘보고가요
            • k17811
              현재 따로 학원으 다니면서 강의도 듣고 있지만
              학원의 강사보다도 여기 강좌가 훨씬 이해가 잘되네요
              정말 초보자의 눈높이에 맞게끔 강의를 잘하시는것 같습니다.

              감사합니다^^
            • yoon88
              완료
            • Nam.ki
              3번까지 완료
            • yoon88
              2번 영상까지 완료
              3번 부터 시청
            • 우성
              단순합니다.

              static void b() {
              System.out.println(i);
              }
              이것은 static 메소드입니다. 그래서 인스턴트 멤버를 접근할 수 없기때문에 오류가 나는것입니다.

              앞강의 다시보시면 아실겁니다
              대화보기
              • 흠.. 이부분에 대해서는 제가 알고있어서 답변을 드린다기보다는 님의 글을 보고 생각(유츄?)해본 결과를 말해드리려고합니다.
                1.답글해주신 부분에 static키워드를 지우면 왜 에러가뜨냐?
                2.다른 예제에서는 static을 안써도 잘동작했다

                이것을 토대로 제가 생각해본결과 차이점은 이거같습니다. 바로 객체를 만들었냐 안만들었냐의 차이죠. 답글해주신 예제에서는 객체가 생성이 안되있죠. 그러므로 당연히 클래스밑에쓰는 변수는 static의 기능을 하게되고
                님이 언급하신 예제에서는 객체가 생성이되있죠. 객체는 독립적입니다. 그래서 저기 int v=10; 이라고 선언된 부분은 객체 기준에서 보신다면 전역변수아닐까? 라고 생각할수도 있겠지만 프로그램 전체의 시야에서 본다면 해당 객체 안에서만쓸수있는 지역변수로 볼수 있기 때문이 아닐까 감히 생각해봅니다. 만약에 전역변수를 쓴다면 클래스 밑에다가 선언해야대겠죠??
                <제말이 맞을수도 틀릴수도 있지만 단지 제생각일뿐입니다. 확실한 답을 알려드리지 못해 죄송합니다..>
                대화보기
                • 티에리앙리
                  질문있습니다
                  님이 말씀해 주신대로 static 키워드가 붙지않은 지역변수는 선언된 괄호 안에서만 동작하게 되는 거라고 하셨는데

                  그렇다면 본 동영상강의의 3/4 : 다양한 유효범위들의
                  scopeDemo6.java 파일에서

                  package 이거는 저장되있는 위치니까 상관 없겠죠;

                  public class scopeDemo3 {
                  static int i = 5;
                  /* ★★ 얘를 static을 없애면 선언된 괄호 안에서만 동작하게 되는 거라고 하셨는데
                  그러면 윗윗행의 static을 없애도 적용이 되야되는데 오류가 뜨네요
                  이건 왜그런거죠????
                  반면에 4/4 강의에서 복습했던 CalculatorDemo1 의 5행에 있는 int left,right;는 앞에 static을 안써도
                  선언된 괄호 안에서 동작을 하네요
                  */
                  static void a() {
                  int i = 10; // 지역변수
                  b();
                  }
                  static void b() {
                  // int i = 30;
                  System.out.println(i);
                  }

                  public static void main(String[] args) {
                  //int i = 1; // 지역변수
                  a();
                  }
                  }
                  대화보기
                  • static 이라는 키워드는 흔히 '클래스 변수'라고 말하죠. 또한. 정적 변수라고도 불립니다. 즉 static 이란 키워드가 붙으면 이 정적 변수는 메모리상에서 항상 상주하게 있게되죠. 그래서 즉 전역 변수로서의 기능도 갖게되고 전역 변수의 기능을 갖게된다는것은 어떤 곳에서든 다쓸수 있다는겁니다. static 키워드가 붙지 않은 지역 변수는 선언된 괄호 안에서만 동작하게 되는 것이구요
                    대화보기
                    • NoobUnityStarter
                      언제나 자세하게 알려주셔서 감사합니다. 그리고 스스로 잘 이해했는지 지적받기 위해서, 한번 올려봅니다.

                      https://youtu.be......1Ef
                    • 미림_likelion
                      수강완료했습니다. 감사합니다.
                    • static의 의미를 잘 모르겠네요..
                      언제 사용되는 것이며, 사용하지 않았을 때와 차이점을 모르겠어요
                    • J_Project
                      감사합니다
                    • 유진
                      감사합니다.
                    • 김인섭
                      감사합니다.

                      일단 전진합니다 ㅎ;
                    • 옹옹
                      감사합니다
                    • 이원준
                      아시는 분이 많겠지만 혹시나 해서 올립니다. 강의 감사히 잘 보고 있습니다.. 이클립스에서 전역변수는 이텔릭체(필기체 비스므리...)로 표시가 되네요. 프로그램이 길어지면 서체로 확인할수 있게 해놓은것 같습니다 :)
                    • 리브
                      this 가 이해가 안갔었는데

                      이렇게 다시 짚어주셔서 감사합니다
                    • 박수빈
                      this 부분이 개인적으로 이해가 잘안갔는데 강의로 이해가 되었습니다. 감사합니다.
                    • 김선용
                      항상 감사드립니다.
                    • Lee Jung Ah
                      이번 강의로 굉장히 이해가 잘 되서 앞선 강의에서 좀 이해가 안되었던 부분도 해결이 되었어요 ㅎㅎ
                      감사합니다~~
                    • 라떼
                      감사합니다!!
                    • 송선욱
                      클래스 인스턴스 강의에서 이해안됐던 this. 에 대한 개념이

                      드디어 여기에서 이해가 가네요!!

                      어서 끝까지 1회독을 후딱 해야겠어요~~~

                      꼼꼼함은 잠시 접어두고..
                    • jae seung
                      와..정말 설명 대단합니다!
                    • Joon Lee
                      와우..앞에서 이해 안됐던 부분이 이번 수업을 통해 다 이해됐습니다!
                    • Sadak
                      전 36세라는 늦은 나이에 프로그램 언어를 배워보고 싶어서 도전을 시작했습니다.
                      고졸의 학력에 외식쪽 서비스업만을 지금까지 하다보니 배경지식이 없어 힘드네요.
                      제가 이해한 부분이 맞는지 피드백을 해주시면 감사하겠습니다.^^

                      Q : a();라는 메서드를 호출했을 때 어떤 i의 값을 사용하게 될까?


                      1. main메소드 의 유효범위에 있는 i의 값 (X)

                      이유
                      - main 메서드 안에 int i = 1; 선언했을 때 값이 출력되지 않는다.

                      - a()의 i는 유효범위의 원칙에 따라 자기 자신의 메서드 안에서만 작동하기 때문에 b()의 메서드에

                      영향을 주지 못한다.


                      2. static void a() 메서드에 있는 i의 값 (X)

                      이유:
                      - a()의 int i = 10;은 출력되지 않는다.

                      - a()의 i는 유효범위의 원칙에 따라 자기 자신의 메서드 안에서만 작동하기 때문에 b()의 메서드에
                      영향을 주지 못한다.


                      3. b() 메서드에 있는 System.out.println(i); 에 있는 i의 값 (O)

                      이유:
                      - b(); int i = 30;은 출력된다.

                      - 만약에 b() 메서드안에 동일한 이름의 변수 i와 변수값으로 30이 할당된다면 유효범위의 원칙에 따라 자기자신의
                      변수값을 더 우선적으로 적용하므로 30인 값을 출력하게 된다.

                      - 하지만 예제에는 b() 메서드에 할당된 변수와 값이 존재하지 않기 때문에 정적 스코프의 원칙에 따라서
                      전역변수의 값인 int i=5;의 값을 출력하게 된다.

                      - 정적 스코프의 원칙은 지역변수가 가지고 있는 자기자신의 변수 값 또는 전역변수의 값에만 접근이 가능하다.

                      - 그러므로 b()의 지역변수 값과 ScopeDemo6의 전역변수 값이 존재하지 않다면 출력할 값이 존재하지 않기
                      때문에 에러가 발생한다.

                      - 만약에 동적 스코프의 원칙이 이 상황에 적용이 된다면 b()의 지역변수 값과 ScopeDemo6의 전역변수 값이
                      존재하지 않기 때문에 순차적으로 우선순위인 a() 메서드의 지역변수 값인 int i=10;의 값을 출력하게 된다.


                      이 예제를 정리하면 스코프의 기본 원칙에 따라서 main에 있는 int i = 1; 과 a() 메서드에 있는 int i = 10; 은
                      b() 메서드에 영향을 줄 수 없으며, 자바라는 언어는 정적 스코프의 원칙을 적용하고 있으므로 b() 메서드의 지역변수 값이
                      존재하지 않기 때문에 ScopeDemo6 클래스의 전역변수 값 5를 출력하게 되는 것이다.

                      아무런 개념이 없는 상태에서 개념을 잡을 수 있게 정말 큰 도움을 받고 있습니다.
                      지금은 제 상황이 좋지 않지만 꼭 제가 받은 이런 부분들을 꼭 보답하고 싶습니다.
                      다시 한번 감사합니다. ^^
                    • ㅎㅎㅎ
                      static 같은 경우에는 클레스 범위도 넘나들수있지만
                      static 없는경우 클레스 안의 메소드만 넘나들수 있는걸로 알고 있어요
                      즉 하나의 클레스 안에서는 전역변수가 된다는 거죠
                      대화보기
                      • 오미풍
                        네 번째 동영상에서 int v=10; 앞에 static이 없는데 어떻게 전역변수가 되는건가요?
                      • 쉬운설명 감사합니다!
                      • weaver
                        친절하고 체계적인 설명 감사드립니다~
                      버전 관리
                      egoing
                      현재 버전
                      선택 버전
                      graphittie 자세히 보기