특정 타임스탬프 사이의 근무 시간을 계산할 때는 주말과 지정된 근무 시간을 고려해야 합니다. PostgreSQL에서는 다양한 기술을 활용하여 이 작업을 수행할 수 있습니다.
특정 타임스탬프 범위의 경우:
1시간 단위를 고려하세요. 분수를 무시합니다. 공식은 다음과 같습니다.
SELECT count(*) AS work_hours FROM generate_series(timestamp '2013-06-24 13:30', timestamp '2013-06-24 15:29' - interval '1h', interval '1h') h WHERE EXTRACT(ISODOW FROM h) < 6 AND h::time >= '08:00' AND h::time <= '14:00';
전체 테이블의 경우:
-- Table Creation and Data Insertion CREATE TABLE t (t_id int PRIMARY KEY, t_start timestamp, t_end timestamp); INSERT INTO t VALUES (1, '2009-12-03 14:00', '2009-12-04 09:00') , (2, '2009-12-03 15:00', '2009-12-07 08:00') , (3, '2013-06-24 07:00', '2013-06-24 12:00') , (4, '2013-06-24 12:00', '2013-06-24 23:00') , (5, '2013-06-23 13:00', '2013-06-25 11:00') , (6, '2013-06-23 14:01', '2013-06-24 08:59'); -- Main Query SELECT t_id, count(*) AS work_hours FROM ( SELECT t_id, generate_series(t_start, t_end - interval '1h', interval '1h') AS h FROM t ) sub WHERE EXTRACT(ISODOW FROM h) < 6 AND h::time >= '08:00' AND h::time <= '14:00' GROUP BY 1 ORDER BY 1;
더 높은 정밀도를 얻으려면 더 작은 값을 사용하세요. 5분 조각과 같은 시간 단위.
-- Precision with 5-minute Slices SELECT t_id, count(*) * interval '5 min' AS work_interval FROM ( SELECT t_id, generate_series(t_start, t_end - interval '5 min', interval '5 min') AS h FROM t ) sub WHERE EXTRACT(ISODOW FROM h) < 6 AND h::time >= '08:00' AND h::time <= '14:55' -- 15.00 - interval '5 min' GROUP BY 1 ORDER BY 1;
마이크로초까지 정확한 결과를 얻으려면 시작 시간과 종료 시간을 별도로 처리하세요.
Postgres 8.4 :
WITH var AS (SELECT '08:00'::time AS v_start, '15:00'::time AS v_end) SELECT t_id , COALESCE(h.h, '0') -- add / subtract fractions - CASE WHEN EXTRACT(ISODOW FROM t_start) < 6 AND t_start::time > v_start AND t_start::time < v_end THEN t_start - date_trunc('hour', t_start) ELSE '0'::interval END + CASE WHEN EXTRACT(ISODOW FROM t_end) < 6 AND t_end::time > v_start AND t_end::time < v_end THEN t_end - date_trunc('hour', t_end) ELSE '0'::interval END AS work_interval FROM t CROSS JOIN var LEFT JOIN ( -- count full hours, similar to above solutions SELECT t_id, count(*)::int * interval '1h' AS h FROM ( SELECT t_id, v_start, v_end , generate_series(date_trunc('hour', t_start), date_trunc('hour', t_end) - interval '1h', interval '1h') AS h FROM t, var ) sub WHERE EXTRACT(ISODOW FROM h) < 6 AND h::time >= v_start AND h::time <= v_end - interval '1h' GROUP BY 1 ) h USING (t_id)
위 내용은 PostgreSQL에서 두 날짜 사이의 근무 시간을 계산하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!