سؤال

لدي عرض قاعدة بيانات الحالة الشهرية وأحتاج إلى إنشاء تقرير بناءً عليها.تبدو البيانات الموجودة في طريقة العرض كما يلي:

Category | Revenue  |  Yearh  |  Month
Bikes      10 000      2008        1
Bikes      12 000      2008        2
Bikes      12 000      2008        3
Bikes      15 000      2008        1
Bikes      11 000      2007        2
Bikes      11 500      2007        3
Bikes      15 400      2007        4


...وهكذا دواليك

يحتوي العرض على فئة المنتج والإيرادات والسنة والشهر.أريد إنشاء تقرير يقارن بين عامي 2007 و2008، مع عرض 0 للأشهر التي لم تكن بها مبيعات.لذلك يجب أن يبدو التقرير كما يلي:

Category  |  Month  |  Rev. This Year  |  Rev. Last Year
Bikes          1          10 000               0
Bikes          2          12 000               11 000
Bikes          3          12 000               11 500
Bikes          4          0                    15 400


الشيء الأساسي الذي يجب ملاحظته هو كيف أن الشهر الأول يحتوي على مبيعات في عام 2008 فقط، وبالتالي يكون 0 لعام 2007.أيضًا، الشهر الرابع فقط ليس لديه مبيعات في عام 2008، وبالتالي 0، في حين أن لديه مبيعات في عام 2007 وما زال يظهر.

كما أن التقرير مخصص في الواقع للسنة المالية - لذا أرغب في الحصول على أعمدة فارغة تحتوي على 0 في كليهما إذا لم تكن هناك مبيعات في الشهر الخامس على سبيل المثال لعام 2007 أو 2008.

الاستعلام الذي تلقيته يبدو كالتالي:

SELECT 
    SP1.Program,
    SP1.Year,
    SP1.Month,
    SP1.TotalRevenue,
    IsNull(SP2.TotalRevenue, 0) AS LastYearTotalRevenue

FROM PVMonthlyStatusReport AS SP1 
     LEFT OUTER JOIN PVMonthlyStatusReport AS SP2 ON 
                SP1.Program = SP2.Program AND 
                SP2.Year = SP1.Year - 1 AND 
                SP1.Month = SP2.Month
WHERE 
    SP1.Program = 'Bikes' AND
    SP1.Category = @Category AND 
    (SP1.Year >= @FinancialYear AND SP1.Year <= @FinancialYear + 1) AND
    ((SP1.Year = @FinancialYear AND SP1.Month > 6) OR 
     (SP1.Year = @FinancialYear + 1 AND SP1.Month <= 6))

ORDER BY SP1.Year, SP1.Month

المشكلة في هذا الاستعلام هي أنه لن يعرض الصف الرابع في بيانات المثال أعلاه، حيث لم يكن لدينا أي مبيعات في عام 2008، ولكننا فعلنا ذلك بالفعل في عام 2007.

من المحتمل أن يكون هذا استعلامًا/مشكلة شائعة، لكن SQL الخاص بي أصبح صدئًا بعد القيام بتطوير الواجهة الأمامية لفترة طويلة.أي مساعدة يحظى بتقدير كبير!

أوه، راجع للشغل، أنا أستخدم SQL 2005 لهذا الاستعلام، لذا إذا كانت هناك أي ميزات جديدة مفيدة قد تساعدني، فأخبرني بذلك.

هل كانت مفيدة؟

المحلول

بيان الحالة هو أفضل صديق لي في SQL.أنت أيضًا بحاجة إلى جدول للوقت لتوليد 0 مراجعة في كلا الشهرين.

تعتمد الافتراضات على توفر الجداول التالية:

مبيعات:فئة | الإيرادات | سنة | شهر

و

تم:سنة | الشهر (يسكنها جميع التواريخ المطلوبة للإبلاغ)

مثال 1 بدون صفوف فارغة:

select
    Category
    ,month
    ,SUM(CASE WHEN YEAR = 2008 THEN Revenue ELSE 0 END) this_year
    ,SUM(CASE WHEN YEAR = 2007 THEN Revenue ELSE 0 END) last_year

from
    sales

where
    year in (2008,2007)

group by
    Category
    ,month

عائدات:

Category  |  Month  |  Rev. This Year  |  Rev. Last Year
Bikes          1          10 000               0
Bikes          2          12 000               11 000
Bikes          3          12 000               11 500
Bikes          4          0                    15 400

المثال 2 مع صفوف فارغة:سأستخدم استعلامًا فرعيًا (ولكن قد لا يستخدمه الآخرون) وسأعيد صفًا فارغًا لكل مجموعة من المنتجات والشهر.

select
    fill.Category
    ,fill.month
    ,SUM(CASE WHEN YEAR = 2008 THEN Revenue ELSE 0 END) this_year
    ,SUM(CASE WHEN YEAR = 2007 THEN Revenue ELSE 0 END) last_year

from
    sales
    Right join (select distinct  --try out left, right and cross joins to test results.
                   product
                   ,year
                   ,month
               from
                  sales --this ideally would be from a products table
                  cross join tm
               where
                    year in (2008,2007)) fill


where
    fill.year in (2008,2007)

group by
    fill.Category
    ,fill.month

عائدات:

Category  |  Month  |  Rev. This Year  |  Rev. Last Year
Bikes          1          10 000               0
Bikes          2          12 000               11 000
Bikes          3          12 000               11 500
Bikes          4          0                    15 400
Bikes          5          0                    0
Bikes          6          0                    0
Bikes          7          0                    0
Bikes          8          0                    0

لاحظ أن معظم أدوات إعداد التقارير ستقوم بهذه الوظيفة الجدولية أو المصفوفة، والآن بعد أن فكرت في الأمر، يحتوي SQL Server 2005 على بناء جملة محوري يقوم بذلك أيضًا.

فيما يلي بعض الموارد الإضافية.قضيةhttp://www.4guysfromrolla.com/webtech/102704-1.shtmlخادم SQL 2005 المحوريةhttp://msdn.microsoft.com/en-us/library/ms177410.aspx

نصائح أخرى

@Christian - محرر تخفيض السعر - UGH؛خاصة عندما تختلف المعاينة والنسخة النهائية لمنشورك...@Christian - صلة خارجية كاملة - يتم إبطال الصلة الخارجية الكاملة من خلال وجود مراجع إلى SP1 في جملة WHERE، ويتم تطبيق جملة WHERE بعد JOIN.لإجراء صلة خارجية كاملة مع التصفية على أحد الجداول، تحتاج إلى وضع جملة WHERE في استعلام فرعي، حتى تتم التصفية قبل الانضمام، أو حاول بناء كافة معايير WHERE الخاصة بك على جملة JOIN ON، والتي تعتبر قبيحة إلى حد الجنون.حسنًا، في الواقع لا توجد طريقة جميلة للقيام بذلك.

@جوناس:النظر في هذا:

كما أن التقرير في الواقع يتعلق بالسنة المالية - لذا أرغب في الحصول على أعمدة فارغة تحتوي على 0 في كليهما إذا لم تكن هناك مبيعات في الشهر الخامس على سبيل المثال لعامي 2007 أو 2008.

وحقيقة أن هذه المهمة لا يمكن إنجازها باستعلام جميل، سأحاول بالتأكيد الحصول على النتائج التي تريدها بالفعل.لا فائدة من وجود استعلام قبيح وعدم حتى الحصول على البيانات الدقيقة التي تريدها بالفعل.;)

لذا أقترح القيام بذلك في 5 خطوات:
1.قم بإنشاء جدول مؤقت بالتنسيق الذي تريد أن تتطابق معه نتائجك
2.املأها باثني عشر صفًا، مع 1-12 في عمود الشهر
3.قم بتحديث العمود "هذا العام" باستخدام منطق SP1 الخاص بك
4.قم بتحديث عمود "العام الماضي" باستخدام منطق SP2 الخاص بك
5.اختر من الجدول المؤقت

بالطبع، أعتقد أنني أعمل على افتراض أنه يمكنك إنشاء إجراء مخزن لإنجاز هذا.قد تكون قادرًا من الناحية الفنية على تشغيل هذه الدفعة بأكملها بشكل مضمن، لكن هذا النوع من القبح نادرًا ما يُرى.إذا لم تتمكن من إنشاء SP، أقترح عليك الرجوع إلى الصلة الخارجية الكاملة عبر استعلام فرعي، ولكنها لن تحصل على صف عندما لا يكون هناك مبيعات في أي شهر في أي من العامين.

بخصوص تخفيض السعر - نعم هذا محبط.قام المحرر بمعاينة جدول HTML الخاص بي، ولكن بعد نشره اختفى - لذلك اضطر إلى إزالة كافة تنسيقات HTML من المنشور...

kcrumley أعتقد أننا توصلنا إلى استنتاجات مماثلة.يصبح هذا الاستعلام قبيحًا بسهولة.لقد قمت بالفعل بحل هذه المشكلة قبل قراءة إجابتك، باستخدام نهج مماثل (لكنه مختلف).لدي حق الوصول لإنشاء الإجراءات والوظائف المخزنة في قاعدة بيانات التقارير.لقد قمت بإنشاء دالة جدول القيمة بقبول فئة المنتج والسنة المالية كمعلمة.بناءً على ذلك، ستقوم الدالة بملء جدول يحتوي على 12 صفًا.سيتم ملء الصفوف بالبيانات من العرض في حالة توفر أي مبيعات، وإذا لم يكن الأمر كذلك فسيكون للصف 0 قيم.

أقوم بعد ذلك بضم الجدولين اللذين تم إرجاعهما بواسطة function.نظرًا لأنني أعلم أن جميع الجداول ستحتوي على اثني عشر مجموعة، فالأمر أسهل كثيرًا، ويمكنني الانضمام إلى فئة المنتج والشهر:

SELECT 
    SP1.Program,
    SP1.Year,
    SP1.Month,
    SP1.TotalRevenue AS ThisYearRevenue,
    SP2.TotalRevenue AS LastYearRevenue
FROM GetFinancialYear(@Category, 'First Look',  2008) AS SP1 
     RIGHT JOIN GetFinancialYear(@Category, 'First Look',  2007) AS SP2 ON 
         SP1.Program = SP2.Program AND 
         SP1.Month = SP2.Month

أعتقد أن أسلوبك ربما يكون أكثر نظافة بعض الشيء لأن وظيفة GetFinancialYear فوضوية تمامًا!ولكن على الأقل يعمل - وهو ما يجعلني سعيدا في الوقت الراهن؛)

الحيلة هي إجراء صلة كاملة، باستخدام ISNULL للحصول على الأعمدة المرتبطة من أي من الجدولين.عادةً ما أقوم بتغليف هذا في عرض أو جدول مشتق، وإلا فستحتاج إلى استخدام ISNULL في جملة WHERE أيضًا.

SELECT 
    Program,
    Month,
    ThisYearTotalRevenue,
    PriorYearTotalRevenue
FROM (
    SELECT 
        ISNULL(ThisYear.Program, PriorYear.Program) as Program,
        ISNULL(ThisYear.Month, PriorYear.Month),
        ISNULL(ThisYear.TotalRevenue, 0) as ThisYearTotalRevenue,
        ISNULL(PriorYear.TotalRevenue, 0) as PriorYearTotalRevenue
    FROM (
        SELECT Program, Month, SUM(TotalRevenue) as TotalRevenue 
        FROM PVMonthlyStatusReport 
        WHERE Year = @FinancialYear 
        GROUP BY Program, Month
    ) as ThisYear 
    FULL OUTER JOIN (
        SELECT Program, Month, SUM(TotalRevenue) as TotalRevenue 
        FROM PVMonthlyStatusReport 
        WHERE Year = (@FinancialYear - 1) 
        GROUP BY Program, Month
    ) as PriorYear ON
        ThisYear.Program = PriorYear.Program
        AND ThisYear.Month = PriorYear.Month
) as Revenue
WHERE 
    Program = 'Bikes'
ORDER BY 
    Month

من المفترض أن يوفر لك ذلك الحد الأدنى من متطلباتك - الصفوف التي بها مبيعات في عام 2007 أو 2008، أو كليهما.للحصول على صفوف لا تحتوي على مبيعات في أي من العامين، تحتاج فقط إلى الانضمام الداخلي إلى جدول الأرقام من 1 إلى 12 (يمكنك القيام بذلك لديك واحدة من هؤلاء, ، أنت , لا؟).

قد أكون مخطئًا ولكن ألا ينبغي عليك استخدام صلة خارجية كاملة بدلاً من مجرد صلة يسرى؟بهذه الطريقة ستحصل على أعمدة "فارغة" من كلا الجدولين.

http://en.wikipedia.org/wiki/Join_(SQL)#Full_outer_join

باستخدام المحور وDynamic Sql يمكننا تحقيق هذه النتيجة

SET NOCOUNT ON
IF OBJECT_ID('TEMPDB..#TEMP') IS NOT NULL
DROP TABLE #TEMP

;With cte(Category , Revenue  ,  Yearh  ,  [Month])
AS
(
SELECT 'Bikes', 10000, 2008,1 UNION ALL
SELECT 'Bikes', 12000, 2008,2 UNION ALL
SELECT 'Bikes', 12000, 2008,3 UNION ALL
SELECT 'Bikes', 15000, 2008,1 UNION ALL
SELECT 'Bikes', 11000, 2007,2 UNION ALL
SELECT 'Bikes', 11500, 2007,3 UNION ALL
SELECT 'Bikes', 15400, 2007,4
)
SELECT * INTO #Temp FROM cte

Declare @Column nvarchar(max),
        @Column2 nvarchar(max),
        @Sql nvarchar(max)


SELECT @Column=STUFF((SELECT DISTINCT ','+ 'ISNULL('+QUOTENAME(CAST(Yearh AS VArchar(10)))+','+'''0'''+')'+ 'AS '+ QUOTENAME(CAST(Yearh AS VArchar(10)))
FROM #Temp order by 1 desc FOR XML PATH ('')),1,1,'')

SELECT @Column2=STUFF((SELECT DISTINCT ','+ QUOTENAME(CAST(Yearh AS VArchar(10)))
FROM #Temp FOR XML PATH ('')),1,1,'')

SET @Sql= N'SELECT Category,[Month],'+ @Column +'FRom #Temp
            PIVOT
            (MIN(Revenue) FOR yearh IN ('+@Column2+')
            ) AS Pvt

            '
EXEC(@Sql)
Print @Sql

نتيجة

Category    Month   2008    2007
----------------------------------
Bikes       1       10000   0
Bikes       2       12000   11000
Bikes       3       12000   11500
Bikes       4       0       15400
مرخصة بموجب: CC-BY-SA مع الإسناد
لا تنتمي إلى StackOverflow
scroll top