(최종) 시리얼번호 생성 함수
사용상에는 아무런 제한이 없습니다.
출처를 밝히고 퍼가는 것도 아무런 제한이 없습니다.
단, 교육(강좌)의 내용이나 출판(책)의 내용으로 포함되거나 인용 될수 없습니다.
몇가지 내용이 변경되었습니다.
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보다 큰 소수에는 짝수가 없다.
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;
}
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자리 유니크한 숫자 문자열
$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, '-');
?>
댓글 14개
Terrorboy
13년 전
추천합니다.!!
13년 전
감사합니다. ^^
혹시 기존의 랜덤 숫자 대신 소수를 사용하신 이유을 여쭤봐도 될까요?
그냥 생각에, 랜덤 숫자보다 소수의 가지수가 훨씬 적어서 결과적으로 나올 수 있는 숫자의 종류도 적어질것 같아서요. ^^'
혹시 기존의 랜덤 숫자 대신 소수를 사용하신 이유을 여쭤봐도 될까요?
그냥 생각에, 랜덤 숫자보다 소수의 가지수가 훨씬 적어서 결과적으로 나올 수 있는 숫자의 종류도 적어질것 같아서요. ^^'
13년 전
네에 그렇게 한이유는
이전방법이든 현재방법이든
일정 자릿수 이상에선 동시에 생성시 중복은 없습니다.
그렇지만, 일정시간이 흐룬후 다시 생성햇을때
이전 생성된 번호가 미약하나마 나올수 있어서(곱하기 때문에)
소수를 사용하게 되었습니다.
소수는 더 나누어질수 없는 수이기 때문에
앞에 시간이 정확히 일치 하지 않으면 완전한 동일한 수는 나올 확률이 아예 없습니다.
그런데 시간은 뒤로 흐르니
언제 어느시점에 해도 일정 자리수 이상에선 절대 중복이 나올수 없습니다.
이전방법이든 현재방법이든
일정 자릿수 이상에선 동시에 생성시 중복은 없습니다.
그렇지만, 일정시간이 흐룬후 다시 생성햇을때
이전 생성된 번호가 미약하나마 나올수 있어서(곱하기 때문에)
소수를 사용하게 되었습니다.
소수는 더 나누어질수 없는 수이기 때문에
앞에 시간이 정확히 일치 하지 않으면 완전한 동일한 수는 나올 확률이 아예 없습니다.
그런데 시간은 뒤로 흐르니
언제 어느시점에 해도 일정 자리수 이상에선 절대 중복이 나올수 없습니다.
13년 전
설명 감사드립니다. ^^
난수 생성 방법중에도 소수를 이용하는 알고리즘이 있는데,
비슷한 원리가 아닐까 생각합니다.
이 정도의 코드라면, 거의 완벽에 가까운 시리얼 키 생성함수가 아닐까 생각되네요.
정말 많은 공부되었습니다. 이런 기회를 주셔서 너무 고맙습니다. ^^*
난수 생성 방법중에도 소수를 이용하는 알고리즘이 있는데,
비슷한 원리가 아닐까 생각합니다.
이 정도의 코드라면, 거의 완벽에 가까운 시리얼 키 생성함수가 아닐까 생각되네요.
정말 많은 공부되었습니다. 이런 기회를 주셔서 너무 고맙습니다. ^^*
13년 전
별말씀을요
잘 보아 주셔서 고맙습니다.
잘 보아 주셔서 고맙습니다.
똥싼너구리
13년 전
와우 점점 디테일해지는군용!!
13년 전
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자리 유니크한 숫자 문자열
이게 맞는거 같습니다.
$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자리 유니크한 숫자 문자열
이게 맞는거 같습니다.
13년 전
동시에 생성시 완벽한 중복을 피하기 위해서는
자릿수에 맞게 마이크로 타임이 항상 달라질수 있도록
usleep 을 적절히 사용해야 합니다.
예를들면
16자리 숫자 시리얼을 생성하고자 할때
while(1) {
get_serial(16, 4, '-');
usleep(10);//십만분의 1초 쉼
}
자릿수에 맞게 마이크로 타임이 항상 달라질수 있도록
usleep 을 적절히 사용해야 합니다.
예를들면
16자리 숫자 시리얼을 생성하고자 할때
while(1) {
get_serial(16, 4, '-');
usleep(10);//십만분의 1초 쉼
}
13년 전
그런데.... 내부에서 소수 구하는 부분에서
0.6초 대의 시간이 발생하므로
안넣어도 상관은 없을거 같기도 합니다.
0.6초 대의 시간이 발생하므로
안넣어도 상관은 없을거 같기도 합니다.
13년 전
아래글에도 댓글을 남겼는데,
pow 계산부분을 $prime 루프 밖으로 빼면 속도에 도움이 되는 것 같습니다. ^^
pow 계산부분을 $prime 루프 밖으로 빼면 속도에 도움이 되는 것 같습니다. ^^
13년 전
좋은 지적입니다.
요샌 집중력이 떨어져서 그런지
하나 적용하면 하나 빼먹고 그러네요.
요샌 집중력이 떨어져서 그런지
하나 적용하면 하나 빼먹고 그러네요.
13년 전
추천합니다.!!!
13년 전
네에 고맙습니다.
takumi22
12년 전
유용하게 사용할수 있을것 같습니다 감사합니다
게시글 목록
| 번호 | 제목 |
|---|---|
| 4405 | |
| 17973 |
Mobile
서브도메인 연결 오류...해결 부탁드립니다.
1
|
| 17969 | |
| 30568 |
HTML
고수님들 도움좀 부탁드립니다!
|
| 17965 |
PHP
웹 프로그래밍 강의사이트
3
|
| 17962 |
JavaScript
PhotoSwipe(갤러리) js 소스 해결좀 부탁드립니다.
2
|
| 17959 | |
| 17955 |
JavaScript
초보자 궁금한게 있습니다!
3
|
| 17953 | |
| 17950 | |
| 17947 | |
| 4403 | |
| 17944 |
jQuery
메뉴에서 무엇이 문제인가요?
2
|
| 17943 |
JavaScript
스크립트 소숫점 자르기
|
| 17941 | |
| 17939 |
Linux
리눅스 yum으로 apm설치하기
1
|
| 17935 |
jQuery
FAQ 예제입니다.
3
|
| 17928 | |
| 4391 | |
| 17926 |
Flash
저도 또 플래시 기본플레이어 질문이네요;
1
|
| 17923 | |
| 17920 |
jQuery
jquery 라이브러리 개발자 사이트
2
|
| 17917 |
jQuery
[질문] 이미지 메뉴 페이드 효과 넣고 싶어요.
2
|
| 26380 | |
| 17916 |
PHP
멀티스크린 대응
|
| 30565 |
HTML
[접기 펴기]간단 소스 문의좀..
2
|
| 17910 | |
| 17907 |
JavaScript
숫자 계산시 NaN 이 나올경우질문드립니다
2
|
| 17905 |
MySQL
ms sql 작업하실분 구합니다.
1
|
| 30555 | |
| 17904 |
MySQL
ssh 터널링을 통한 mysql 접속
|
| 30552 | |
| 17901 |
JavaScript
버튼 롤아웃시에 버튼 온 이미지를 줄수있는 방법 좀 갈켜 주세요.
2
|
| 17900 |
JavaScript
따라다니는 배너 스크립트인데요
|
| 17898 |
Flash
플래시가 검게만 보입니다 ㅠㅠ
1
|
| 30549 | |
| 17895 | |
| 26379 | |
| 26378 | |
| 26377 | |
| 17893 | |
| 17891 |
JavaScript
메뉴바 - 이런걸 원하시는건가??
1
|
| 17888 | |
| 17885 | |
| 17883 | |
| 17882 | |
| 17861 | |
| 17859 |
Flash
플레시 메뉴 만들기중 질문드립니다.
1
|
| 17857 | |
| 4388 | |
| 24413 | |
| 4383 | |
| 17849 | |
| 17847 |
PHP
효과에 대한 질문입니다.
1
|
| 17845 |
Flash
플래시 플레이어가 안나오네요!
1
|
| 17844 |
Linux
mod_ssl 설치 문제
|
| 17839 | |
| 17838 | |
| 30548 |
HTML
이미지 게시판 질문드려요
|
| 17836 |
JavaScript
메뉴바 도움 부탁드립니다.
1
|
| 17833 | |
| 17831 | |
| 17827 |
JavaScript
모바일 홈페이지 연동소스들중에 베가레이스2 기종이 연동이 안된다고 하는데
3
|
| 17826 | |
| 17823 |
JavaScript
네이버 나눔글꼴 적용 좀 봐 주세요
2
|
| 17820 |
MySQL
중복제거 쿼리문좀 간절히 부탁드립니다^^
2
|
| 17818 | |
| 24412 | |
| 30547 |
HTML
모바일 퍼스트와 반응형 웹
|
| 17814 |
MySQL
정중히 문제해결 도움 요청합니다
3
|
| 4377 | |
| 30544 |
HTML
프레임 문의 드려요 ^^;;
2
|
| 17813 |
JavaScript
새창뜨우기 소스에요
|
| 17810 |
기타
서브도메인
2
|
| 26373 | |
| 17804 | |
| 30543 |
HTML
HTML5 스터디가 가능할까요?
|
| 17803 | |
| 4370 | |
| 17801 |
JavaScript
스맛폰 에서 보면 메뉴가 뒤로 숨어서 안보이는데요?
1
|
| 17796 | |
| 26372 | |
| 17794 | |
| 30536 |
HTML
반응형 웹 스타디 초안?
6
|
| 17791 |
JavaScript
클래스 안에 태그의 src 추출
2
|
| 4364 | |
| 17789 | |
| 30535 |
HTML
opacity 질문입니다.!
|
| 17788 |
JavaScript
페이지 경고창 없애고 무한 롤링
|
| 30527 | |
| 30525 | |
| 30524 | |
| 4358 | |
| 30521 | |
| 17781 |
기타
DTD가 뭔가요?
6
|
| 26371 | |
| 17776 | |
| 4343 | |
| 17772 | |
| 17768 |
댓글 작성
댓글을 작성하시려면 로그인이 필요합니다.
로그인하기