PGR21.com
- PGR21 관련된 질문 및 건의는 [건의 게시판]을 이용바랍니다.
- (2013년 3월 이전) 오래된 질문글은 [이전 질문 게시판]에 있습니다.
통합 규정을 준수해 주십시오. (2015.12.25.)
Date 2021/05/02 20:31:04
Name ArcanumToss
Subject [질문] 파이썬 반올림 함수 round 질문입니다.
도무지 이해가 안 되네요.
파이썬의 round 함수는 어떤 원리로 저런 값이 나오는 건가요?

print(round(1.5)) # 가까운 짝수로? 2
print(round(3.5)) # 가까운 짝수로? 4
print(round(5.5)) # 가까운 짝수로? 6
print(round(7.5)) # 가까운 짝수로? 8
print(round(9.5)) # 가까운 짝수로? 10

print(round(0.5)) # 가까운 짝수로? 0
print(round(2.5)) # 가까운 짝수로? 2
print(round(4.5)) # 가까운 짝수로? 4
print(round(6.5)) # 가까운 짝수로? 6
print(round(8.5)) # 가까운 짝수로? 8
print()

print(round(1.55)) # 가까운 짝수로? 2
print(round(3.55)) # 가까운 짝수로? 4
print(round(5.55)) # 가까운 짝수로? 6
print(round(7.55)) # 가까운 짝수로? 8
print(round(9.55)) # 가까운 짝수로? 10

print(round(0.55)) # 가까운 홀수로? 1
print(round(2.55)) # 가까운 홀수로? 3
print(round(4.55)) # 가까운 홀수로? 5
print(round(6.55)) # 가까운 홀수로? 7
print(round(8.55)) # 가까운 홀수로? 9
print()

print(round(0.55, 1)) # 그럼 이건??? 0.6
print(round(1.55, 1)) # 그럼 이건??? 1.6
print(round(8.55, 1)) # 그럼 이건??? 8.6
print(round(9.55, 1)) # 그럼 이건??? 9.6
print()
print(round(2.55, 1)) # 그럼 이건??? 2.5
print(round(3.55, 1)) # 그럼 이건??? 3.5
print(round(4.55, 1)) # 그럼 이건??? 4.5
print(round(5.55, 1)) # 그럼 이건??? 5.5
print(round(6.55, 1)) # 그럼 이건??? 6.5
print(round(7.55, 1)) # 그럼 이건??? 7.5

통합규정 1.3 이용안내 인용

"Pgr은 '명문화된 삭제규정'이 반드시 필요하지 않은 분을 환영합니다.
법 없이도 사는 사람, 남에게 상처를 주지 않으면서 같이 이야기 나눌 수 있는 분이면 좋겠습니다."
클린코더
21/05/02 20:40
수정 아이콘
파이썬에 익숙하지 않아서 저도 궁금해서 찾아보니 파이썬은 Round half to even 이라는 반올림 처리법을 따르고 있네요. 한국말로는 오사오입 이라고 하는 듯 합니다. 추가로 나무위키에 따르면 파이썬 버전마다 다른것 같습니다
"Python에서는 Microsoft Excel처럼 round 함수로 반올림을 할 수 있다. 다만, Python2에서는 사사오입법을, Python3에서는 오사오입법을 쓴다는 게 특징." https://namu.wiki/w/%EB%B0%98%EC%98%AC%EB%A6%BC 참고하세요
ArcanumToss
21/05/02 20:58
수정 아이콘
제가 올린 소스를 보면...

소숫점 이하 첫째 자리까지의 수는 가까운 짝수로 되는데
소숫점 이하 둘째 자리까지의 수는 반(정수부가 홀수면)은 가까운 짝수, 나머지 반(정수부가 짝수면)은 가까운 홀수로 바뀌고 있습니다.

그런데 소숫점 이하 둘째 자리까지의 수를 소숫점 이하 첫째 자리까지만 표현하도록 하면 규칙성이 안 보입니다... ㅠ.ㅠ



골아프니 그냥 반올림 함수를 직접 만들어서 쓰는 게 나은 듯 하네요. 흐흐

import math

def round_ori(n, decimals=0):
expoN = n * 10 ** decimals
if abs(expoN) - abs(math.floor(expoN)) < 0.5:
return math.floor(expoN) / 10 ** decimals
return math.ceil(expoN) / 10 ** decimals
클린코더
21/05/02 21:03
수정 아이콘
두번쨰 파라미터가 반올림을 한후 소숫점 자릿수인것 같은데, 기본값이 0일 겁니다. 그래서 [print(round(0.55)) # 가까운 홀수로? 1] 의 경우엔 소숫점 자릿수가 0개로 반올림 하기때문에 0.55는 0.5 보다 커서 오사오입의 대상이 아니라 그냥 바로 1로 올림하는 거일거구요.
지구사랑
21/05/02 22:51
수정 아이콘
0.55 는 논할 것이 없고 (무조건 1로 올라가는 것이 맞죠), 0 과 1 의 중간값인 0.5 를 1 로 올리느냐 0 으로 내리느냐가 문제인데,
항상 올리는 것으로 처리하면 통계적으로 + 노이즈가 더해지는 효과가 생기므로, 한 번은 올리고 한 번은 내리는 것이 합리적일 수 있습니다.
다만 부동소숫점 표현은 유효 숫자 문제가 있으므로, 0.5 가 아닌 0.05 에 대해서 0.1 을 기준으로 round() 를 하면 위의 원칙이 잘 맞지 않게 됩니다.
(0.5 는 1/2 이지만 0.05 는 이진수로 딱 떨어지지 않죠.)
X.05 가 내부적으로 X.049999999 로 표현될지 X.0500000001 로 표현될지는 X 값에 따라 달라지니까요.
ArcanumToss
21/05/03 02:10
수정 아이콘
Rorschach님의 설명과 함께 보니 님의 말씀이 무슨 말씀인지 알겠습니다.
2진수로 처리하는 문제 때문에 발생하는 문제점이로군요.
이해했습니다.
고맙습니다.
Rorschach
21/05/03 01:27
수정 아이콘
그냥 파이썬이 숫자 처리를 그렇게 해서 생기는 문제인데, 숫자를 소수점 아래 16자리 넘게 표시해보면 실제 값이 딱 떨어지지가 않아요.
위에 지구사랑님 말씀처럼 0.5 기준의 반올림은 이진수로 딱 떨어지는데 소수점이 더 내려가면 5 단위가 정확히 잘 떨어지지 않습니다.
그래서 그냥 round(a*10)/10 으로 하면 제대로 값이 나옵니다.

for i in range(20):
a = (i+0.55)
print("%20.18f"%a, round(a,1), round(a*10)/10)

이거 한 번 실행시켜 보시면 숫자가 어떻게 나오길래 저렇게 값이 나오는지 확인하실 수 있으실거예요.
ArcanumToss
21/05/03 02:15
수정 아이콘
님의 설명 덕분에 파이썬의 수치 처리 방식에 대한 이해가 생겼습니다.

round(a*10)/10 <- 요게 def round_ori() 함수의 처리 부분의 방식과 딱 동일한 원리군요!!
고맙습니다.
ArcanumToss
21/05/03 02:57
수정 아이콘
결론은 함수를 새로 만들어서 써야 하는군요.
엑셀과 똑같이 작동하는 함수를 만들었습니다.
필요하면 쓰세요.
뭐라도 답례를 드리고 싶어서요. ^^;


from math import *

def round_ori(n, decimals=0):
expoN = n * 10 ** decimals
if abs(expoN) - abs(floor(expoN)) < 0.5:
if decimals <= 0:
return int(floor(expoN) / 10 ** decimals)
return floor(expoN) / 10 ** decimals
if decimals <= 0:
return int(ceil(expoN) / 10 ** decimals)
return ceil(expoN) / 10 ** decimals
ArcanumToss
21/05/03 03:57
수정 아이콘
print("2.55 -> %.1f" % 2.55) # 2.55 -> 2.5
print("2.55 -> %s" % round(2.55, 1)) # 2.55 -> 2.5

둘 다 똑같이 나오는 것도 문제네요.
round 함수를 안 쓰고 %.1f로 해 보려 했더니만... ㅠ.ㅠ
허긴 내부적으로 처리하는 방식이 같을테니 그럴 수밖에 없겠네요.
솁첸코
21/05/03 01:37
수정 아이콘
(수정됨) https://docs.python.org/ko/3/library/functions.html#round
https://docs.python.org/ko/3/tutorial/floatingpoint.html
공식 문서에서도 주의해야될 부분으로 나오네요, 제가 쓰기보다 공식 문서 설명이 좋은 것 같아 링크 가져옵니다.

첨언하자면 파이썬에서 숫자를 좀 편하게 다루는 편이다보니 간과하기 쉬운데 부동소수점 표현법에 대한 이해도는 꽤 중요하다고 생각하는 편이라 한번 기회되시면 위 문서정도라도 정독해보시는걸 추천드립니다 이미 아시는 내용을 연관시키지 못하신걸수도 있지만...
ArcanumToss
21/05/03 02:53
수정 아이콘
그러고 보니 공식 문서에 주의하라고 나오네요.
파이썬... 묘하네요. 흐흐
21/05/03 06:58
수정 아이콘
스택오버플로우 찾아보니 decimal 모듈 쓰는게 제일 좋다고 하네요.

from decimal import Decimal, ROUND_UP

Decimal(str(16.2)).quantize(Decimal('.01'), rounding=ROUND_UP)
=
16.20

https://stackoverflow.com/questions/56820/round-doesnt-seem-to-be-rounding-properly
ArcanumToss
21/05/03 16:24
수정 아이콘
크크크 파이썬의 반올림 문제 때문에 다들 골머리가 아팠었나보네요.
다양한 방법이 올라와 있는 걸 보니...
round(16.2, -1)처럼 1의 자리에서 반올림하는 게 안 되는 걸 빼면 아주 좋네요.
고맙습니다. ^^
목록 삭게로! 맨위로
번호 제목 이름 날짜 조회
154841 [질문] 아이패드 에어3에서 프로5세대(12.9인치)로 갈아타도 될까요? [12] 맛있는새우6772 21/05/03 6772
154840 [질문] 간헐적 단식중입니다. 저탄고지를 같이 해볼까 해서 질문 좀 남겨요 [11] 바알키리7277 21/05/03 7277
154839 [질문] 테일즈 오브 베르세리아 재미있나요? [9] 아타락시아19610 21/05/03 9610
154838 [질문] 미국이 강대국이 된 원인을 분석해놓은 글이 있을까요? [3] 아웅이6363 21/05/03 6363
154837 [질문] 메타버스란게 실체가 있나요? [20] 그말싫9626 21/05/03 9626
154836 [질문] 자동차 보험료 할증이 생각보다 많이 되네요.. [12] 해먹8927 21/05/03 8927
154835 [질문] 예전에 본 글인데 기억이 나지 않아 질문게시판에 올립니다. [2] aMiCuS7490 21/05/03 7490
154834 [질문] 차 사고 문의드립니다. [3] TranceDJ8330 21/05/03 8330
154833 [질문] 인감증명서 사본 효력이 있나요? [10] 피쳐15973 21/05/03 15973
154832 [질문] 우울증 치료 혹은 약에대해 잘 아시는분 있으실까요? [5] Secundo9588 21/05/03 9588
154831 [질문] 월세 세입자가 중간에 조기 퇴거한다고 하면, 복비 정도 물어주면 되나요? [10] LG의심장박용택11259 21/05/03 11259
154830 [질문] 안드로이드 오토 무선 지원? [3] 서울7136 21/05/03 7136
154829 [질문] 마블 캐릭터 저작권 사용은 어디에 요청하는지 아시는 분 계시나요? [2] 넙이아니5643 21/05/03 5643
154828 [질문] 구형 컴퓨터 듀얼모니터용 그래픽 카드 구매문의드립니다. [3] APONO6450 21/05/03 6450
154827 [질문] 외국인 환자가 특별히 많은 치과도 있나요? [1] CastorPollux10132 21/05/03 10132
154826 [질문] 어린이날 선물로 살만한 드론 추천 부탁드립니다. [3] banelingMD8188 21/05/03 8188
154825 [질문] 바지를 좀 내려입는데.. 맞는 브랜드 잇나요? [3] 살다보니별일이6670 21/05/03 6670
154824 [질문] 편한 신발 [3] 당근병아리6062 21/05/03 6062
154823 [질문] 엔진오일 경고등이 자꾸 들어옵니다. [5] LeeDongGook7190 21/05/03 7190
154822 [질문] 제주도 렌터카 질문입니다. [6] 달달합니다7701 21/05/03 7701
154821 [삭제예정] 컴퓨터 케이스의 잔기스 제거하는법 [4] 틀림과 다름10681 21/05/02 10681
154820 [질문] 도로에서 이런 경우 박아버리면 누가 더 잘못인가요? [24] 삭제됨7633 21/05/02 7633
154819 [삭제예정] 30초반 회사원 세후급여 6천가까이인데 대출이 하나도 안되는데.. 진짜 방법이 없는걸까요? [26] 삭제됨9604 21/05/02 9604
목록 이전 다음
댓글

+ : 최근 1시간내에 달린 댓글
+ : 최근 2시간내에 달린 댓글
맨 위로