(최종) 시리얼번호 생성 함수 > 개발자팁

개발자팁

개발과 관련된 유용한 정보를 공유하세요.
질문은 QA에서 해주시기 바랍니다.

(최종) 시리얼번호 생성 함수 정보

PHP (최종) 시리얼번호 생성 함수

본문

사용상에는 아무런 제한이 없습니다.
출처를 밝히고 퍼가는 것도 아무런 제한이 없습니다.
단, 교육(강좌)의 내용이나 출판(책)의 내용으로 포함되거나 인용 될수 없습니다.
 
 

몇가지 내용이 변경되었습니다.

1. 베이스가 되는 숫자 문자열을 36자로 유니크하게 생성합니다.
2. 유니크한 숫자에 랜덤한 5자리 소수 가 곱해집니다.
3. 자리수의 한도가 생겻습니다. 숫자형은 36자리 까지, 믹스형은 24자리까지 입니다.
4. 12자리 미만은 대량 생성시 중복이나 연번 발생 할수 있습니다. 생성시 중복체크는 필수 입니다.


[사용예시]
보통 이런 시리얼 번호는
미리 100 만건 정도를 데이타베이스에 생성해놓고
사용자 발급 요청이 있을시 사용되지 않은것중 하나를 발급합니다.
 
이미 생성할때 랜덤하게 되었기 때문에 발급되지 않은 것중 하나를 순차적으로 뽑아서 발급합니다.
발급후 발급 으로 상태를 변경합니다.
 
시리얼 사용 입력을 받을 때는 아이피당으로 하루 실패 5회 이상을 넘지 못하게 합니다.
악의적인 프로그램을 사용할수 있는 경우 때문입니다.
 
입력받은 시리얼은 데이타베이스에서 비교하여
존재하는 시리얼인지
발급되었던 시리얼인지
사용되었던 시리얼인지
체크 한 연후에 아무 이상이 없을 경우 사용 으로 상태를 변경하고
사용할수 있는 서비스 정보를 업데이트 합니다.


<?php

//지정된 자릿수의 랜덤한 숫자를 반환합니다. 최대 10까지 가능합니다. 4 이면 1000 에서 9999 사이의 랜덤 숫자
function get_rand_number($len=4) {

    $len = abs((int)$len);
    if ($len < 1) $len = 1;
    else if ($len > 10) $len = 10;

    return rand(pow(10, $len - 1), (pow(10, $len) - 1));
}

//넘어온 세자리수를 36진수로 변환해서 반환합니다. preg_match_callback 을 통해서만 사용됩니다.
function get_simple_36($m){

    $str = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ';
    $div = floor($m[0] / 36);
    $rest = $m[0] % 36;

    return $str[$div] . $str[$rest];
}

//지정된 자리수에 존재하는 소수 전체를 배열로 반환합니다. max len = 5
function get_simple_prime_number($len=5){

    $len = abs((int)$len);
    if ($len < 1) $len = 1;
    else if ($len > 5) $len = 5;

    $prime_1 = Array(1, 2, 3, 5, 7);

    if ($len == 1) return $prime_1;

    $start = pow(10, ($len - 1)) + 1;//101
    $end = pow(10, $len) - 1;//999
    $prime = $prime_1;

    unset($prime[0]);//1제거
    unset($prime[1]);//2제거
    $array = Array();
    for($i = 11; $i <= $end; $i+=2){//10보다 큰 소수에는 짝수가 없다.

$max = floor(sqrt($i));
        foreach($prime as $j) {

            if ($j > $max) break;
            if ($i % $j == 0) continue 2;
        }

        $prime[] = $i;
        if ($i >= $start) $array[] = $i;
    }

    return $array;
}

//지정된 자릿수의 숫자로된 시리얼을 반환합니다. - 를 포함하고 싶지 않을때는 $cut 이 $len 보다 크거나 같으면 됩니다. max len = 36
function get_serial($len=16, $cut=4, $hipen='-'){

    $len = abs((int)$len);
    if ($len < 1) $len = 16;
    else if ($len > 36) $len = 36;

    $cut = abs((int)$cut);
    if ($cut < 1) $cut = 4;
    else if ($cut > $len) $cut = $len;

    list($usec, $sec) = explode(' ', microtime());
    $base_number = (string)$sec . str_replace('0.', '', (string)$usec);
    $base_number .= (string)get_rand_number(10) . (string)get_rand_number(8);//36자리 유니크한 숫자 문자열
 

    $prime = get_simple_prime_number(5);//5자리 소수 배열
    shuffle($prime);
    $prime = $prime[0];//랜덤한 5자리 소수

    $serial = bcmul(substr($base_number, 0, $len), $prime);
    $serial_length = strlen($serial);
    $sub = $len - $serial_length;

    if ($sub > 0) $serial .= (string)get_rand_number($sub);
    else if ($sub < 0) $serial = substr($serial, 0, $len);

    return preg_replace("`(.{" . $cut . "})`", "$1" . $hipen, $serial, floor(($len-1) / $cut));
}

//지정된 자릿수의 숫자와 영문으로된 시리얼을 반환합니다. - 를 포함하고 싶지 않을때는 $cut 이 $len 보다 크거나 같으면 됩니다. max len = 24
function get_serial_mix($len=16, $cut=4, $hipen='-'){

    $len = abs((int)$len);
    if ($len < 1) $len = 16;
    else if ($len > 24) $len = 24;

    $cut = abs((int)$cut);
    if ($cut < 1) $cut = 4;
    else if ($cut > $len) $cut = $len;

    $len2 = (int)($len * 3 / 2);
    if ($len2 % 2 == 1) $len2 += 1;

    $serial = get_serial($len2, $len2, $hipen);

    $serial = substr(preg_replace_callback("`.{3}`", "get_simple_36", $serial), 0, $len);

    return preg_replace("`(.{" . $cut . "})`", "$1" . $hipen, $serial, floor(($len-1) / $cut));
}

echo get_serial_mix(16, 4, '-');
?>
추천
5

댓글 14개

감사합니다. ^^
혹시 기존의 랜덤 숫자 대신 소수를 사용하신 이유을 여쭤봐도 될까요?
그냥 생각에, 랜덤 숫자보다 소수의 가지수가 훨씬 적어서 결과적으로 나올 수 있는 숫자의 종류도 적어질것 같아서요. ^^'
네에 그렇게 한이유는
이전방법이든 현재방법이든
일정 자릿수 이상에선 동시에 생성시 중복은 없습니다.

그렇지만, 일정시간이 흐룬후 다시 생성햇을때
이전 생성된 번호가 미약하나마 나올수 있어서(곱하기 때문에)
소수를 사용하게 되었습니다.
소수는 더 나누어질수 없는 수이기 때문에
앞에 시간이 정확히 일치 하지 않으면 완전한 동일한 수는 나올 확률이 아예  없습니다.
그런데 시간은 뒤로 흐르니
언제 어느시점에 해도 일정 자리수 이상에선 절대 중복이 나올수 없습니다.
설명 감사드립니다. ^^

난수 생성 방법중에도 소수를 이용하는 알고리즘이 있는데,
비슷한 원리가 아닐까 생각합니다.
이 정도의 코드라면, 거의 완벽에 가까운 시리얼 키 생성함수가 아닐까 생각되네요.

정말 많은 공부되었습니다. 이런 기회를 주셔서 너무 고맙습니다. ^^*
list($usec, $sec) = explode(' ', microtime());
    $base_number = str_repeat((string)$sec, 2);
    $base_number .= str_repeat(str_replace('0.', '', (string)$usec), 2);//36자리 유니크한 숫자 문자열

이부분은

list($usec, $sec) = explode(' ', microtime());
    $base_number = (string)$sec . str_replace('0.', '', (string)$usec);
    $base_number .= (string)get_rand_number(10) . (string)get_rand_number(8);//36자리 유니크한 숫자 문자열

이게 맞는거 같습니다.
동시에 생성시 완벽한 중복을 피하기 위해서는
자릿수에 맞게 마이크로 타임이 항상 달라질수 있도록
usleep 을 적절히 사용해야 합니다.

예를들면
16자리 숫자 시리얼을 생성하고자 할때

while(1) {
    get_serial(16, 4, '-');
    usleep(10);//십만분의 1초 쉼
}
그런데.... 내부에서 소수 구하는 부분에서
0.6초 대의 시간이 발생하므로
안넣어도 상관은 없을거 같기도 합니다.
전체 64
개발자팁 내용 검색

회원로그인

(주)에스아이알소프트 / 대표:홍석명 / (06211) 서울특별시 강남구 역삼동 707-34 한신인터밸리24 서관 1404호 / E-Mail: admin@sir.kr
사업자등록번호: 217-81-36347 / 통신판매업신고번호:2014-서울강남-02098호 / 개인정보보호책임자:김민섭(minsup@sir.kr)
© SIRSOFT