mysql 의 last_insert_id() 가 최초 접속시에 0을 반환합니다.

cococo의 이미지

게시판을 만들고 있는데, 기초적이라고 생각되는 부분에서 안 풀려서, 질문 올려봅니다.

일단 제 환경은 WINAPM 입니다. mysql 버전은 5.1.41 로 나오는군요.

WINAPM으로 테스트 하고, 완성 후 웹호스팅으로 올리는 방법을 사용하고 있는데...이게 중요한 건 아니겠죠?

간단히 말씀드리면, 'id' 라는 필드를 AUTO_INCREMENT 로 선언하였습니다. 그리고 그룹을 선언하기 위해 'rootID' 라는 필드를 선언했습니다.
즉,

1. TABLE 에 row 가 새로 입력되면
2. 'id' FIELD 는 AUTO_INCREMENT 되고,
3. AUTO_INCREMENT 된 값을 방금 입력한 row의 'rootID' 필드에 씁니다.
4. 그러기 위해 last_insert_id() 를 씁니다.

쿼리는 대략

INSERT INTO ( rootID, title, ... ) value ( LAST_INSERT_ID(), ... ) ;

이렇습니다.

처음엔 별 고민 없이 썼는데, 쓰다보니 LAST_INSERT_ID() 가 "최초의 INSERT" 일 경우는 0 을 반환하더군요.( 당연하다면 당연하겠죠.)
뒤져보니, LAST_INSERT_ID() 는 CONNECTION 이 유지된 경우만 값을 반환해서, 접속해서 INSERT 를 한번이라도 수행하지 않으면 무조건 0을 반환하더군요.

근데, 저러면...못쓰잖아요?

그러니까, 예를 들어, 오늘 100명이 글을 쓰고, 내일 사용자가 다시 글을 쓴다면,
내일 접속해서 처음 글 쓰는 사람은, 무조건 rootID 가 0 된다는 의미죠.

물론 transaction 걸고 select MAX() 로 쓰면 되겠지만... 그렇다면 굳이 LAST_INSERT_ID() 를 쓸 필요가 없잖아요?

질문입니다.
1. "최초 접속( = connection 이 끊겼다가 처음 접속) 하는 사람이, AUTO_INCREMENT 속성의 FIELD 값을 알 수 있는 방법" 이 무엇인지요?
2. 혹은, 다들 원래 TRANSACTION + SELECT MAX() + COMMIT 으로 쓰시나요?

조언 부탁드립니다. 뭔가 기초적인 거 같은 느낌은 드는데, 명확한 해답이 보이지 않는군요.

ps: DB WRITE 는 PDO 를 사용하고 있습니다. 혹시 mysql_insert_id() 를 말씀하신다면... pdo 버전의 mysql_insert_id() 가 있는지도 알려주시면 대단히 감사하겠습니다. (--)(__)(--)

yhsuk의 이미지

LAST_INSERT_ID는 커넥션마다 고유한 값을 가지므로 그냥

Begin Transaction
INSERT INTO TB_XXX (rootID, title, ...) value ( rootID 디폴트값, ... );
UPDATE TB_XXX SET rootID = LAST_INSERT_ID();
Commit
End Transactoin

하면 될 것 같은데요.

Signature :) - "여유를 갖고 행동하되 게을러지지 말자"

cococo의 이미지

그...그게... 한방쿼리 쓰려고 하는 거라서요. ㅡ,.ㅡ;; 트랜젝션 들어가면 처리가 귀찮... OTL..

답변 감사드립니다. 좋은 하루 되시기 바랍니다.

익명 사용자의 이미지

last_insert_id()는 해당 connection 객체 마지막으로 "insert 한" 결과를 단순히 cache하고 있는 것에 불과하므로, "insert 될" 값을 얻기 위해 쓰는 건 안됩니다. 심지어 말씀하신대로 하면 rootID는 그 row의 id와 같은 값을 가지게 되는게 아니라 그 전에 insert된 row의 id 값을 가지게 되겠죠.

정확한 requirement는 모르겠지만, 어쨌든 제시된 상황으로는 rootID가 auto_increment되는 id와 늘 같을테니, rootID에도 auto_increment를 적용하시면 되지 않을까 싶네요.

익명 사용자의 이미지

죄송.. 한 개 컬럼에만 auto_increment를 적용할 수 있군요.

cococo의 이미지

한방쿼리의 꿈은 로또와 같다... 라는 뜻으로 알겠습니다.
답변 감사드립니다. 좋은 하루 되시기 바랍니다.

댓글 달기

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
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.