MySQL Query GROUP BY 일 / 월 / 년


649

1 년, 월 또는 일과 같이 결정된 기간 동안 TIMESTAMP필드 가있는 레코드 수를 계산하는 간단한 쿼리를 만들 수 있습니까?

SELECT COUNT(id)
FROM stats
WHERE record_date.YEAR = 2009
GROUP BY record_date.YEAR

또는:

SELECT COUNT(id)
FROM stats
GROUP BY record_date.YEAR, record_date.MONTH

월간 통계가 있어야합니다.

감사!


1
GROUP BY record_date.MONTH첫 번째 코드 스 니펫에 있어야한다고 생각 합니까?
chiccodoro

답변:


1012
GROUP BY YEAR(record_date), MONTH(record_date)

MySQL 의 날짜 및 시간 기능 을 확인하십시오 .


27
레코드가 몇 년에 걸쳐있는 경우와 같이, 명확성을 높이기 위해 추가 열을 추가 할 수도 있습니다. COUNT (event_id), DATE_FORMAT (event_start, '% Y / % m') 선택
Ric

간단한 완전한 예 : SELECT count(*), record_date FROM anytable WHERE anytable.anycolumn = 'anycondition' GROUP BY YEAR(record_date), month(record_date);참고 : record_date는 날짜 유형 TIMESTAMP
renedet

아마도 이것이 COUNT 별칭 열과 함께 MySQL 5.7에서 실행되지 않았다는 것을 언급 할 가치가 있습니다 (오류 없음, 결과가 없습니다). 별명으로 해당 필드를 선택하도록 변경하면 별명별로 그룹화 할 수 있습니다. 이것은 로컬 환경에서 실행되는 표준 MySQL 5.7 도커 이미지이므로 오류가 발생하거나 결과를 반환하지 않는 이유를 모르겠습니다.
MrMesees

3
오 신 이여, 내가 이것을 이전에 알고 있다면 ... mysql이 한 줄에서 할 수있는 일을하기 위해 많은 PHP 줄이 있습니다.

230
GROUP BY DATE_FORMAT(record_date, '%Y%m')

참고로 (주로 잠재적 다운 보더에게). 현재 이것은 다른 제안만큼 효율적이지 않을 수 있습니다. 아직도, 나는 그것을 다른 대안으로 남겨두고 다른 솔루션이 얼마나 빠른지 알 수있는 대안으로 남겨 둡니다. (차이점을 볼 때까지 천천히 느리게 알 수 없기 때문에) 시간이 지남에 따라 최적화와 관련하여 MySQL 엔진을 변경 하여이 솔루션을 일부 (아마도 그렇지 않은 경우) 변경할 수 있습니다 먼 미래), 대부분의 다른 사람들과 효율성이 상당히 비슷합니다.


3
형식 함수가 날짜 열에서 인덱스를 사용할 수 없기 때문에 이것이 제대로 수행되지 않을 것이라고 생각합니다.
Sonny

@Stv : @ fu-chi의 대답 을 고려하고 싶을 수도 있습니다 . 내가 알 수있는 한, 그 대답과 광산의 그룹화 표현은 같은 것으로 평가되지만 EXTRACT()보다 효율적일 수 있습니다 DATE_FORMAT(). (하지만 적절한 테스트를위한 MySQL은 없습니다.)
Andriy M

45

이거 한번 해봐

SELECT COUNT(id)
FROM stats
GROUP BY EXTRACT(YEAR_MONTH FROM record_date)

EXTRACT (단위 FROM 날짜) 함수는 그룹화를 적게 사용하고 숫자 값을 반환하므로 더 좋습니다.

그룹화시 비교 조건은 DATE_FORMAT 함수 (문자열 값을 반환)보다 빠릅니다. SQL 비교 조건 (WHERE, HAVING, ORDER BY, GROUP BY)에 문자열이 아닌 값을 리턴하는 function | field를 사용해보십시오.


43

위의 'WHERE'문을 사용해 보았습니다. 아무도 수정하지 않았지만 잘못되었습니다. 일부 검색 후 나는 이것이 WHERE 문에 대한 올바른 수식이라는 것을 알았으므로 코드는 다음과 같습니다.

SELECT COUNT(id)  
FROM stats  
WHERE YEAR(record_date) = 2009  
GROUP BY MONTH(record_date)

30

검색 기간이 몇 년이 지났는데도 여전히 월 단위로 그룹화하려면 다음을 제안합니다.

버전 # 1 :

SELECT SQL_NO_CACHE YEAR(record_date), MONTH(record_date), COUNT(*)
FROM stats
GROUP BY DATE_FORMAT(record_date, '%Y%m')

버전 # 2 (보다 효율적) :

SELECT SQL_NO_CACHE YEAR(record_date), MONTH(record_date), COUNT(*)
FROM stats
GROUP BY YEAR(record_date)*100 + MONTH(record_date)

이 버전을 큰 테이블에서 1,357,918 행으로 비교했습니다.), 두 번째 버전의 결과가 더 좋습니다.

version1 (평균 10 회 실행) : 1.404 초
version2 (평균 10 회 실행) : 0.780 초

( SQL_NO_CACHE키가 추가되어 MySQL이 CACHING을 쿼리하지 못하게합니다.)


1
@ fu-chi의 제안을 테스트에 포함시키는 것이 더 효율적일 수 있습니다. 또한 테스트 GROUP BY YEAR(record_date)*100 + MONTH(record_date)했지만 테스트하지 않은 이유는 GROUP BY YEAR(record_date), MONTH(record_date)무엇입니까?
Andriy M

2
COUNT (1) insteed COUNT (*)를 사용하면 훨씬 빠르며 결과 데이터는 동일합니다.
Pa0l0

2
*100Versión # 2에서 이것이 무엇입니까 ? 미리 감사드립니다.
Avión

1
*100~YEAR(record_date)*100 + MONTH(record_date) == DATE_FORMAT(record_date, '%Y%m')
Phu Duy

17

MySQL에서 날짜별로 그룹화하려면 아래 코드를 사용하십시오.

 SELECT COUNT(id)
 FROM stats
 GROUP BY DAYOFMONTH(record_date)

이 스레드를 찾을 사람들에게 시간이 절약되기를 바랍니다.


6
MONTH(record_date)여러 달을 설명하기 위해 그룹화해야 할 수도 있습니다 .
웹넷

14

특정 연도 (예 : 2000)에 대한 레코드를 필터링하려면 다음 WHERE과 같이 절 을 최적화하십시오 .

SELECT MONTH(date_column), COUNT(*)
FROM date_table
WHERE date_column >= '2000-01-01' AND date_column < '2001-01-01'
GROUP BY MONTH(date_column)
-- average 0.016 sec.

대신에:

WHERE YEAR(date_column) = 2000
-- average 0.132 sec.

결과는 300k 개의 행과 인덱스 날짜 열이 포함 된 테이블에 대해 생성되었습니다.

GROUP BY조항에 관해서 는 위에서 언급 한 표에 대해 세 가지 변형을 테스트했습니다. 결과는 다음과 같습니다.

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY YEAR(date_column), MONTH(date_column)
-- codelogic
-- average 0.250 sec.

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY DATE_FORMAT(date_column, '%Y%m')
-- Andriy M
-- average 0.468 sec.

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY EXTRACT(YEAR_MONTH FROM date_column)
-- fu-chi
-- average 0.203 sec.

마지막이 승자입니다.


10

유사하지만 성능이 더 짧고 유연한 대안이 현재 사용중인 완전하고 간단한 솔루션 :

SELECT COUNT(*) FROM stats
-- GROUP BY YEAR(record_date), MONTH(record_date), DAYOFMONTH(record_date)
GROUP BY DATE_FORMAT(record_date, '%Y-%m-%d')

7

최신 달을 기준으로 매년 한 달에 행 수를 가진 월별 통계를 얻으려면 다음을 시도하십시오.

SELECT count(id),
      YEAR(record_date),
      MONTH(record_date) 
FROM `table` 
GROUP BY YEAR(record_date),
        MONTH(record_date) 
ORDER BY YEAR(record_date) DESC,
        MONTH(record_date) DESC

7

GROUP BY에서 간단히 Mysql DATE_FORMAT () 함수를 수행 할 수 있습니다 . 레코드가 몇 년에 걸쳐 있고 같은 달에 다른 해에 발생하는 경우와 같이 경우에 따라 명확성을 높이기 위해 추가 열을 추가 할 수도 있습니다. 시작하기 전에이 내용을 읽으십시오. 그것이 당신에게 매우 도움이되기를 바랍니다. 이해를 돕기위한 샘플 쿼리는 다음과 같습니다.

SELECT
    COUNT(id),
    DATE_FORMAT(record_date, '%Y-%m-%d') AS DAY,
    DATE_FORMAT(record_date, '%Y-%m') AS MONTH,
    DATE_FORMAT(record_date, '%Y') AS YEAR

FROM
    stats
WHERE
    YEAR = 2009
GROUP BY
    DATE_FORMAT(record_date, '%Y-%m-%d ');

4

다음 쿼리는 Oracle Database 12c Release 12.1.0.1.0에서 효과적이었습니다.

SELECT COUNT(*)
FROM stats
GROUP BY 
extract(MONTH FROM TIMESTAMP),
extract(MONTH FROM TIMESTAMP),
extract(YEAR  FROM TIMESTAMP);

2

1 년 그룹 선택을 다음과 같이 최적화하는 것을 선호합니다.

SELECT COUNT(*)
  FROM stats
 WHERE record_date >= :year 
   AND record_date <  :year + INTERVAL 1 YEAR;

이 방법으로 연도를 한 번에 묶을 수 있습니다 (예 : '2009'명명 된 매개 변수 사용) . 따로 추가 '-01-01'하거나 전달할 필요가 없습니다 '2010'.

또한, 우리가 행을 계산되어 같은 아마도 id결코 NULL, 내가 선호 COUNT(*)COUNT(id).


0

.... group by to_char(date, 'YYYY') -> 1989

.... group by to_char(date,'MM') -> 05

.... group by to_char(date,'DD') ---> 23

.... group by to_char(date,'MON') ---> 마이

.... group by to_char(date,'YY') ---> 89


이것은 매우 느릴 것입니다.
earl3s
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.