[C#] अधिकतम या डिफ़ॉल्ट?


Answers

मुझे बस एक ही समस्या थी, लेकिन मैं क्वेरी वाक्यविन्यास की बजाय सूची में LINQ एक्सटेंशन विधियों का उपयोग कर रहा था। एक निरर्थक चाल का कास्टिंग वहां भी काम करता है:

int max = list.Max(i => (int?)i.MyCounter) ?? 0;
Question

LINQ क्वेरी से अधिकतम मान प्राप्त करने का सबसे अच्छा तरीका क्या है जो कोई पंक्ति नहीं लौटा सकता है? अगर मैं बस करता हूँ

Dim x = (From y In context.MyTable _
         Where y.MyField = value _
         Select y.MyCounter).Max

जब कोई क्वेरी कोई पंक्ति नहीं लौटाती है तो मुझे एक त्रुटि मिलती है। मैं कर सकता था

Dim x = (From y In context.MyTable _
         Where y.MyField = value _
         Select y.MyCounter _
         Order By MyCounter Descending).FirstOrDefault

लेकिन ऐसा लगता है कि इस तरह के एक सरल अनुरोध के लिए थोड़ा उलझन में है। क्या मुझे ऐसा करने का बेहतर तरीका याद आ रहा है?

अद्यतन: यहां पिछली कहानी है: मैं एक बच्चे की मेज से अगले योग्यता काउंटर को पुनः प्राप्त करने की कोशिश कर रहा हूं (विरासत प्रणाली, मुझे शुरू नहीं करें ...)। प्रत्येक रोगी के लिए पहली पात्रता पंक्ति हमेशा 1 होती है, दूसरा 2 होता है, आदि (जाहिर है यह बाल तालिका की प्राथमिक कुंजी नहीं है)। इसलिए, मैं एक रोगी के लिए अधिकतम मौजूदा काउंटर वैल्यू का चयन कर रहा हूं, और फिर एक नई पंक्ति बनाने के लिए इसमें 1 जोड़ रहा हूं। जब कोई मौजूदा बच्चा मूल्य नहीं होता है, तो मुझे 0 को वापस करने के लिए क्वेरी की आवश्यकता होती है (इसलिए 1 जोड़ना मुझे 1 का काउंटर वैल्यू देगा)। ध्यान दें कि मैं बाल पंक्तियों की कच्ची गिनती पर भरोसा नहीं करना चाहता हूं, अगर विरासत ऐप काउंटर मूल्यों (संभव) में अंतराल पेश करता है। प्रश्न को सामान्य बनाने की कोशिश करने के लिए मेरा बुरा।




मैंने MaxOrDefault एक्सटेंशन विधि को MaxOrDefault कर दिया है। इसके लिए बहुत कुछ नहीं है लेकिन इंटेलिसेंस में इसकी उपस्थिति एक उपयोगी अनुस्मारक है कि खाली अनुक्रम पर Max अपवाद का कारण बन जाएगा। इसके अतिरिक्त, यदि आवश्यक हो तो विधि डिफ़ॉल्ट को निर्दिष्ट करने की अनुमति देती है।

    public static TResult MaxOrDefault<TSource, TResult>(this 
    IQueryable<TSource> source, Expression<Func<TSource, TResult?>> selector,
    TResult defaultValue = default (TResult)) where TResult : struct
    {
        return source.Max(selector) ?? defaultValue;
    }

Id नामित प्रकार के कॉलम या संपत्ति पर उपयोग Id :

    sequence.DefaultOrMax(s => (int?)s.Id);



बस हर किसी को यह जानने के लिए कि लिंक से इकाइयों का उपयोग कर रहा है ऊपर दिए गए तरीके काम नहीं करेंगे ...

यदि आप कुछ ऐसा करने की कोशिश करते हैं

var max = new[]{0}
      .Concat((From y In context.MyTable _
               Where y.MyField = value _
               Select y.MyCounter))
      .Max();

यह एक अपवाद फेंक देगा:

System.NotSupportedException: LINQ अभिव्यक्ति नोड प्रकार 'NewArrayInit' LINQ से इकाइयों में समर्थित नहीं है ..

मैं बस सुझाव दे रहा हूँ

(From y In context.MyTable _
                   Where y.MyField = value _
                   Select y.MyCounter))
          .OrderByDescending(x=>x).FirstOrDefault());

और अगर आपकी सूची खाली है तो FirstOrDefault 0 वापस आ जाएगा।




decimal Max = (decimal?)(context.MyTable.Select(e => e.MyCounter).Max()) ?? 0;



देर से लिट, लेकिन मुझे एक ही चिंता थी ...

मूल कोड से अपने कोड को दोबारा दोहराएं, आप सेट एस के अधिकतम परिभाषित करना चाहते हैं

(From y In context.MyTable _
 Where y.MyField = value _
 Select y.MyCounter)

खाते में अपनी आखिरी टिप्पणी लेना

कहने का पर्याप्त कारण है कि मुझे पता है कि मुझे 0 चाहिए जब से चुनने के लिए कोई रिकॉर्ड नहीं है, जो निश्चित रूप से अंतिम समाधान पर असर डालता है

मैं आपकी समस्या को फिर से बदल सकता हूं: आप अधिकतम {0 + एस} चाहते हैं। और ऐसा लगता है कि कॉन्सैट के साथ प्रस्तावित समाधान अर्थात् सही है :-)

var max = new[]{0}
          .Concat((From y In context.MyTable _
                   Where y.MyField = value _
                   Select y.MyCounter))
          .Max();



मुझे लगता है कि समस्या यह है कि जब आप क्वेरी का कोई परिणाम नहीं लेते हैं तो आप क्या करना चाहते हैं। यदि यह एक असाधारण मामला है तो मैं क्वेरी को एक कोशिश / पकड़ ब्लॉक में लपेटूंगा और मानक क्वेरी उत्पन्न करने वाले अपवाद को संभालेगा। यदि क्वेरी ठीक से कोई परिणाम नहीं लौटाती है, तो आपको यह पता लगाना होगा कि आप उस मामले में क्या परिणाम चाहते हैं। यह हो सकता है कि @ डेविड का जवाब (या कुछ ऐसा ही काम करेगा)। यही है, यदि MAX हमेशा सकारात्मक होगा, तो सूची में ज्ञात "खराब" मान डालने के लिए पर्याप्त हो सकता है जो केवल तभी चुना जाएगा जब कोई परिणाम न हो। आम तौर पर, मैं एक ऐसी क्वेरी की अपेक्षा करता हूं जो अधिकतम डेटा प्राप्त करने के लिए अधिकतम डेटा प्राप्त कर रहा हो और मैं कोशिश / पकड़ मार्ग पर जाऊंगा अन्यथा आपको हमेशा यह जांचने के लिए मजबूर किया जाता है कि आपके द्वारा प्राप्त मूल्य सही है या नहीं। मैं बल्कि इतना असाधारण मामला प्राप्त मूल्य का उपयोग करने में सक्षम था।

Try
   Dim x = (From y In context.MyTable _
            Where y.MyField = value _
            Select y.MyCounter).Max
   ... continue working with x ...
Catch ex As SqlException
       ... do error processing ...
End Try



int max = list.Any() ? list.Max(i => i.MyCounter) : 0;

यदि सूची में कोई तत्व है (यानी खाली नहीं), तो यह अधिकतम माईकॉन्टर फ़ील्ड लेगा, अन्यथा 0 वापस आ जाएगा।




आप जो पूछ रहे हैं उसके बारे में सोचें!

{1, 2, 3, -1, -2, -3} का अधिकतम स्पष्ट रूप से 3. अधिकतम {2} स्पष्ट रूप से 2 है। लेकिन खाली सेट का अधिकतम {} क्या है? जाहिर है कि यह एक अर्थहीन सवाल है। खाली सेट का अधिकतम बस परिभाषित नहीं किया गया है। उत्तर पाने का प्रयास करना गणितीय त्रुटि है। किसी भी सेट का अधिकतम स्वयं उस सेट में एक तत्व होना चाहिए। खाली सेट में कोई तत्व नहीं है, इसलिए दावा करना कि कुछ विशेष संख्या उस सेट के बिना सेट के अधिकतम है, वह गणितीय विरोधाभास है।

जैसे ही प्रोग्रामर अपवाद फेंकने के लिए कंप्यूटर के लिए सही व्यवहार करता है, जब प्रोग्रामर इसे शून्य से विभाजित करने के लिए कहता है, इसलिए कंप्यूटर के लिए अपवाद फेंकने के लिए यह सही व्यवहार है जब प्रोग्रामर इसे खाली सेट को अधिकतम करने के लिए कहता है। शून्य से डिवीजन, रिक्त सेट का अधिकतम भाग लेना, स्पैक्लरॉर्के को घुमाएं, और उड़ने वाले यूनिकॉर्न को नेवरलैंड में सवार करना सभी अर्थहीन, असंभव, अपरिभाषित हैं।

अब, आप वास्तव में क्या करना चाहते हैं?






Links