একটি সিটিই হায়ারার্কি অনুকূল করা


15

নীচে আপডেট করুন

অ্যাকাউন্টগুলির একটি শ্রেণিবিন্যাস উপস্থাপনের জন্য আমার কাছে একটি সাধারণ অ্যাক্ট / প্যারেন্ট অ্যাকাউন্ট আর্কিটেকচারের সাথে অ্যাকাউন্টগুলির একটি টেবিল রয়েছে (এসকিউএল সার্ভার ২০১২)। শ্রেণিবদ্ধ হ্যাশ করতে আমি সিটিই ব্যবহার করে একটি ভিউ তৈরি করেছি এবং সামগ্রিকভাবে এটি সুন্দরভাবে এবং উদ্দেশ্য হিসাবে কাজ করে। আমি যে কোনও স্তরের স্তরক্রমকে জিজ্ঞাসা করতে পারি, এবং শাখাগুলি সহজেই দেখতে পারি।

একটি ব্যবসায়ের যুক্তিযুক্ত ক্ষেত্র রয়েছে যা শ্রেণিবিন্যাসের ক্রিয়াকলাপ হিসাবে ফিরে আসতে হবে। প্রতিটি অ্যাকাউন্ট রেকর্ডের একটি ক্ষেত্র ব্যবসায়ের আকার বর্ণনা করে (আমরা এটিকে গ্রাহককাউন্ট বলব)। সম্পূর্ণ শাখা থেকে গ্রাহককাউন্টটি রোল আপ করার জন্য আমার যে যুক্তিটি প্রকাশ করতে হবে সে বিষয়ে রিপোর্ট করতে হবে। অন্য কথায়, একটি অ্যাকাউন্ট দেওয়া হয়েছে, আমাকে ক্রমবর্ধমান বরাবর অ্যাকাউন্টের নীচে প্রতিটি শাখার প্রতিটি শিশু সহ সেই অ্যাকাউন্টের গ্রাহকসংখ্যার মানগুলি যোগ করতে হবে।

আমি সিটিইর মধ্যে নির্মিত একটি শ্রেণিবদ্ধ ক্ষেত্রটি ব্যবহার করে ক্ষেত্রটি সফলভাবে গণনা করেছি, যা acct4.acct3.acct2.acct1 এর মতো দেখাচ্ছে। আমি যে সমস্যার মধ্যে যাচ্ছি তা হ'ল এটি দ্রুত চালানো। এই এক গণনা করা ক্ষেত্র ছাড়া ক্যোয়ারীটি 3 সেকেন্ডে চলে। আমি যখন গণনা করা ক্ষেত্রটি যুক্ত করি, তখন এটি 4 মিনিটের ক্যোয়ারিতে পরিণত হয়।

এখানে আমি সেরা সংস্করণটি উপস্থিত করতে সক্ষম হয়েছি যা সঠিক ফলাফলগুলি দেয়। পারফরম্যান্সের জন্য এত বড় ত্যাগ ছাড়াই আমি কীভাবে এই দৃশ্যটির পুনর্গঠন করতে পারি সে সম্পর্কে ধারণাগুলি খুঁজছি।

আমি বুঝতে পারি যে এটি ধীরে ধীরে চলেছে (যেখানে যেখানে ক্লজটিতে প্রিডিকেট গণনা করা দরকার) তবে আমি এটির কাঠামোগত করার অন্য কোনও উপায় সম্পর্কে ভাবতে পারি না এবং এখনও একই ফলাফল পেতে পারি।

একটি টেবিল তৈরি করতে এবং সিটিইটি ঠিক আমার পরিবেশে যেমন কাজ করে ঠিক তেমন করার জন্য এখানে কিছু নমুনা কোড রয়েছে।

Use Tempdb
go
CREATE TABLE dbo.Account
(
   Acctid varchar(1) NOT NULL
    , Name varchar(30) NULL
    , ParentId varchar(1) NULL
    , CustomerCount int NULL
);

INSERT Account
SELECT 'A','Best Bet',NULL,21  UNION ALL
SELECT 'B','eStore','A',30 UNION ALL
SELECT 'C','Big Bens','B',75 UNION ALL
SELECT 'D','Mr. Jimbo','B',50 UNION ALL
SELECT 'E','Dr. John','C',100 UNION ALL
SELECT 'F','Brick','A',222 UNION ALL
SELECT 'G','Mortar','C',153 ;


With AccountHierarchy AS

(                                                                           --Root values have no parent
    SELECT
        Root.AcctId                                         AccountId
        , Root.Name                                         AccountName
        , Root.ParentId                                     ParentId
        , 1                                                 HierarchyLevel  
        , cast(Root.Acctid as varchar(4000))                IdHierarchy     --highest parent reads right to left as in id3.Acctid2.Acctid1
        , cast(replace(Root.Name,'.','') as varchar(4000))  NameHierarchy   --highest parent reads right to left as in name3.name2.name1 (replace '.' so name parse is easy in last step)
        , cast(Root.Acctid as varchar(4000))                HierarchySort   --reverse of above, read left to right name1.name2.name3 for sorting on reporting only
        , cast(Root.Name as varchar(4000))                  HierarchyLabel  --use for labels on reporting only, indents names under sorted hierarchy
        , Root.CustomerCount                                CustomerCount   

    FROM 
        tempdb.dbo.account Root

    WHERE
        Root.ParentID is null

    UNION ALL

    SELECT
        Recurse.Acctid                                      AccountId
        , Recurse.Name                                      AccountName
        , Recurse.ParentId                                  ParentId
        , Root.HierarchyLevel + 1                           HierarchyLevel  --next level in hierarchy
        , cast(cast(recurse.Acctid as varchar(40)) + '.' + Root.IdHierarchy as varchar(4000))   IdHierarchy --cast because in real system this is a uniqueidentifier type needs converting
        , cast(replace(recurse.Name,'.','') + '.' + Root.NameHierarchy as varchar(4000)) NameHierarchy  --replace '.' for parsing in last step, cast to make room for lots of sub levels down the hierarchy
        , cast(Root.AccountName + '.' + Recurse.Name as varchar(4000)) HierarchySort    
        , cast(space(root.HierarchyLevel * 4) + Recurse.Name as varchar(4000)) HierarchyLabel
        , Recurse.CustomerCount                             CustomerCount

    FROM
        tempdb.dbo.account Recurse INNER JOIN
        AccountHierarchy Root on Root.AccountId = Recurse.ParentId
)


SELECT
    hier.AccountId
    , Hier.AccountName
    , hier.ParentId
    , hier.HierarchyLevel
    , hier.IdHierarchy
    , hier.NameHierarchy
    , hier.HierarchyLabel
    , parsename(hier.IdHierarchy,1) Acct1Id
    , parsename(hier.NameHierarchy,1) Acct1Name     --This is why we stripped out '.' during recursion
    , parsename(hier.IdHierarchy,2) Acct2Id
    , parsename(hier.NameHierarchy,2) Acct2Name
    , parsename(hier.IdHierarchy,3) Acct3Id
    , parsename(hier.NameHierarchy,3) Acct3Name
    , parsename(hier.IdHierarchy,4) Acct4Id
    , parsename(hier.NameHierarchy,4) Acct4Name
    , hier.CustomerCount

    /* fantastic up to this point. Next block of code is what causes problem. 
        Logic of code is "sum of CustomerCount for this location and all branches below in this branch of hierarchy"
        In live environment, goes from taking 3 seconds to 4 minutes by adding this one calc */

    , (
        SELECT  
            sum(children.CustomerCount)
        FROM
            AccountHierarchy Children
        WHERE
            hier.IdHierarchy = right(children.IdHierarchy, (1 /*length of id field*/ * hier.HierarchyLevel) + hier.HierarchyLevel - 1 /*for periods inbetween ids*/)
            --"where this location's idhierarchy is within child idhierarchy"
            --previously tried a charindex(hier.IdHierarchy,children.IdHierarchy)>0, but that performed even worse
        ) TotalCustomerCount
FROM
    AccountHierarchy hier

ORDER BY
    hier.HierarchySort


drop table tempdb.dbo.Account

11/20/2013 আপডেট

প্রস্তাবিত কয়েকটি সমাধান আমার রস প্রবাহিত করল এবং আমি একটি নতুন পদ্ধতির চেষ্টা করে যা কাছাকাছি আসে তবে একটি নতুন / ভিন্ন প্রতিবন্ধকতার পরিচয় দেয়। সত্যি বলতে কী, আমি জানি না এটি একটি পৃথক পোস্টের ওয়ারেন্ট দেয় কি না, তবে এটি এই সমস্যার সমাধানের সাথে সম্পর্কিত related

আমি যা স্থির করেছিলাম তা হ'ল যে পরিমাণটি (গ্রাহক হিসাব) তৈরি করছিল তা হ'ল শীর্ষস্থান থেকে শুরু হওয়া এবং নিম্নগঠিত হওয়া শ্রেণিবিন্যাসের প্রসঙ্গে শিশুদের সনাক্তকরণ। সুতরাং আমি "নীচে থেকে অন্য কোনও অ্যাকাউন্টের পিতামাতার নয় এমন অ্যাকাউন্টগুলি" দ্বারা সংজ্ঞায়িত রুটটি ব্যবহার করে পিছন দিকে যোগ দেওয়া (রুট.পিয়েনটসিটিটিড = recurse.acctid) দ্বারা নীচে থেকে তৈরি হওয়া একটি শ্রেণিবিন্যাস তৈরি করে শুরু করেছি root

পুনরাবৃত্তি হওয়ার সাথে সাথে আমি কেবলমাত্র সন্তানের গ্রাহকের গণনা পিতামাতার সাথে যুক্ত করতে পারি। আমার কীভাবে রিপোর্টিং এবং স্তরের প্রয়োজন তার কারণে আমি নীচের অংশে উপরে নীচের অংশেও এই নীচে আপ সিটি করছি, তারপরে কেবল অ্যাকাউন্ট আইডির মাধ্যমে তাদের সাথে যোগদান করছি। এই পন্থাটি মূল বাহ্যিক ক্যোয়ারী গ্রাহকগণের চেয়ে অনেক দ্রুত গতিতে পরিণত হয়েছে, তবে আমি কয়েকটি প্রতিবন্ধকতার মধ্যে পড়েছিলাম।

প্রথমত, আমি একাধিক বাচ্চার পিতামাতার অ্যাকাউন্টগুলির জন্য অনবচ্ছতভাবে সদৃশ গ্রাহক গণনা ক্যাপচার করছিলাম। কিছু স্বীকৃত ব্যক্তির জন্য আমি দ্বিগুণ বা ট্রিপল কাউন্টিং গ্রাহক সংখ্যা ছিলাম, সেখানে শিশু সংখ্যা ছিল by আমার সমাধানটি ছিল আরও একটি সিটি তৈরি করা যা কোনও অ্যাক্টের কয়টি নোড রয়েছে তা গণনা করে এবং পুনরাবৃত্তির সময় acct.customercount বিভক্ত করে, তাই আমি যখন পুরো শাখাটি যুক্ত করি তখন অ্যাক্টটি দ্বিগুণ গণনা করা হচ্ছে না।

সুতরাং এই মুহুর্তে, এই নতুন সংস্করণের ফলাফলগুলি সঠিক নয়, তবে আমি জানি কেন। নীচের অংশে সিটি সদৃশ তৈরি করছে। পুনরাবৃত্তিটি পাস করার পরে, এটি অ্যাকাউন্টের সারণীতে কোনও অ্যাকাউন্টে সন্তানের মূলের (নীচের স্তরের শিশুদের) যে কোনও কিছু সন্ধান করে। তৃতীয় পুনরাবৃত্তিতে, এটি দ্বিতীয়টিতে এটি করে একই অ্যাকাউন্টগুলি তুলে নিয়ে আবার এগুলিতে রাখে।

সিটি কীভাবে করবেন তা সম্পর্কে ধারণা, বা এটি কোনও অন্যান্য ধারণা প্রবাহিত করে?

Use Tempdb
go


CREATE TABLE dbo.Account
(
    Acctid varchar(1) NOT NULL
    , Name varchar(30) NULL
    , ParentId varchar(1) NULL
    , CustomerCount int NULL
);

INSERT Account
SELECT 'A','Best Bet',NULL,1  UNION ALL
SELECT 'B','eStore','A',2 UNION ALL
SELECT 'C','Big Bens','B',3 UNION ALL
SELECT 'D','Mr. Jimbo','B',4 UNION ALL
SELECT 'E','Dr. John','C',5 UNION ALL
SELECT 'F','Brick','A',6 UNION ALL
SELECT 'G','Mortar','C',7 ;



With AccountHierarchy AS

(                                                                           --Root values have no parent
    SELECT
        Root.AcctId                                         AccountId
        , Root.Name                                         AccountName
        , Root.ParentId                                     ParentId
        , 1                                                 HierarchyLevel  
        , cast(Root.Acctid as varchar(4000))                IdHierarchy     --highest parent reads right to left as in id3.Acctid2.Acctid1
        , cast(replace(Root.Name,'.','') as varchar(4000))  NameHierarchy   --highest parent reads right to left as in name3.name2.name1 (replace '.' so name parse is easy in last step)
        , cast(Root.Acctid as varchar(4000))                HierarchySort   --reverse of above, read left to right name1.name2.name3 for sorting on reporting only
        , cast(Root.Acctid as varchar(4000))                HierarchyMatch 
        , cast(Root.Name as varchar(4000))                  HierarchyLabel  --use for labels on reporting only, indents names under sorted hierarchy
        , Root.CustomerCount                                CustomerCount   

    FROM 
        tempdb.dbo.account Root

    WHERE
        Root.ParentID is null

    UNION ALL

    SELECT
        Recurse.Acctid                                      AccountId
        , Recurse.Name                                      AccountName
        , Recurse.ParentId                                  ParentId
        , Root.HierarchyLevel + 1                           HierarchyLevel  --next level in hierarchy
        , cast(cast(recurse.Acctid as varchar(40)) + '.' + Root.IdHierarchy as varchar(4000))   IdHierarchy --cast because in real system this is a uniqueidentifier type needs converting
        , cast(replace(recurse.Name,'.','') + '.' + Root.NameHierarchy as varchar(4000)) NameHierarchy  --replace '.' for parsing in last step, cast to make room for lots of sub levels down the hierarchy
        , cast(Root.AccountName + '.' + Recurse.Name as varchar(4000)) HierarchySort    
        , CAST(CAST(Root.HierarchyMatch as varchar(40)) + '.' 
            + cast(recurse.Acctid as varchar(40))   as varchar(4000))   HierarchyMatch
        , cast(space(root.HierarchyLevel * 4) + Recurse.Name as varchar(4000)) HierarchyLabel
        , Recurse.CustomerCount                             CustomerCount

    FROM
        tempdb.dbo.account Recurse INNER JOIN
        AccountHierarchy Root on Root.AccountId = Recurse.ParentId
)

, Nodes as
(   --counts how many branches are below for any account that is parent to another
    select
        node.ParentId Acctid
        , cast(count(1) as float) Nodes
    from AccountHierarchy  node
    group by ParentId
)

, BottomUp as
(   --creates the hierarchy starting at accounts that are not parent to any other
    select
        Root.Acctid
        , root.ParentId
        , cast(isnull(root.customercount,0) as float) CustomerCount
    from
        tempdb.dbo.Account Root
    where
        not exists ( select 1 from tempdb.dbo.Account OtherAccts where root.Acctid = OtherAccts.ParentId)

    union all

    select
        Recurse.Acctid
        , Recurse.ParentId
        , root.CustomerCount + cast ((isnull(recurse.customercount,0) / nodes.nodes) as float) CustomerCount
        -- divide the recurse customercount by number of nodes to prevent duplicate customer count on accts that are parent to multiple children, see customercount cte next
    from
        tempdb.dbo.Account Recurse inner join 
        BottomUp Root on root.ParentId = recurse.acctid inner join
        Nodes on nodes.Acctid = recurse.Acctid
)

, CustomerCount as
(
    select
        sum(CustomerCount) TotalCustomerCount
        , hier.acctid
    from
        BottomUp hier
    group by 
        hier.Acctid
)


SELECT
    hier.AccountId
    , Hier.AccountName
    , hier.ParentId
    , hier.HierarchyLevel
    , hier.IdHierarchy
    , hier.NameHierarchy
    , hier.HierarchyLabel
    , hier.hierarchymatch
    , parsename(hier.IdHierarchy,1) Acct1Id
    , parsename(hier.NameHierarchy,1) Acct1Name     --This is why we stripped out '.' during recursion
    , parsename(hier.IdHierarchy,2) Acct2Id
    , parsename(hier.NameHierarchy,2) Acct2Name
    , parsename(hier.IdHierarchy,3) Acct3Id
    , parsename(hier.NameHierarchy,3) Acct3Name
    , parsename(hier.IdHierarchy,4) Acct4Id
    , parsename(hier.NameHierarchy,4) Acct4Name
    , hier.CustomerCount

    , customercount.TotalCustomerCount

FROM
    AccountHierarchy hier inner join
    CustomerCount on customercount.acctid = hier.accountid

ORDER BY
    hier.HierarchySort 



drop table tempdb.dbo.Account

1
টেম্প টেবিল থেকে জিজ্ঞাসা করে আপনি হিসাবরক্ষণের সিটিইর ফলাফলগুলি কোনও টেম্প টেবিলের (আইডিয়ায়ার্কির উপর সূচিত) রেখে দেওয়ার চেষ্টা করেছেন? আপনি সিটিইর বাস্তবায়নের যে পথে চালিত হতে পারেন; এটা সম্ভব যে আপনি একবার সিটিইতে প্রতিটি সারির জন্য পুরো সিটিই চালাচ্ছেন।
জন বোলিনিউ

1
অন্তর্নিহিত টেবিলের সূচকগুলি কী কী?
মাইক ওয়ালশ

1
এবং আসল টেবিলের মধ্যে কত সারি?
মাইক ওয়ালশ

2
@ ম্যাক্স ভার্নন ধন্যবাদ বেশি পোস্ট করা হয়নি, তবে স্পষ্টতই অস্পষ্ট প্রশ্নের উত্তরগুলির মানের পার্থক্য দেখুন।
লিভার.লারসন

@ জোনবউলাইনো আমি টেম্প টেবিলগুলির সাথে কিছু চেষ্টা করার বিষয়টি বিবেচনা করেছি, তবে বিশেষত এটি একটি দৃষ্টিভঙ্গি হিসাবে চালানোর চেষ্টা করছি, যা টেম্প টেবিলগুলিকে অন্তর্ভুক্ত করে। আপনার শেষ দৃser়তার জন্য কীভাবে কাছাকাছি যাবেন বা পরীক্ষা করবেন সে সম্পর্কে কোনও ধারণা?
লিভার.লারসন

উত্তর:


6

সম্পাদনা করুন: এটি দ্বিতীয় প্রচেষ্টা

@ ম্যাক্স ভার্ননের উত্তরের উপর ভিত্তি করে, এখানে একটি ইনলাইন সাবকোয়ারির ভিতরে সিটিই ব্যবহারকে বাইপাস করার একটি উপায়, যা সিটিইতে স্ব-যোগদানের মতো এবং আমি ধারণা করি যে দুর্বল দক্ষতার কারণ। এটি কেবলমাত্র এসকিউএল-সার্ভারের 2012 সংস্করণে উপলব্ধ বিশ্লেষণমূলক ফাংশন ব্যবহার করে। এসকিউএল-ফিডল পরীক্ষিত

এই অংশটি পড়া থেকে বাদ দেওয়া যেতে পারে, এটি ম্যাক্সের উত্তর থেকে একটি অনুলিপি:

;With AccountHierarchy AS
(                                                                           
    SELECT
        Root.AcctId                                         AccountId
        , Root.Name                                         AccountName
        , Root.ParentId                                     ParentId
        , 1                                                 HierarchyLevel  
        , cast(Root.Acctid as varchar(4000))                IdHierarchyMatch        
        , cast(Root.Acctid as varchar(4000))                IdHierarchy
        , cast(replace(Root.Name,'.','') as varchar(4000))  NameHierarchy   
        , cast(Root.Acctid as varchar(4000))                HierarchySort
        , cast(Root.Name as varchar(4000))                  HierarchyLabel          ,
        Root.CustomerCount                                  CustomerCount   

    FROM 
        account Root

    WHERE
        Root.ParentID is null

    UNION ALL

    SELECT
        Recurse.Acctid                                      AccountId
        , Recurse.Name                                      AccountName
        , Recurse.ParentId                                  ParentId
        , Root.HierarchyLevel + 1                           HierarchyLevel
        , CAST(CAST(Root.IdHierarchyMatch as varchar(40)) + '.' 
            + cast(recurse.Acctid as varchar(40))   as varchar(4000))   IdHierarchyMatch
        , cast(cast(recurse.Acctid as varchar(40)) + '.' 
            + Root.IdHierarchy  as varchar(4000))           IdHierarchy
        , cast(replace(recurse.Name,'.','') + '.' 
            + Root.NameHierarchy as varchar(4000))          NameHierarchy
        , cast(Root.AccountName + '.' 
            + Recurse.Name as varchar(4000))                HierarchySort   
        , cast(space(root.HierarchyLevel * 4) 
            + Recurse.Name as varchar(4000))                HierarchyLabel
        , Recurse.CustomerCount                             CustomerCount
    FROM
        account Recurse INNER JOIN
        AccountHierarchy Root on Root.AccountId = Recurse.ParentId
)

এখানে আমরা ব্যবহার করে সিটিইর সারিগুলি অর্ডার করি IdHierarchyMatchএবং আমরা সারি সংখ্যা এবং একটি চলমান মোট গণনা করি (পরবর্তী সারি থেকে শেষ অবধি) ulate

, cte1 AS 
(
SELECT
    h.AccountId
    , h.AccountName
    , h.ParentId
    , h.HierarchyLevel
    , h.IdHierarchy
    , h.NameHierarchy
    , h.HierarchyLabel
    , parsename(h.IdHierarchy,1) Acct1Id
    , parsename(h.NameHierarchy,1) Acct1Name
    , parsename(h.IdHierarchy,2) Acct2Id
    , parsename(h.NameHierarchy,2) Acct2Name
    , parsename(h.IdHierarchy,3) Acct3Id
    , parsename(h.NameHierarchy,3) Acct3Name
    , parsename(h.IdHierarchy,4) Acct4Id
    , parsename(h.NameHierarchy,4) Acct4Name
    , h.CustomerCount
    , h.HierarchySort
    , h.IdHierarchyMatch
        , Rn = ROW_NUMBER() OVER 
                  (ORDER BY h.IdHierarchyMatch)
        , RunningCustomerCount = COALESCE(
            SUM(h.CustomerCount)
            OVER
              (ORDER BY h.IdHierarchyMatch
               ROWS BETWEEN 1 FOLLOWING
                        AND UNBOUNDED FOLLOWING)
          , 0) 
FROM
    AccountHierarchy AS h  
)

তারপরে আমাদের আরও একটি মধ্যবর্তী সিটিই রয়েছে যেখানে আমরা পূর্বের চলমান মোট সংখ্যা এবং সারি সংখ্যাগুলি ব্যবহার করি - মূলত গাছের কাঠামোর শাখার শেষ প্রান্তটি কোথায় তা খুঁজে বের করতে:

, cte2 AS
(
SELECT
    cte1.*
    , rn3  = LAST_VALUE(Rn) OVER 
               (PARTITION BY Acct1Id, Acct2Id, Acct3Id 
                ORDER BY Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING)       
    , rn2  = LAST_VALUE(Rn) OVER 
               (PARTITION BY Acct1Id, Acct2Id 
                ORDER BY Acct3Id, Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
    , rn1  = LAST_VALUE(Rn) OVER 
               (PARTITION BY Acct1Id 
                ORDER BY Acct2Id, Acct3Id, Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
    , rcc3 = LAST_VALUE(RunningCustomerCount) OVER 
               (PARTITION BY Acct1Id, Acct2Id, Acct3Id 
                ORDER BY Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING)       
    , rcc2 = LAST_VALUE(RunningCustomerCount) OVER 
               (PARTITION BY Acct1Id, Acct2Id 
                ORDER BY Acct3Id, Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
    , rcc1 = LAST_VALUE(RunningCustomerCount) OVER 
               (PARTITION BY Acct1Id 
                ORDER BY Acct2Id, Acct3Id, Acct4Id
                ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
FROM
    cte1 
) 

এবং অবশেষে আমরা শেষ অংশটি নির্মাণ করি:

SELECT
    hier.AccountId
    , hier.AccountName
    ---                        -- columns skipped 
    , hier.CustomerCount

    , TotalCustomerCount = hier.CustomerCount
        + hier.RunningCustomerCount 
        - ca.LastRunningCustomerCount

    , hier.HierarchySort
    , hier.IdHierarchyMatch
FROM
    cte2 hier
  OUTER APPLY
    ( SELECT  LastRunningCustomerCount, Rn
      FROM
      ( SELECT LastRunningCustomerCount
              = RunningCustomerCount, Rn
        FROM (SELECT NULL a) x  WHERE 4 <= HierarchyLevel 
      UNION ALL
        SELECT rcc3, Rn3
        FROM (SELECT NULL a) x  WHERE 3 <= HierarchyLevel 
      UNION ALL
        SELECT rcc2, Rn2 
        FROM (SELECT NULL a) x  WHERE 2 <= HierarchyLevel 
      UNION ALL
        SELECT rcc1, Rn1
        FROM (SELECT NULL a) x  WHERE 1 <= HierarchyLevel 
      ) x
      ORDER BY Rn 
      OFFSET 0 ROWS
      FETCH NEXT 1 ROWS ONLY
      ) ca
ORDER BY
    hier.HierarchySort ; 

এবং সরলকরণ, cte1উপরের কোডটির মতো একই ব্যবহার করে । এ টেস্ট এসকিউএল বেহালার-2 । দয়া করে মনে রাখবেন যে উভয় সমাধান আপনার গাছে সর্বাধিক চারটি স্তর রয়েছে এই ধারণার অধীনে কাজ করে:

SELECT
    hier.AccountId
    ---                      -- skipping rows
    , hier.CustomerCount

    , TotalCustomerCount = CustomerCount
        + RunningCustomerCount 
        - CASE HierarchyLevel
            WHEN 4 THEN RunningCustomerCount
            WHEN 3 THEN LAST_VALUE(RunningCustomerCount) OVER 
                   (PARTITION BY Acct1Id, Acct2Id, Acct3Id 
                    ORDER BY Acct4Id
                    ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING)       
            WHEN 2 THEN LAST_VALUE(RunningCustomerCount) OVER 
                   (PARTITION BY Acct1Id, Acct2Id 
                    ORDER BY Acct3Id, Acct4Id
                    ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
            WHEN 1 THEN LAST_VALUE(RunningCustomerCount) OVER 
                   (PARTITION BY Acct1Id 
                    ORDER BY Acct2Id, Acct3Id, Acct4Id
                    ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) 
          END

    , hier.HierarchySort
    , hier.IdHierarchyMatch
FROM cte1 AS hier
ORDER BY
    hier.HierarchySort ; 

পুনরাবৃত্ত অংশের জন্য কেবল একটি সিটিই সহ তৃতীয় পন্থা এবং তারপরে কেবল উইন্ডো সমষ্টিগত ফাংশন ( SUM() OVER (...)) ব্যবহার করা যায়, সুতরাং এটি 2005 এর পরবর্তী সংস্করণে কোনও সংস্করণে কাজ করা উচিত। টেস্ট এ এসকিউএল বেহালার -3 এই সমাধান অনুমান, আগের মতই আছে অনুক্রমের গাছ 4 মাত্রা সর্বোচ্চ:

;WITH AccountHierarchy AS
(                                                                           
    SELECT
          AccountId      = Root.AcctId                                         
        , AccountName    = Root.Name                                         
        , ParentId       = Root.ParentId                                     
        , HierarchyLevel = 1                                                   
        , HierarchySort  = CAST(Root.Acctid AS VARCHAR(4000))                
        , HierarchyLabel = CAST(Root.Name AS VARCHAR(4000))                   
        , Acct1Id        = CAST(Root.Acctid AS VARCHAR(4000))                
        , Acct2Id        = CAST(NULL AS VARCHAR(4000))                       
        , Acct3Id        = CAST(NULL AS VARCHAR(4000))                       
        , Acct4Id        = CAST(NULL AS VARCHAR(4000))                       
        , Acct1Name      = CAST(Root.Name AS VARCHAR(4000))                  
        , Acct2Name      = CAST(NULL AS VARCHAR(4000))                       
        , Acct3Name      = CAST(NULL AS VARCHAR(4000))                       
        , Acct4Name      = CAST(NULL AS VARCHAR(4000))                       
        , CustomerCount  = Root.CustomerCount                                   

    FROM 
        account AS Root

    WHERE
        Root.ParentID IS NULL

    UNION ALL

    SELECT
          Recurse.Acctid 
        , Recurse.Name 
        , Recurse.ParentId 
        , Root.HierarchyLevel + 1 
        , CAST(Root.AccountName + '.' 
            + Recurse.Name AS VARCHAR(4000)) 
        , CAST(SPACE(Root.HierarchyLevel * 4) 
            + Recurse.Name AS VARCHAR(4000)) 
        , Root.Acct1Id 
        , CASE WHEN Root.HierarchyLevel = 1 
              THEN cast(Recurse.Acctid AS VARCHAR(4000)) 
              ELSE Root.Acct2Id 
          END 
        , CASE WHEN Root.HierarchyLevel = 2 
              THEN CAST(Recurse.Acctid AS VARCHAR(4000)) 
              ELSE Root.Acct3Id 
          END 
        , CASE WHEN Root.HierarchyLevel = 3 
              THEN CAST(Recurse.Acctid AS VARCHAR(4000)) 
              ELSE Root.Acct4Id 
          END 

        , cast(Root.AccountName as varchar(4000))          
        , CASE WHEN Root.HierarchyLevel = 1 
              THEN CAST(Recurse.Name AS VARCHAR(4000)) 
              ELSE Root.Acct2Name 
          END 
        , CASE WHEN Root.HierarchyLevel = 2 
              THEN CAST(Recurse.Name AS VARCHAR(4000)) 
              ELSE Root.Acct3Name 
          END 
        , CASE WHEN Root.HierarchyLevel = 3 
              THEN CAST(Recurse.Name AS VARCHAR(4000)) 
              ELSE Root.Acct4Name 
          END 
        , Recurse.CustomerCount 
    FROM 
        account AS Recurse INNER JOIN 
        AccountHierarchy AS Root ON Root.AccountId = Recurse.ParentId
)

SELECT
      h.AccountId
    , h.AccountName
    , h.ParentId
    , h.HierarchyLevel
    , IdHierarchy = 
          CAST(COALESCE(h.Acct4Id+'.','') 
               + COALESCE(h.Acct3Id+'.','') 
               + COALESCE(h.Acct2Id+'.','') 
               + h.Acct1Id AS VARCHAR(4000))
    , NameHierarchy = 
          CAST(COALESCE(h.Acct4Name+'.','') 
               + COALESCE(h.Acct3Name+'.','') 
               + COALESCE(h.Acct2Name+'.','') 
               + h.Acct1Name AS VARCHAR(4000))   
    , h.HierarchyLabel
    , h.Acct1Id
    , h.Acct1Name
    , h.Acct2Id
    , h.Acct2Name
    , h.Acct3Id
    , h.Acct3Name
    , h.Acct4Id
    , h.Acct4Name
    , h.CustomerCount
    , TotalCustomerCount =  
          CASE h.HierarchyLevel
            WHEN 4 THEN h.CustomerCount
            WHEN 3 THEN SUM(h.CustomerCount) OVER 
                   (PARTITION BY h.Acct1Id, h.Acct2Id, h.Acct3Id)       
            WHEN 2 THEN SUM(h.CustomerCount) OVER 
                   (PARTITION BY Acct1Id, h.Acct2Id) 
            WHEN 1 THEN SUM(h.CustomerCount) OVER 
                   (PARTITION BY h.Acct1Id) 
          END
    , h.HierarchySort
    , IdHierarchyMatch = 
          CAST(h.Acct1Id 
               + COALESCE('.'+h.Acct2Id,'') 
               + COALESCE('.'+h.Acct3Id,'') 
               + COALESCE('.'+h.Acct4Id,'') AS VARCHAR(4000))   
FROM
    AccountHierarchy AS h  
ORDER BY
    h.HierarchySort ; 

একটি চতুর্থ পন্থা, যা মধ্যবর্তী সিটিই, শ্রেণিবিন্যাসের ক্লোজার টেবিল হিসাবে গণনা করে। এ টেস্ট এসকিউএল বেহালার -4 । সুবিধাটি হ'ল অঙ্কের গণনার জন্য স্তরের সংখ্যার উপর কোনও নিয়ন্ত্রণ নেই।

;WITH AccountHierarchy AS
( 
    -- skipping several line, identical to the 3rd approach above
)

, ClosureTable AS
( 
    SELECT
          AccountId      = Root.AcctId  
        , AncestorId     = Root.AcctId  
        , CustomerCount  = Root.CustomerCount 
    FROM 
        account AS Root

    UNION ALL

    SELECT
          Recurse.Acctid 
        , Root.AncestorId 
        , Recurse.CustomerCount
    FROM 
        account AS Recurse INNER JOIN 
        ClosureTable AS Root ON Root.AccountId = Recurse.ParentId
)

, ClosureGroup AS
(                                                                           
    SELECT
          AccountId           = AncestorId  
        , TotalCustomerCount  = SUM(CustomerCount)                             
    FROM 
        ClosureTable AS a
    GROUP BY
        AncestorId
)

SELECT
      h.AccountId
    , h.AccountName
    , h.ParentId
    , h.HierarchyLevel 
    , h.HierarchyLabel
    , h.CustomerCount
    , cg.TotalCustomerCount 

    , h.HierarchySort
FROM
    AccountHierarchy AS h  
  JOIN
    ClosureGroup AS cg
      ON cg.AccountId = h.AccountId
ORDER BY
    h.HierarchySort ;  

কোডটি সংশোধন করা হয়েছে (এবং লিঙ্কযুক্ত ফিডাল)) উত্তরে একটি WHEN বিকল্প ছিল।
ypercubeᵀᴹ

+1 - আমি 2012 ফাংশনগুলির ব্যবহার পছন্দ করি। আমি এখন অনেক শিখছি!
ম্যাক্স ভার্নন

ঠিক আছে, সবেমাত্র কিছু সময় ডুব দিয়ে কাটিয়েছি এবং বুঝতে পেরেছি যে পারফরম্যান্স দুর্দান্ত, তবে সংখ্যাগুলি মেলে না। আমার মূলটির বিপরীতে ফলাফলগুলি পরীক্ষা করুন। আপনি চলমান মোটের সাথে আপনি কোথায় যাচ্ছেন তা আমি দেখতে পাচ্ছি, তবে লক্ষ্য হিসাবে কাজ করতে কিছু পরিবর্তন করতে হবে এবং আমি সঠিক সমাধানে পৌঁছিনি। আপনার পদ্ধতির সাথে কাজ করার জন্য আমাকে কিছু পশুপাল পাওয়া যায়, তবে এটি এখনও একটি কার্যকর সমাধান নয় not
লিভার.লারসন

ওহ, আমি মনে করি এটি মারাত্মক ভুল। অগ্রহণযোগ্য দয়া করে।
ypercubeᵀᴹ

আমি এটি সংশোধন করার চেষ্টা করেছি। এটি আমার ছোট নমুনার সাথে ঠিক আছে তবে আপনার ডেটা দিয়ে সঠিকতা পরীক্ষা করুন। দক্ষতা সম্পর্কে, আমি কী বলতে পারি, আমরা কেবল পরীক্ষার মাধ্যমে জানতে পারি (যদি না আপনার নাম @ পল হোয়াইট না থাকে)।
ypercubeᵀᴹ

5

আমি বিশ্বাস করি এটি দ্রুত করা উচিত:

;With AccountHierarchy AS
(                                                                           
    SELECT
        Root.AcctId                                         AccountId
        , Root.Name                                         AccountName
        , Root.ParentId                                     ParentId
        , 1                                                 HierarchyLevel  
        , cast(Root.Acctid as varchar(4000))                IdHierarchyMatch        
        , cast(Root.Acctid as varchar(4000))                IdHierarchy
        , cast(replace(Root.Name,'.','') as varchar(4000))  NameHierarchy   
        , cast(Root.Acctid as varchar(4000))                HierarchySort
        , cast(Root.Name as varchar(4000))                  HierarchyLabel          ,
        Root.CustomerCount                                  CustomerCount   

    FROM 
        tempdb.dbo.account Root

    WHERE
        Root.ParentID is null

    UNION ALL

    SELECT
        Recurse.Acctid                                      AccountId
        , Recurse.Name                                      AccountName
        , Recurse.ParentId                                  ParentId
        , Root.HierarchyLevel + 1                           HierarchyLevel
        , CAST(CAST(Root.IdHierarchyMatch as varchar(40)) + '.' 
            + cast(recurse.Acctid as varchar(40))   as varchar(4000))   IdHierarchyMatch
        , cast(cast(recurse.Acctid as varchar(40)) + '.' 
            + Root.IdHierarchy  as varchar(4000))           IdHierarchy
        , cast(replace(recurse.Name,'.','') + '.' 
            + Root.NameHierarchy as varchar(4000))          NameHierarchy
        , cast(Root.AccountName + '.' 
            + Recurse.Name as varchar(4000))                HierarchySort   
        , cast(space(root.HierarchyLevel * 4) 
            + Recurse.Name as varchar(4000))                HierarchyLabel
        , Recurse.CustomerCount                             CustomerCount
    FROM
        tempdb.dbo.account Recurse INNER JOIN
        AccountHierarchy Root on Root.AccountId = Recurse.ParentId
)


SELECT
    hier.AccountId
    , Hier.AccountName
    , hier.ParentId
    , hier.HierarchyLevel
    , hier.IdHierarchy
    , hier.NameHierarchy
    , hier.HierarchyLabel
    , parsename(hier.IdHierarchy,1) Acct1Id
    , parsename(hier.NameHierarchy,1) Acct1Name
    , parsename(hier.IdHierarchy,2) Acct2Id
    , parsename(hier.NameHierarchy,2) Acct2Name
    , parsename(hier.IdHierarchy,3) Acct3Id
    , parsename(hier.NameHierarchy,3) Acct3Name
    , parsename(hier.IdHierarchy,4) Acct4Id
    , parsename(hier.NameHierarchy,4) Acct4Name
    , hier.CustomerCount
    , (
        SELECT  
            sum(children.CustomerCount)
        FROM
            AccountHierarchy Children
        WHERE
            Children.IdHierarchyMatch LIKE hier.IdHierarchyMatch + '%'
        ) TotalCustomerCount
        , HierarchySort
        , IdHierarchyMatch
FROM
    AccountHierarchy hier
ORDER BY
    hier.HierarchySort

আমি সিটিইতে একটি কলাম যুক্ত করেছিলাম IdHierarchyMatchযা সাবকোরি ক্লজটি সের্গেবল IdHierarchyকরতে সক্ষম করার ফরওয়ার্ড সংস্করণ ।TotalCustomerCountWHERE

এক্সিকিউশন পরিকল্পনাগুলির জন্য আনুমানিক সাবট্রি ব্যয়ের তুলনা করা, এই পথে প্রায় 5 গুণ দ্রুত হওয়া উচিত।


এটি দেখার জন্য সময় দেওয়ার জন্য ধন্যবাদ। এটি মজাদার, এটি সত্যই আমার প্রথম প্রবৃত্তি এবং ভেবেছিল যে কোনও ক্ষেত্রে ওয়াইল্ডকার্ড যুক্ত করা কেবল গতিশীল এসকিউএল ব্যবহার করেই সম্ভব ছিল, এবং তাই চেষ্টাও করেনি। আমার চেক করা উচিত ছিল। সুতরাং ফলাফলটি 2:49 (3:53 থেকে নীচে) এ একটি চিহ্নিত উন্নতি হয়েছে, তবে আমি যতটা আশা করেছিলাম তেমন নয়। অন্যান্য ধারণাগুলি কী উত্থিত হয় তা দেখার জন্য আমি উত্তরহীন হিসাবে চলে যাচ্ছি। সত্যিকার অর্থে এটি মূল্যায়নের জন্য সময় দেওয়ার জন্য আবারও ধন্যবাদ। আমি এটিকে সমর্থন করি.
লিভার.লারসন

এফওয়াইআই, সবেমাত্র আমার প্রয়োগে একটি বাক্য গঠন ত্রুটি লক্ষ্য করেছে। আমরা কার্যকর করার সময় 2:04 এ চলেছি। এখনও আমি যেখানে শুরু করেছি আরও ভাল। এখনও দ্রুত জন্য লক্ষ্য।
লিভার.লারসন

1
আমি খুশি যে আমি কিছু ছোট উপায়ে সহায়তা করেছি। আমি গতরাতে প্রায় 2 ঘন্টা সমস্যাটি সম্পর্কে মাথা পেতে চেষ্টা করেছিলাম। আমি আমার অন্ত্রে গভীর অনুভূতি পেয়েছি এটি কোনও একরকম ROW_NUMER() OVER (ORDER BY...)বা কিছু ব্যবহার করে সমাধান করা যেতে পারে । আমি এটি থেকে সঠিক নম্বর পেতে পারি না। এটি একটি দুর্দান্ত এবং আকর্ষণীয় প্রশ্ন। ভালো মস্তিষ্কের অনুশীলন!
ম্যাক্স ভার্নন

আমি IdHierarchyMatchক্ষেত্রটিতে একটি সূচক যুক্ত করার উদ্দেশ্যে, এটিকে একটি স্কিমা-আবদ্ধ (বস্তুগত) দর্শন হিসাবে তৈরি করার চেষ্টা করেছি , তবে আপনি স্কিমা-বাউন্ড দৃশ্যে একটি ক্লাস্টারড সূচক যুক্ত করতে পারবেন না যার মধ্যে একটি সিটিই রয়েছে। আমি ভাবছি যে এই সীমাবদ্ধতাটি এসকিউএল সার্ভার ২০১৪
ম্যাক্স ভার্নন

2
@ ম্যাক্স ভার্নন ২০১২ সংস্করণটির জন্য: এসকিউএল-ফিডল
ইপারকিউবেᵀᴹ

3

আমি এটি একটি শটও দিয়েছি। এটি খুব সুন্দর নয়, তবে এটি আরও ভাল করে দেখায় বলে মনে হচ্ছে।

USE Tempdb
go

SET STATISTICS IO ON;
SET STATISTICS TIME OFF;
SET NOCOUNT ON;

--------
-- assuming the original table looks something like this 
-- and you cannot control it's indexes 
-- (only widened the data types a bit for the extra sample rows)
--------
CREATE TABLE dbo.Account
    (
      Acctid VARCHAR(10) NOT NULL ,
      Name VARCHAR(100) NULL ,
      ParentId VARCHAR(10) NULL ,
      CustomerCount INT NULL
    );

--------
-- inserting the same records as in your sample
--------
INSERT  Account
        SELECT  'A' ,
                'Best Bet' ,
                NULL ,
                21
        UNION ALL
        SELECT  'B' ,
                'eStore' ,
                'A' ,
                30
        UNION ALL
        SELECT  'C' ,
                'Big Bens' ,
                'B' ,
                75
        UNION ALL
        SELECT  'D' ,
                'Mr. Jimbo' ,
                'B' ,
                50
        UNION ALL
        SELECT  'E' ,
                'Dr. John' ,
                'C' ,
                100
        UNION ALL
        SELECT  'F' ,
                'Brick' ,
                'A' ,
                222
        UNION ALL
        SELECT  'G' ,
                'Mortar' ,
                'C' ,
                153;

--------
-- now lets up the ante a bit and add some extra rows with random parents 
-- to these 7 items, it is hard to measure differences with so few rows
--------
DECLARE @numberOfRows INT = 25000
DECLARE @from INT = 1
DECLARE @to INT = 7
DECLARE @T1 TABLE ( n INT ); 

WITH    cte ( n )
          AS ( SELECT   ROW_NUMBER() OVER ( ORDER BY CURRENT_TIMESTAMP )
               FROM     sys.messages
             )
    INSERT  INTO @T1
            SELECT  n
            FROM    cte
            WHERE   n <= @numberOfRows;

INSERT  INTO dbo.Account
        ( acctId ,
          name ,
          parentId ,
          Customercount
        )
        SELECT  CHAR(64 + RandomNumber) + CAST(n AS VARCHAR(10)) AS Id ,
                CAST('item ' + CHAR(64 + RandomNumber) + CAST(n AS VARCHAR(10)) AS VARCHAR(100)) ,
                CHAR(64 + RandomNumber) AS parentId ,
                ABS(CHECKSUM(NEWID()) % 100) + 1 AS RandomCustCount
        FROM    ( SELECT    n ,
                            ABS(CHECKSUM(NEWID()) % @to) + @from AS RandomNumber
                  FROM      @T1
                ) A;

--------
-- Assuming you cannot control it's indexes, in my tests we're better off taking the IO hit of copying the data
-- to some structure that is better optimized for this query. Not quite what I initially expected,  but we seem 
-- to be better off that way.
--------
CREATE TABLE tempdb.dbo.T1
    (
      AccountId VARCHAR(10) NOT NULL
                            PRIMARY KEY NONCLUSTERED ,
      AccountName VARCHAR(100) NOT NULL ,
      ParentId VARCHAR(10) NULL ,
      HierarchyLevel INT NULL ,
      HPath VARCHAR(1000) NULL ,
      IdHierarchy VARCHAR(1000) NULL ,
      NameHierarchy VARCHAR(1000) NULL ,
      HierarchyLabel VARCHAR(1000) NULL ,
      HierarchySort VARCHAR(1000) NULL ,
      CustomerCount INT NOT NULL
    );

CREATE CLUSTERED INDEX IX_Q1
ON tempdb.dbo.T1  ([ParentId]);

-- for summing customer counts over parents
CREATE NONCLUSTERED INDEX IX_Q2 
ON tempdb.dbo.T1  (HPath) INCLUDE(CustomerCount);

INSERT  INTO tempdb.dbo.T1
        ( AccountId ,
          AccountName ,
          ParentId ,
          HierarchyLevel ,
          HPath ,
          IdHierarchy ,
          NameHierarchy ,
          HierarchyLabel ,
          HierarchySort ,
          CustomerCount 
        )
        SELECT  Acctid AS AccountId ,
                Name AS AccountName ,
                ParentId AS ParentId ,
                NULL AS HierarchyLevel ,
                NULL AS HPath ,
                NULL AS IdHierarchy ,
                NULL AS NameHierarchy ,
                NULL AS HierarchyLabel ,
                NULL AS HierarchySort ,
                CustomerCount AS CustomerCount
        FROM    tempdb.dbo.account;



--------
-- I cannot seem to force an efficient way to do the sum while selecting over the recursive cte, 
-- so I took it aside. I am sure there is a more elegant way but I can't seem to make it happen. 
-- At least it performs better this way. But it remains a very expensive query.
--------
;
WITH    AccountHierarchy
          AS ( SELECT   Root.AccountId AS AcId ,
                        Root.ParentId ,
                        1 AS HLvl ,
                        CAST(Root.AccountId AS VARCHAR(1000)) AS [HPa] ,
                        CAST(Root.accountId AS VARCHAR(1000)) AS hid ,
                        CAST(REPLACE(Root.AccountName, '.', '') AS VARCHAR(1000)) AS hn ,
                        CAST(Root.accountid AS VARCHAR(1000)) AS hs ,
                        CAST(Root.accountname AS VARCHAR(1000)) AS hl
               FROM     tempdb.dbo.T1 Root
               WHERE    Root.ParentID IS NULL
               UNION ALL
               SELECT   Recurse.AccountId AS acid ,
                        Recurse.ParentId ParentId ,
                        Root.Hlvl + 1 AS hlvl ,
                        CAST(Root.HPa + '.' + Recurse.AccountId AS VARCHAR(1000)) AS hpa ,
                        CAST(recurse.AccountId + '.' + Root.hid AS VARCHAR(1000)) AS hid ,
                        CAST(REPLACE(recurse.AccountName, '.', '') + '.' + Root.hn AS VARCHAR(1000)) AS hn ,
                        CAST(Root.hs + '.' + Recurse.AccountName AS VARCHAR(1000)) AS hs ,
                        CAST(SPACE(root.hlvl * 4) + Recurse.AccountName AS VARCHAR(1000)) AS hl
               FROM     tempdb.dbo.T1 Recurse
                        INNER JOIN AccountHierarchy Root ON Root.AcId = Recurse.ParentId
             )
    UPDATE  tempdb.dbo.T1
    SET     HierarchyLevel = HLvl ,
            HPath = Hpa ,
            IdHierarchy = hid ,
            NameHierarchy = hn ,
            HierarchyLabel = hl ,
            HierarchySort = hs
    FROM    AccountHierarchy
    WHERE   AccountId = AcId;

SELECT  --HPath ,
        AccountId ,
        AccountName ,
        ParentId ,
        HierarchyLevel ,
        IdHierarchy ,
        NameHierarchy ,
        HierarchyLabel ,
        PARSENAME(IdHierarchy, 1) Acct1Id ,
        PARSENAME(NameHierarchy, 1) Acct1Name ,
        PARSENAME(IdHierarchy, 2) Acct2Id ,
        PARSENAME(NameHierarchy, 2) Acct2Name ,
        PARSENAME(IdHierarchy, 3) Acct3Id ,
        PARSENAME(NameHierarchy, 3) Acct3Name ,
        PARSENAME(IdHierarchy, 4) Acct4Id ,
        PARSENAME(NameHierarchy, 4) Acct4Name ,
        CustomerCount ,
        Cnt.TotalCustomerCount
FROM    tempdb.dbo.t1 Hier
        CROSS APPLY ( SELECT    SUM(CustomerCount) AS TotalCustomerCount
                      FROM      tempdb.dbo.t1
                      WHERE     HPath LIKE hier.HPath + '%'
                    ) Cnt
ORDER BY HierarchySort;

DROP TABLE tempdb.dbo.t1;
DROP TABLE tempdb.dbo.Account;

সাহসী প্রচেষ্টা। এবং এটি বেশ কিছু মিষ্টি নমুনা ডেটা জেনারেশন। আমার সেই কৌশলগুলি থেকে কিছু করা ভাল। এখনও সেই মার্জিত সমাধানের সন্ধান করছি আমি নিশ্চিত সেখানে অপেক্ষা করছি।
লিভার.লারসন
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.