CREATE DEFINER = 'root'@'localhost'
PROCEDURE test.GetHierarchyUsers(IN StartKey INT)
BEGIN
-- prepare a hierarchy level variable
SET @hierlevel := 00000;
-- prepare a variable for total rows so we know when no more rows found
SET @lastRowCount := 0;
-- pre-drop temp table
DROP TABLE IF EXISTS MyHierarchy;
-- now, create it as the first level you want...
-- ie: a specific top level of all "no parent" entries
-- or parameterize the function and ask for a specific "ID".
-- add extra column as flag for next set of ID's to load into this.
CREATE TABLE MyHierarchy AS
SELECT U.ID
, U.Parent
, U.`name`
, 00 AS IDHierLevel
, 00 AS AlreadyProcessed
FROM
Users U
WHERE
U.ID = StartKey;
-- how many rows are we starting with at this tier level
-- START the cycle, only IF we found rows...
SET @lastRowCount := FOUND_ROWS();
-- we need to have a "key" for updates to be applied against,
-- otherwise our UPDATE statement will nag about an unsafe update command
CREATE INDEX MyHier_Idx1 ON MyHierarchy (IDHierLevel);
-- NOW, keep cycling through until we get no more records
WHILE @lastRowCount > 0
DO
UPDATE MyHierarchy
SET
AlreadyProcessed = 1
WHERE
IDHierLevel = @hierLevel;
-- NOW, load in all entries found from full-set NOT already processed
INSERT INTO MyHierarchy
SELECT DISTINCT U.ID
, U.Parent
, U.`name`
, @hierLevel + 1 AS IDHierLevel
, 0 AS AlreadyProcessed
FROM
MyHierarchy mh
JOIN Users U
ON mh.Parent = U.ID
WHERE
mh.IDHierLevel = @hierLevel;
-- preserve latest count of records accounted for from above query
-- now, how many acrual rows DID we insert from the select query
SET @lastRowCount := ROW_COUNT();
-- only mark the LOWER level we just joined against as processed,
-- and NOT the new records we just inserted
UPDATE MyHierarchy
SET
AlreadyProcessed = 1
WHERE
IDHierLevel = @hierLevel;
-- now, update the hierarchy level
SET @hierLevel := @hierLevel + 1;
END WHILE;
-- return the final set now
SELECT *
FROM
MyHierarchy;
-- and we can clean-up after the query of data has been selected/returned.
-- drop table if exists MyHierarchy;
END
यह बोझिल प्रकट हो सकता है, लेकिन यह उपयोग करने के लिए, कर
call GetHierarchyUsers(5);
(या जो कुछ भी कुंजी आईडी आप के लिए श्रेणीबद्ध पेड़ लगाना चाहते हैं)।
आधार एक कुंजी के साथ शुरू करना है जिसके साथ आप काम कर रहे हैं। फिर, उपयोगकर्ता तालिका में फिर से जुड़ने के आधार के रूप में इसका उपयोग करें, लेकिन पहली प्रविष्टि के माता-पिता आईडी पर आधारित है। एक बार मिलने के बाद, अस्थायी तालिका को अद्यतन करें और उस चक्र के लिए अगले चक्र पर फिर से शामिल न हों। फिर तब तक जारी रखें जब तक कोई और "पैरेंट" आईडी कुंजी नहीं मिल पाती।
यह अभिभावकों तक रिकॉर्ड के पूरे पदानुक्रम को वापस कर देगा चाहे कितना घोंसला घोंसला हो। हालांकि, अगर आप केवल अंतिम माता-पिता चाहते हैं, तो आप @hierlevel वैरिएबल का उपयोग केवल फ़ाइल में जोड़े गए नवीनतम संस्करण को वापस करने के लिए कर सकते हैं, या ऑर्डर बाय और LIMIT 1
विधेयक Karwin सही है। MySQL में 'SQL सर्वर' जैसी रिकर्सिव क्वेरी के लिए कोई फ़ंक्शन नहीं है क्योंकि इसमें 'CTE' है। लेकिन पुनरावृत्ति का व्यवहार अभी भी अनुकरण किया जा सकता है। ': डी' –
मुझे नहीं लगता कि आप एक प्रश्न से MySQL में रिकर्सन कर सकते हैं, लेकिन मैंने संग्रहित प्रक्रियाओं के माध्यम से समान पेरेंट पदानुक्रम पूछताछ की है जो माता-पिता स्तर की तलाश में रहते हैं जब तक कोई और अभिभावक प्रविष्टियां नहीं मिलतीं ... लेकिन किया जाता है समाप्त होने पर हटाए गए एक अस्थायी तालिका के माध्यम से ... क्या यह आपके लिए काम करेगा? – DRapp
@DRapp: शायद यह स्वीकार्य होगा। किसी भी तरह से सीखना मजेदार होगा –