निम्नलिखित स्कीमा पर विचार करें जिसमें तीसरी तालिका वर्ष/माह हेल्पर टेबल का उल्लेख है। हेल्पर टेबल बहुत आम हैं और स्वाभाविक रूप से आपके पूरे कोड में फिर से उपयोग की जा सकती हैं। मैं इसे पर्याप्त दिनांक डेटा के साथ लोड करने के लिए आप पर छोड़ दूंगा। ध्यान दें कि जिस तरह से प्रत्येक महीने की समाप्ति तिथि हममें से उन लोगों के लिए एक साथ रखी गई थी जो कम काम करना चाहते हैं, जबकि डीबी इंजन को हमारे लिए लीप वर्ष का पता लगाने की इजाजत देता है।
उस सहायक तालिका में आपके पास केवल एक कॉलम हो सकता है। लेकिन इसके लिए आपके कुछ कार्यों में समाप्ति तिथियों के लिए फ़ंक्शन कॉल के उपयोग की आवश्यकता होगी और इसका मतलब है कि अधिक धीमापन। हमें तेज़ पसंद है।
स्कीमा
create table workerRecords
( id int auto_increment primary key,
the_date date not null,
staff_no int not null
);
-- truncate workerRecords;
insert workerRecords(the_date,staff_no) values
('2016-06-10',1),
('2016-06-09',1),
('2016-05-09',1),
('2016-04-09',1),
('2016-03-02',2),
('2016-07-02',2);
create table workers
( staff_no int primary key,
full_name varchar(100) not null
);
-- truncate workers;
insert workers(staff_no,full_name) values
(1,'David Higgins'),(2,"Sally O'Riordan");
नीचे सहायक तालिका
create table ymHelper
( -- Year Month helper table. Used for left joins to pick up all dates.
-- PK is programmer's choice.
dtBegin date primary key, -- by definition not null
dtEnd date null
);
-- truncate ymHelper;
insert ymHelper (dtBegin,dtEnd) values
('2015-01-01',null),('2015-02-01',null),('2015-03-01',null),('2015-04-01',null),('2015-05-01',null),('2015-06-01',null),('2015-07-01',null),('2015-08-01',null),('2015-09-01',null),('2015-10-01',null),('2015-11-01',null),('2015-12-01',null),
('2016-01-01',null),('2016-02-01',null),('2016-03-01',null),('2016-04-01',null),('2016-05-01',null),('2016-06-01',null),('2016-07-01',null),('2016-08-01',null),('2016-09-01',null),('2016-10-01',null),('2016-11-01',null),('2016-12-01',null),
('2017-01-01',null),('2017-02-01',null),('2017-03-01',null),('2017-04-01',null),('2017-05-01',null),('2017-06-01',null),('2017-07-01',null),('2017-08-01',null),('2017-09-01',null),('2017-10-01',null),('2017-11-01',null),('2017-12-01',null),
('2018-01-01',null),('2018-02-01',null),('2018-03-01',null),('2018-04-01',null),('2018-05-01',null),('2018-06-01',null),('2018-07-01',null),('2018-08-01',null),('2018-09-01',null),('2018-10-01',null),('2018-11-01',null),('2018-12-01',null),
('2019-01-01',null),('2019-02-01',null),('2019-03-01',null),('2019-04-01',null),('2019-05-01',null),('2019-06-01',null),('2019-07-01',null),('2019-08-01',null),('2019-09-01',null),('2019-10-01',null),('2019-11-01',null),('2019-12-01',null);
-- will leave as an exercise for you to add more years. Good idea to start, 10 in either direction, at least.
update ymHelper set dtEnd=LAST_DAY(dtBegin); -- data patch. Confirmed leap years.
alter table ymHelper modify dtEnd date not null; -- there, ugly patch above worked fine. Can forget it ever happened (until you add rows)
-- show create table ymHelper; -- this confirms that dtEnd is not null
तो वह एक सहायक तालिका है। इसे एक बार सेट करें और कुछ वर्षों के लिए भूल जाएं
नोट :उपरोक्त अद्यतन stmt को चलाना न भूलें
आपकी क्वेरी के लिए त्वरित परीक्षण
SELECT DATE_FORMAT(ymH.dtBegin,'%b %Y') as month,
ifnull(COUNT(wr.the_date),0) as total_records,@soloName as full_name
FROM ymHelper ymH
left join workerRecords wr
on wr.the_date between ymH.dtBegin and ymH.dtEnd
and wr.staff_no = 1 and wr.the_date between '2016-04-01' and '2016-07-31'
LEFT JOIN workers w on w.staff_no = wr.staff_no
cross join (select @soloName:=full_name from workers where staff_no=1) xDerived
WHERE ymH.dtBegin between '2016-04-01' and '2016-07-31'
GROUP BY ymH.dtBegin
order by ymH.dtBegin;
+----------+---------------+---------------+
| month | total_records | full_name |
+----------+---------------+---------------+
| Apr 2016 | 1 | David Higgins |
| May 2016 | 1 | David Higgins |
| Jun 2016 | 2 | David Higgins |
| Jul 2016 | 0 | David Higgins |
+----------+---------------+---------------+
यह बढ़िया काम करता है। पहली mysql टेबल हेल्पर टेबल है। कार्यकर्ता रिकॉर्ड (शून्य के लिए अनुमति) लाने के लिए बाएं शामिल हों। चलो यहीं रुकते हैं। आख़िरकार आपके प्रश्न का यही उद्देश्य था:डेटा अनुपलब्ध . अंत में कार्यकर्ता तालिका एक क्रॉस में शामिल हो जाती है।
cross join
एक वैरिएबल को इनिशियलाइज़ करना है (@soloName
) वह कार्यकर्ता का नाम है। जबकि आपके द्वारा अनुरोधित अनुपलब्ध तिथियों की शून्य स्थिति को ifnull()
. के माध्यम से ठीक से उठाया गया है फ़ंक्शन 0 लौट रहा है, हमारे पास एक कार्यकर्ता के नाम के लिए वह विलासिता नहीं है। यह cross join
. को बाध्य करता है ।
एक क्रॉस जॉइन एक कार्टेशियन उत्पाद है। लेकिन चूंकि यह एक ही पंक्ति है, इसलिए हम कार्टेशियन के साथ होने वाली सामान्य समस्याओं से ग्रस्त नहीं होते हैं, जिससे परिणाम सेट में कई पंक्तियों का रास्ता बन जाता है। वैसे भी, यह काम करता है।
लेकिन यहां एक समस्या है:6 स्थानों पर मूल्यों को बनाए रखना और प्लग इन करना बहुत कठिन है जैसा कि देखा जा सकता है। तो इसके लिए एक संग्रहित खरीद के नीचे विचार करें।
संग्रहीत प्रक्रिया
drop procedure if exists getOneWorkersRecCount;
DELIMITER $$
create procedure getOneWorkersRecCount
(pStaffNo int, pBeginDt date, pEndDt date)
BEGIN
SELECT DATE_FORMAT(ymH.dtBegin,'%b %Y') as month,ifnull(COUNT(wr.the_date),0) as total_records,@soloName as full_name
FROM ymHelper ymH
left join workerRecords wr
on wr.the_date between ymH.dtBegin and ymH.dtEnd
and wr.staff_no = pStaffNo and wr.the_date between pBeginDt and pEndDt
LEFT JOIN workers w on w.staff_no = wr.staff_no
cross join (select @soloName:=full_name from workers where staff_no=pStaffNo) xDerived
WHERE ymH.dtBegin between pBeginDt and pEndDt
GROUP BY ymH.dtBegin
order by ymH.dtBegin;
END$$
DELIMITER ;
संग्रहीत खरीद का कई बार परीक्षण करें
call getOneWorkersRecCount(1,'2016-04-01','2016-06-09');
call getOneWorkersRecCount(1,'2016-04-01','2016-06-10');
call getOneWorkersRecCount(1,'2016-04-01','2016-07-01');
call getOneWorkersRecCount(2,'2016-02-01','2016-11-01');
आह, काम करना बहुत आसान है (PHP, c#, Java में, आप इसे नाम दें)। पसंद आपकी है, संग्रहित खरीद है या नहीं।
बोनस संग्रहित प्रक्रिया
drop procedure if exists getAllWorkersRecCount;
DELIMITER $$
create procedure getAllWorkersRecCount
(pBeginDt date, pEndDt date)
BEGIN
SELECT DATE_FORMAT(ymH.dtBegin,'%b %Y') as month,ifnull(COUNT(wr.the_date),0) as total_records,w.staff_no,w.full_name
FROM ymHelper ymH
cross join workers w
left join workerRecords wr
on wr.the_date between ymH.dtBegin and ymH.dtEnd
and wr.staff_no = w.staff_no and wr.the_date between pBeginDt and pEndDt
-- LEFT JOIN workers w on w.staff_no = wr.staff_no
-- cross join (select @soloName:=full_name from workers ) xDerived
WHERE ymH.dtBegin between pBeginDt and pEndDt
GROUP BY ymH.dtBegin,w.staff_no,w.full_name
order by ymH.dtBegin,w.staff_no;
END$$
DELIMITER ;
इसका त्वरित परीक्षण
call getAllWorkersRecCount('2016-03-01','2016-08-01');
+----------+---------------+----------+-----------------+
| month | total_records | staff_no | full_name |
+----------+---------------+----------+-----------------+
| Mar 2016 | 0 | 1 | David Higgins |
| Mar 2016 | 1 | 2 | Sally O'Riordan |
| Apr 2016 | 1 | 1 | David Higgins |
| Apr 2016 | 0 | 2 | Sally O'Riordan |
| May 2016 | 1 | 1 | David Higgins |
| May 2016 | 0 | 2 | Sally O'Riordan |
| Jun 2016 | 2 | 1 | David Higgins |
| Jun 2016 | 0 | 2 | Sally O'Riordan |
| Jul 2016 | 0 | 1 | David Higgins |
| Jul 2016 | 1 | 2 | Sally O'Riordan |
| Aug 2016 | 0 | 1 | David Higgins |
| Aug 2016 | 0 | 2 | Sally O'Riordan |
+----------+---------------+----------+-----------------+
द टेकअवे
हेल्पर टेबल्स का इस्तेमाल दशकों से किया जा रहा है। उनका उपयोग करने से डरो या शर्मिंदा मत हो। वास्तव में, उनके बिना कुछ विशेष कार्य करने की कोशिश करना कई बार लगभग असंभव होता है।