++i 와 같은 선행증가에서 궁금한점

zelon의 이미지

오랜만에 글을 시작해보는군요.

우리과 후배가 이런 질문을 했습니다.

int k = 0;
printf("%d %d %d", ++k, ++k, ++k);

일 때 왜 3 2 1 이 찍혀요? 라구요 ;;

혹시나 해서 cout 으로 해도 마찬가지 결과를 내더군요. cout 은 operator<< 의 함수 호출로 오른쪽부터 불리는 것 같은데, printf() 의 경우에는 인자가 오른쪽부터 들어간다고 생각하면 되는건가요?

제 짧은 생각으로는 함수 호출 방식(stdcall 같은...) 문제인것 같은데 정확한지를 몰라서 좀 그렇네요.

혹시 명확하게 아시는 분 설명 부탁드려봅니다. ^^

mrmirang의 이미지

http://bbs.kldp.org/viewtopic.php?t=31800
예전에 있었던 답변이니 참조하여 보세요.

ifyou의 이미지

위와 같은 경우 표준에서 정해지지 않았기 때문에

컴파일러 마음입니다.

자세하고 정확한 답변이 나오기 전에 간단하게 함 적어봤습니다..
토욜 출근해서 일도 안되고.. :cry:

lsj0713의 이미지

printf("%d %d %d", ++k, ++k, ++k); 

일단, 함수 매개변수의 평가 순서는 정의된 바 없습니다. (undefined인지 unspecified인지는 잘...) 다만 함수 내부로 매개변수들이 전달되기 전에 모든 매개변수들이 평가된다는 것만은 보장이 됩니다. 따라서 위의 코드의 결과는 1 2 3이 될수도 있고 1 3 2가 될수도 있고 3 2 1 이 될 수도 있습니다. , 연산자가 시퀸스 포인트이며 왼쪽부터 평가가 된다는 사실 때문에 1 2 3이 나와야 된다고 생각하는 사람도 있는데 함수 호출 수식에서 사용되는 ,는 연산자가 아니라 매개변수를 구분해주는 구두점입니다.

그러나 우리가 사용하는 대부분의 컴파일러에서는 3 2 1의 결과가 나올 텐데 그것은 함수의 매개변수가 전달될 때 스택에 쌓여서 전달되기 때문에 그렇습니다. 하지만 항상 그런 것은 아닙니다. 어떤 ARM 용 컴파일러에서는 '좀 더 복잡한 매개변수'부터 평가하기도 합니다. 예를 들면 func(1, func2(), 3)에서 func2()를 가장 먼저 평가합니다.

std::cout << ++k << ++k << ++k;

이것은 다음과 같은 코드로 변환이 됩니다.

operator<<(operator<<(operator<<(std::cout, ++k), ++k), ++k);

아니면 다음과 같이 될 수도 있겠지요. 둘중의 어느쪽인지는 제 밑천이 딸려서 모르겠습니다. -_-;

std::cout.operator<<(++k).operator<<(++k).operator<<(++k);

첫번째 경우라면 함수 호출시 매개변수의 평가순서에 영향을 받으며 두번째 경우라면 함수 호출 수식의 평가순서(func_name(...) 가 있을 때 func_name 먼저 평가할 것인가 (...)먼저 평가할 것인가)의 영향을 받습니다. 둘 다 제대로 정의된 바 없습니다. (이것도 역시 undefined인지 unspecified인지는 잘...)

관련글: http://groups.google.com/groups?selm=3EEF4854.EB0F2451%40bawi.org

결론을 내리자면, '대부분의 컴파일러에서는 3 2 1을 찍어준다. 이것은 대부분의 컴파일러가 함수 호출시에 매개변수를 함수 내부로 전달할 때 스택에 쌓아서 전달해 주기 때문에 그렇다. 그러나 C 표준에 의하면 매개변수의 평가 순서에 대해서는 정의된 바 없기 때문에 이런 정의되지 않은 동작에 의존하는 프로그래밍은 좋지 않다.' 가 되겠습니다.

ps. 좀 더 자세한 것을 알고 싶으시면 C 언어에서 side effect와 sequence point에 대해 찾아보시면 되겠습니다.

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.