Java

상수의 데이터 타입

상수

변수는 변하는 값을 의미한다. 그 대척점에 있는 것이 상수인데, 상수(常數, constant)란 변하지 않는 값을 의미한다. 아래의 코드 중에서 a는 변수이고, 1은 상수이다.

int a = 1;

변수 a는 대입 연산자(=)에 의해서 1이 되었다. 아래 예제를 보자.

1 = 3;

컴파일이 되지 않는다. 1은 3이 될 수 없기 때문이다. 1은 1이고, 3은 3이다. 고유한 값을 가지고 있고, 그 값을 변경 할 수 없는 데이터 타입을 상수라고 한다. 

상수와 데이터 타입

실수의 표현

앞 수업에서 변수를 만들 때 데이터 타입을 지정했었다. 그것은 변수가 메모리를 얼마나 사용할 것인가를 지정하는 의미를 갖는다. 그렇다면 변수에 저장되는 상수도 데이터 타입이 있을까? 물론 있다. 아래 코드는 오류를 발생한다. (실행)

int a = 2.2;

2.2는 실수다. 실수를 정수 타입의 변수 a에 저장하려고 했기 때문에 오류가 발생한 것이다. 이것은 변수와 똑같이 상수도 데이터 타입이 있다는 것이다. 그럼 위의 예에서 사용한 상수 2.2의 데이터 타입은 무엇일까? float일까? double일까?

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

float a = 2.2;

"Type mismatch: cannot convert from double to float"

위와 같은 에러가 발생한다. 즉 2.2는 float가 아니라는 뜻이다. 예제를 조금 바꿔보자. (실행)

double a = 2.2;

오류가 사라졌다. 자바에서 실수형 상수는 double의 데이터 타입이다. 그럼 float 형 변수에는 어떻게 값을 대입할 수 있나? 2.2가 float 형이라는 것을 분명하게 명시해주면 된다. (실행)

float a = 2.2F;

F는 이 기호 앞의 숫자가 float 데이터 타입이라는 것을 명시적으로 표현하는 방법이다.

정수의 표현

그럼 데이터 타입이 정수인 상수는 어떤 데이터 타입이 될까? int다. 아래 예제는 오류가 발생할 것이다. (실행)

int a = 2147483648;

int의 최댓값인 2147483647 보다 1 많기 때문이다. 코드를 고쳐보자. (실행)

long a = 2147483648;

"The literal 2147483648 of type int is out of range"

변수는 long 타입이지만 이 변수에 대입되는 상수가 여전히 int 타입이기 때문에 int로 표현할 수 있는 최대 숫자를 여전히 초과하고 있다. 아래와 같이 코드를 변경해보자. (실행)

long a = 2147483648L;

이제 오류가 해결되었다. 상수도 long 타입이 되었고, 그 상수를 담을 변수도 long 타입이 되었다. 그럼 int 보다 작은 데이터 타입인 short나 byte는 어떻게 해야 표현할 수 있을까? (실행)

byte a = 100;
short b = 200;

이번에는 오류가 발생하지 않는다. 자바는 byte와 short 타입에 대해서는 int 형을 허용하기 때문에 오류가 발생하지 않는다.

댓글

댓글 본문
작성자
비밀번호
  1. 변수의 데이터타입 : 기본형, 참조형 등
    상수의 데이터타입 : long형의 L, float형의 F 등
  2. ljn425
    상수에도 데이터타입이 있군요. 감사합니다.
  3. Ryan.L
    감사합니다.!
  4. 컴알못
    long c = 214748647(즉 int로 표현가능한 범위에 있는수)
    는 왜 에러가 안뜰까요?? 아직 다음 강의인 형변환을 안들은 상태인데 듣고 다시오겠습니다 핳ㅎㅎ감사합니다 좋은 자료들~
  5. 감사합니다!!
  6. ㅎㅎ
    쉽게 잘가르쳐주시네요 감사합니당ㅎㅎ
  7. 하하
    미투
  8. 휴식중
    저는 아무것도 이해가 않되네요.
    저부 다 외계인 말 같아요 ㅠㅠ
    하나도 모르겠어요 ㅠㅠㅠ
  9. JustStudy
    고맙습니다
  10. somnium
    Really thank you for nice lectures~~
  11. ㅅㅇ
    ㅅㅇ
    대화보기
    • yskioi
      감사합니다.
    • 방금 저도 실습 해 보았습니다.
      long a = 2147483648 은 에러가 발생하지만
      long a = 2147483647 하면 에러가 발생하지 않습니다.
      생각해보니, 상수의 기본 데이터 타입은 int이므로, int 범위 내에만 있으면 변수의 type과는 자동으로 호환 되는듯 합니다.
      대화보기
      • 213124
        쉽게말해서 상수는 INT 로 기본 인식이 되지만
        바이트나 숏트는 기본 인식된 INT 를 변수 선언 데이터 형식에 맞게 자동으로 바꿔주지만
        LONG 은 안바까주기때문에 직접 L 을 써서 바까줘야 한다 인거군요
      • 나도초보
        전 어제 시작했던 초보입니다
        제가 봤을때 자바에서는 사용자의 편리를 위하여 기본적으로 상수에 데이터타입이 int로 되어있어서
        long에 포함되는 수를 기입을 했음에도 불구하고 오류가 뜬다고 생각이되네요
        그래서 만일 long을 써야 하는 경우가 왔을 때 오류가 나지 않기 위해서 상수의 데이터타입을 L을 작성하면서
        변경해준다 는 말 같아요.
        대화보기
        • Devani0310
          감사합니다.
        • 오빠는다르다
          감사합니다!!!!!
        • 으에엑
          집합의 개념으로보시면 편하실듯 싶습니다
          집합으로보시면 바이트(byte)와 쇼트(short)는 인트(int)안에 포함돼있어 인트라고 적어도 무방한것 같고 3000000000의 경우는 int에는 포함이돼있지 않아 long이라는 타입을 명시해줘야하는것 같습니다
          추측이지만요^^
          대화보기
          • 초보중에
            궁금한것이잇습니다. 자바 오늘 처음배우는 초보인데요;;; 여기까지 강좌를 듣다가 한가지 의문점을 갖게됩니다 ``;

            동영상 강의중 2147483648 이 아닌 3000000000 이라고 해도

            long a = 3000000000 오류가 뜨나용 ?

            또한 byte b = 100; 당연히 byte 값안에 있으니 오류가 안나야되지 않나용 ``?
          • 원래 실수형 사용의 경우 소숫점이 표현됩니다
            그리고 모든 컴퓨터의 소숫점 이하 계산은 약간의 오차가 있으니 참고하시길
            대화보기
            • 아솔
              좋은 강의 감사드립니다.
              byte b = 100;가 허용된다는 것은,
              기본 상수 타입이 int이지만 뒤에 B 등을 붙이지 않아도 자동으로 byte 변수로 인식되어서
              1바이트 메모리만 쓴다는 것인가요?
            • Byunghawk Lee
              저는 상수라하길래 constant 형을 의미하는 줄 알아 조금 헷갈렸습니다.....
              대입되는 값이 상수긴 하지만 ...
              아무튼 잘 보고 넘어갑니다.
            • hotsipa
              감사합니다.
            • 현팡
              정주행중입니다
            • 허니버터
              잘보고 갑니다
            • elsa
              ㅋㅋㅋ short byte 처럼 편의 좀 봐주는김에 long 이나 float 도 편의좀 봐주지..
              괜히 뒤에 명시적으로 표현해서 좀 당황스럽네요 초보자 입장에선ㅋㅋ -_-+
            • 나도고수
              와 대박이네요
              상수에도 데이터타입이 존재한다는 걸 오늘 처음 알게되었습니다.
              설명도 실습을 통하여 무지 쉽게 해주셔서 이해하기 쉬웠습니다!
            • etree16@gmail.com
              나이스!
            • 김정민
              강의 잘 듣고 있습니다!
              다름이 아니라 몇가지 실험적인(?) 변수 설정을 해보고 있는데요
              예로 double a=2-3; a값을 빼면 -1.0 이라고 나와요. 원래 소수첫째까지 표현되는건가요?
              그리고 double a=2.2-3; 하면 -0.7999999999999998 로 쪼금 부족하게 되더라구요?
              변수에서 바로 계산값을 써넣으면 오류가 생기는 것인가요?
            • 노수한
              잘 봤습니다!
            • 도화지
              long타입보다 더 큰 데이터 타입이 있나요?
            • egoing
              float를 사용하는 이유는 메모리를 아껴쓰기 위해서라고 생각하시면 될 것 같습니다.
              대화보기
              • 궁그미
                그렇다면 1.2 2.2 3.3 모두 다 float형 변수에 대입하려면

                float a = 1.2F;
                float b = 2.2F;
                float c = 3.3F;

                라고 해줘야 하나요?

                그렇다면 왜 float형을 쓰나요? 용량을 줄이기 위해서 사용하는 건가요?
                그럼 0.000000000000000000000000...000001부터 아무튼 소수점이 있는 모든 실수들
                크던 작던 다 float형 변수에 대입하려면 뒤에 F를 붙여야 하나요?
              • 별콩이
                서적에는 이렇게 세세하게 설명이 안되어 있는데 지금까지 몰랐던 부분까지 알게 되네요~ 감사합니다.
                짱짱!!! c# 공부하게 되는데 비슷해서 계속 듣게 되네요~
              • 지나가는 인
                정수의 기본형은 4바이트 int
                실수의 기본형은 8바이트 double

                기본형이라는 말은 아무 표기 없이 숫자를 썼을 때 인식하는 형태라고 말할 수 있겠네요.

                큰 사이즈에 작은 것을 넣는 것은 허용해주나
                int에 byte, short 을 넣는 경우
                작은 사이즈에 큰 것을 넣는 것은 허용이 안됩니다.
                short에 int, long을 넣는 경우.

                정수는 사실 범위만 초과하지 않으면 허용 가능한데 그 이유가 32비트 컴퓨터에서 2바이트를 다루는 것과 4바이트를 다루는 것은 같은 노력이 필요하기 때문에 효율상 모두 다 4바이트로 처리합니다.
                그렇기에 byte a = 100 이 가능하게 됩니다. 다만 범위가 초과하면 문제를 일으키죠.

                여기서 실수의 경우는 조금 다릅니다.
                실수의 표기는 비트 단위로 첫 비트는 부호, 그 다음 몇 비트는 10을 몇 번 곱할지의 지수, 그 다음은 숫자를 소수로 표현했을 때 표기로 '부동 소숫점'이라는 표기 방식을 표현합니다.

                작은 것(float)에 큰 것(double)을 넣으면 이 실수가 표현 불가능한 경우가 많습니다.
                부가적으로 말씀드리면 0~1 사이만해도 무한개의 실수가 존재하는데 이것을 8바이트로 모두 표현할 수 없습니다. 왜냐하면 무한개를 유한한 데이터로 표현할 수 없으니까요. 그래서 실수를 표현할 때 그냥 꽤 촘촘히만 표현하자라는 생각을 했고 이러한 점 때문에 몇몇 계산에서는 컴퓨터가 틀린 값을 주기도 합니다.(다른 방식으로 보정하기는 합니다만)
                이 촘촘함의 정도는 0에 가까워질수록 촘촘해지고 0에서 멀어질 수 록 뜨문뜨문해지죠. 극단적인 예로 10000.0이 표현가능한 최대의 실수라면 9999.0은 표현이 불가능할 수도 있습니다. 왜냐하면 뜨문뜨문 그 값이 존재하기 때문이고 우리가 쓰는 10진법이 컴퓨터와 쓰는 2진법과 다르기 때문에 우리가 보기에는 당연히 존재할 것 같은 구간(예의 경우에는 정수 9999와 같은 구간인 9999.0)이 2진법에서는 존재하기 힘든 수일 수도 있습니다.
                다른 쉬운 예로
                1/3이라고 분수로 쓰면 정확하게 표기 가능한 이것을 십진법으로 쓰면 표기 불가능합니다. 0.333••• 나 3위에 점을 찍어서 계속된다로 쓰거나 다른 방식을 써야하죠.
                실수의 경우가 이와 비슷하기에 이미 기본적으로 정확도가 떨어진다고 기본적으로 생각하고 이 정확도는 정밀함이 높을수록 올라가니 변수의 크기가 높은 double이 float에 비해 같은 범위에서 더 정밀하게 표현하기에 float a = 2.2; 와 같은 구문이 손실 없이 들어가지는지 애매해집니다. 우리 눈에는 쉽지만 컴퓨터 눈에는 0.333인지 0.333333인지의 차이가 발생할 수 있다는거죠.
                쉽게 설명하려고 해봤지만 그다지 쉬워보이지 않네요.
                부동소수점 표기법을 한번 찾아보시면 이해가 더 잘가실거에요.
                대화보기
                • moby
                  Java에서 정수형 데이터 타입의 영문표기 허용범위(오류가 나기까지의 허용범위)는 int의 최하위형인 byte부터 기본형인 int까지, 그 위로는 나타내고자 하는 데이터 타입의 단어(예; long) 첫번째 문자(l)를 정수 뒤에 붙여주는 것이군요.

                  정수형
                  데이터 타입 메모리의 크기 표현 가능 범위
                  byte 1 byte -128 ~ 127
                  short 2 byte -32,768 ~ 32,767
                  int 4 byte -2,147,483,648~2,147,483,647
                  long 8 byte -9,223,372,036,854,775,808 ~ 9,223,372,036,854,775,807


                  반면에 실수형 데이터 타입같은 경우, 정수형의 기본인 int가 가지고있는 4바이트 속성이 아닌, 실수형의 기본인 double의 8바이트 속성이므로 영문표기의 첫번째 문자를 실수 뒤에 붙여주는 것이군요. 실수형의 4 바이트는 float이지만 애석하게도 기본형의 주도권이 8바이트인 double이 가지고 있기 때문에 4바이트와 8바이트를 넘나들기 위해서는 영문표기의 첫 문자를 입력해주어야 하는걸로 이해할 수 있겠군요.


                  실수형
                  float 4byte ±(1.40129846432481707e-45 ~ 3.40282346638528860e+38)
                  double 8byte
                  ±(4.94065645841246544e-324d ~ 1.79769313486231570e+308d)

                  4바이트와 8바이트의 갭이 이렇게 클줄이야. 이것에 대한 논리적인 해석이 가능할까요? 아니면 그저 자바의 특성이기 때문에 그저 egoing님의 말씀처럼 운명으로 받아들여야 하는것일까요 ㅎㅎㅎㅎ
                • jeyul
                  실수는 기본적으로 double형입니다.
                  double형은 크기가 8바이트인 반면에 float형의 크기는 그 절반인 4바이트입니다.
                  float형 변수에 그냥 실수값을 대입하면 짤리기 때문에 에러가 납니다.
                  따라서 float형 변수에 값을 대입할때는 실수값 뒤에 F를 붙여야 합니다.

                  double a = 1.1; // 1.1은 8바이트 크기의 double형 상수, dobule형 변수 a에 대입 가능
                  float b = 1.2f; // 1.2f는 4바이트 크기의 float형 상수, float형 변수 b에 대입 가능
                  float c = 1.3; // 1.3은 8바이트 크기의 double형 상수, float형 변수 c에 대입 불가능(에러!)
                  대화보기
                  • 이동주
                    쉽고 친절한 강의 정말 감사합니다ㅜㅜ!!
                  • 장이쁨
                    짱이에요 ㅎㅎㅎ이해 잘가고 공부도 잘되는거같아요 너무감사합니다.ㅇ
                  • 정범석
                    강좌 잘봤습니다~
                    고맙습니다~!~
                  • egoing
                    질문이 잘 이해가 안되네요 ^^
                    대화보기
                    • 김정은
                      강좌 너무 고맙습니다.
                      거의 항상 새벽에 강좌를 보고 있는데
                      정말 좋습니다.
                      복많이 받으세요.~ ^ ^
                    • Healthyryu
                      언제나 감사합니다. 이고잉님 ^^

                      그러네 여쭙고 싶은게 하나 있습니다. int 형이 byte나 short 보다 큰 형이라서 'byte, short' 에 대입하는 값 뒤에는 'b' 라던제 's' 를 붙일 필요가 없는데, 실수의 경우는 기본이 double 이며, 왜 float 형을 쓸 때 대입하는 값에 F 를 써야하나요?
                      기본이 double이면 당연히 float의 범위까지 포함되어서 굳이 뒤에 'F' 를 붙일 필요가 없는게 아닐까라는 의문이 들어서 질문을 드립니다.
                    • 박세현
                      잘보고 가요. 감사합니다~^^
                    • teaksoojjang
                      잘보고 갑니다^^
                    • egoing
                      int는 2147483647까지를 수용할 수 있습니다. 그런데 2147483648이 되면 int로는 수용 할 수 없기 때문에 오류가 발생합니다. 그렇기 때문에 2147483648을 long 형으로 변환하기 위해서 뒤에 L을 붙여준 것이죠.

                      궁금하신 내용은 다음 토픽인 '형변환'에서 이유가 나옵니다. 그 내용을 한번 보시고 궁금하시면 다시 한번 질문해주셔요~
                      대화보기
                      • egoing
                        거의 이해하신 것 같아요.
                        2.2는 기본적으로 double 입니다.
                        그런데 이것을 데이터 타입 double의 변수 a에 넣으려고 하니까 데이터 타입이 맞지 않는다는 오류가 발생하는 것이죠. 이 문제를 해결하기 위해서는 2.2를 float 형식으로 명시적으로 변환해줘야 합니다. 이 때 사용하는 키워드가 F 입니다.

                        이해가 되셨는지요?
                        대화보기
                        • 벼녕
                          그리고 또 궁금한 것은.. long a = 2147483648; 이 오류가 나는 이유가 변수는 long 타입이지만 상수가 여전히 int 타입이라서 그렇다고 하셨는데요. 2147483648은 int의 수용범위인 2147483647보다 1이 초과되는 것인데도 int타입이라고 할 수 있나요?? 아니면 모든 정수가 기본적으로 int라는 데이터타입을 갖는 건가요?
                        • 벼녕
                          float a = 2.2; 라고 하면 오류가 나지만 float a = 2.2F; 라고 하면 오류가 안 나는 게 이해가 잘 안 돼요!
                          전자에서 오류가 난 이유는 2.2라는 상수의 데이터타입과 변수의 데이터타입이 서로 맞질 않아서 오류가 난 것이잖아요? 즉 2.2는 float라는 데이터타입에 맞지 않는 성질을 가진 거 같은데.. 후자에서 2.2에 F를 하나 붙여준다고 해서 상수의 데이터타임과 변수의 데이터타입이 맞아지는 게 이해가 안 돼요 ㅠㅠ
                        버전 관리
                        egoing
                        현재 버전
                        선택 버전
                        graphittie 자세히 보기