특정월에 대한 생일인 사람들의 리스트를 뽑는 쿼리문입니다.
단순히 양력만 사용한다면 아주 쉽게 할 수 있고,
음력포함하더라도 오늘 날짜 (특정일) 의 생일 목록도 쉽게 나오는데..
음력이 포함된 특정 달.. 이거 생각보다 상당히 복잡해지네요..
일단 음력 생일 데이타가 거의 없어 제대로 테스트는 안 되었습니다.
대충 해보니 잘 나오는거 같습니다.
비용은 크게 마니 안 드는거 같은데... 일단 쿼리문에서 반복이 좀 많은데...
이걸 어캐 좀 간단하게 처리할 방법은 없을까요? 반복구문을 함수로 만들긴 좀.... 뷰도 곤란하고..
의견있으신분 댓글 달아주세요..
PostgreSQL 용입니다만.. 쿼리문이 특별한 부분이 없으므로 타 DBMS 에서도 아주 조금만 고치면 얼마던지 적용 가능할 거 같습니다.
(이하 존칭 생략)
member - 회원테이블중 일부
user_id text : 사용자 아이디
birth_year smallint : 생년월일 중 년도
birth_md char(4) : 생년월일 중 월일
solar boolean : 양/음 구분 (양력 true)
lunar - 양/음력 변환 테이블 중 일부
lumar_md char(4) : 음력 년월일 중 월일
solars integer : 음력에 해당하는 양력을 년월일로 붙여 만든 숫자형
date 형의 컬럼도 있지만.. to_char 등의 함수를 쓰야 하기 때문에..
yun boolean : 윤달 여부 (윤달 true)
select lunar_y, lunar_md, solar, yun, solars from lunar;
-------+---------+------------+-------+-----------
lunar_y lunar_md solar yun solars
-------+---------+------------+-------+-----------
1899 | 1201 | 1900-01-01 | f | 19000101
1899 | 1202 | 1900-01-02 | f | 19000102
1899 | 1203 | 1900-01-03 | f | 19000103
1899 | 1204 | 1900-01-04 | f | 19000104
...........
<참고>
1. PostgreSQL 에서는 날짜형이 엄격하기 때문에 음력을 제대로 표현할 수 없다.
이럴테면 1990-02-30 과 같은 경우이다.
그래서 음력 표현 방법중 년도와 월일을 분리하고 년도는 숫자 월일은 문자로 했다.
2. 양/음력을 프로그램으로 변환할 수도 있지만 완벽하지 않으므로
돌아다니는 1900 ~ 2200 까지의 음력 데이타 를 이용하였다.
3. 윤달을 전혀 고려하지 않았다. 만일 윤달이라면 yun 을 true 로 하면 된다.
4. 생일에 대한 년도는 필요없으나, 특정월에 대한 년도는 필요하다. (년마다 음력 날짜가 틀리므로)
먼저 특정 년월에 대한 마지막날을 구하는 함수를 만든다.
물론 만들 필요없이 바로 해당 위치에 집어 넣어도 아무런 상관은 없지만,
마지막일을 구하는 일은 종종 발생할 수 있기 때문에 미리 함수로 만들어놓으면 편하다.
create or replace function last_date (integer) returns date as
$$
select (to_date($1, 'YYYYMM') + interval '1 month' - interval '1 day')::date;
$$ language sql;
select last_date(200702);
last_date
------------
2007-02-28
이런식으로 나옵니다.
prepare get_birthday (integer) as
(
select * from member
where (birth_md between
to_char(last_date($1), 'MM01')
and to_char(last_date($1), 'MMDD')
and solar = true
)
or case when
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
>
(select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
then (
(birth_md between
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
and '1231'
)
or
(birth_md between
'0101'
and (select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
)
and solar = false
)
else
(birth_md between
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
and (select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
and solar = false
)
end
order by user_id
);
샤용법은
execute get_birthday(200706);
prepare 로 지정하지 않고
저것 자체를 function 으로 만들고 함수 인자를 interger 로 받아 써도 좋겠죠?
단순히 양력만 사용한다면 아주 쉽게 할 수 있고,
음력포함하더라도 오늘 날짜 (특정일) 의 생일 목록도 쉽게 나오는데..
음력이 포함된 특정 달.. 이거 생각보다 상당히 복잡해지네요..
일단 음력 생일 데이타가 거의 없어 제대로 테스트는 안 되었습니다.
대충 해보니 잘 나오는거 같습니다.
비용은 크게 마니 안 드는거 같은데... 일단 쿼리문에서 반복이 좀 많은데...
이걸 어캐 좀 간단하게 처리할 방법은 없을까요? 반복구문을 함수로 만들긴 좀.... 뷰도 곤란하고..
의견있으신분 댓글 달아주세요..
PostgreSQL 용입니다만.. 쿼리문이 특별한 부분이 없으므로 타 DBMS 에서도 아주 조금만 고치면 얼마던지 적용 가능할 거 같습니다.
(이하 존칭 생략)
member - 회원테이블중 일부
user_id text : 사용자 아이디
birth_year smallint : 생년월일 중 년도
birth_md char(4) : 생년월일 중 월일
solar boolean : 양/음 구분 (양력 true)
lunar - 양/음력 변환 테이블 중 일부
lumar_md char(4) : 음력 년월일 중 월일
solars integer : 음력에 해당하는 양력을 년월일로 붙여 만든 숫자형
date 형의 컬럼도 있지만.. to_char 등의 함수를 쓰야 하기 때문에..
yun boolean : 윤달 여부 (윤달 true)
select lunar_y, lunar_md, solar, yun, solars from lunar;
-------+---------+------------+-------+-----------
lunar_y lunar_md solar yun solars
-------+---------+------------+-------+-----------
1899 | 1201 | 1900-01-01 | f | 19000101
1899 | 1202 | 1900-01-02 | f | 19000102
1899 | 1203 | 1900-01-03 | f | 19000103
1899 | 1204 | 1900-01-04 | f | 19000104
...........
<참고>
1. PostgreSQL 에서는 날짜형이 엄격하기 때문에 음력을 제대로 표현할 수 없다.
이럴테면 1990-02-30 과 같은 경우이다.
그래서 음력 표현 방법중 년도와 월일을 분리하고 년도는 숫자 월일은 문자로 했다.
2. 양/음력을 프로그램으로 변환할 수도 있지만 완벽하지 않으므로
돌아다니는 1900 ~ 2200 까지의 음력 데이타 를 이용하였다.
3. 윤달을 전혀 고려하지 않았다. 만일 윤달이라면 yun 을 true 로 하면 된다.
4. 생일에 대한 년도는 필요없으나, 특정월에 대한 년도는 필요하다. (년마다 음력 날짜가 틀리므로)
먼저 특정 년월에 대한 마지막날을 구하는 함수를 만든다.
물론 만들 필요없이 바로 해당 위치에 집어 넣어도 아무런 상관은 없지만,
마지막일을 구하는 일은 종종 발생할 수 있기 때문에 미리 함수로 만들어놓으면 편하다.
create or replace function last_date (integer) returns date as
$$
select (to_date($1, 'YYYYMM') + interval '1 month' - interval '1 day')::date;
$$ language sql;
select last_date(200702);
last_date
------------
2007-02-28
이런식으로 나옵니다.
prepare get_birthday (integer) as
(
select * from member
where (birth_md between
to_char(last_date($1), 'MM01')
and to_char(last_date($1), 'MMDD')
and solar = true
)
or case when
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
>
(select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
then (
(birth_md between
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
and '1231'
)
or
(birth_md between
'0101'
and (select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
)
and solar = false
)
else
(birth_md between
(select lunar_md from lunar
where solars = ($1 || '01')::int
and yun = false
)
and (select lunar_md from lunar
where solars = to_char(last_date($1), 'YYYYMMDD')::int
and yun = false
)
and solar = false
)
end
order by user_id
);
샤용법은
execute get_birthday(200706);
prepare 로 지정하지 않고
저것 자체를 function 으로 만들고 함수 인자를 interger 로 받아 써도 좋겠죠?
반응형
'인터넷정보' 카테고리의 다른 글
갤러리(최근 or 랜덤) 슬라이드 + 새창(이미지크기로..) + 해당글 이동 (0) | 2007.10.13 |
---|---|
19금 - 성인인증 소스 (0) | 2007.10.13 |
19금 - 성인인증 소스 (0) | 2007.10.13 |
제로보드4 에 무료 휴대폰 알림기능 추가 (0) | 2007.10.13 |
제로보드4 에 무료 휴대폰 알림기능 추가 (0) | 2007.10.13 |
특정한 달의 생일인 사람 리스트 가져오기 (0) | 2007.10.13 |
히라가나/가타카나를 한글 발음으로 변환/일본어 번역/일본어 한국어 번역/한국어 일본어 번역 (0) | 2007.10.13 |
히라가나/가타카나를 한글 발음으로 변환/일본어 번역/일본어 한국어 번역/한국어 일본어 번역 (0) | 2007.10.13 |
이미지 리사이즈(resize) 및 새창 (0) | 2007.10.13 |
이미지 리사이즈(resize) 및 새창 (0) | 2007.10.13 |