[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 का काउंटर वैल्यू देगा)। ध्यान दें कि मैं बाल पंक्तियों की कच्ची गिनती पर भरोसा नहीं करना चाहता हूं, अगर विरासत ऐप काउंटर मूल्यों (संभव) में अंतराल पेश करता है। प्रश्न को सामान्य बनाने की कोशिश करने के लिए मेरा बुरा।




मुझे लगता है कि समस्या यह है कि जब आप क्वेरी का कोई परिणाम नहीं लेते हैं तो आप क्या करना चाहते हैं। यदि यह एक असाधारण मामला है तो मैं क्वेरी को एक कोशिश / पकड़ ब्लॉक में लपेटूंगा और मानक क्वेरी उत्पन्न करने वाले अपवाद को संभालेगा। यदि क्वेरी ठीक से कोई परिणाम नहीं लौटाती है, तो आपको यह पता लगाना होगा कि आप उस मामले में क्या परिणाम चाहते हैं। यह हो सकता है कि @ डेविड का जवाब (या कुछ ऐसा ही काम करेगा)। यही है, यदि 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



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

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

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 वापस आ जाएगा।




मैंने 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);



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

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

(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();



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

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




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

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

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

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




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



Related