2008-09-09 11 views
8

एक निश्चित क्षेत्र में शामिल है तो मैं कुछ व्यापार वस्तुओं हाइड्रेट करने के लिए एक IDataReader उपयोग कर रहा हूँ, लेकिन मैं क्रम में वास्तव में क्या क्षेत्रों पाठक में हो जाएगा पता नहीं है। कोई भी फ़ील्ड जो पाठक में नहीं है परिणामी वस्तु पर शून्य हो जाएगी। यदि आप किसी पाठक को किसी विशेष फ़ील्ड को किसी कोशिश/पकड़ में लपेटने के बिना एक विशिष्ट फ़ील्ड रखते हैं तो आप कैसे परीक्षण करते हैं?का पता लगा रहा है, तो एक IDataReader से पहले यात्रा

उत्तर

-4

आप नल या डीबीएनयूएल के लिए केवल पाठक ["फ़ील्ड"] का परीक्षण नहीं कर सकते हैं क्योंकि कॉलम पाठक में नहीं है तो indexOutOfRangeException फेंक दिया जाता है।

कोड मैं डोमेन वस्तुओं और संग्रहित प्रक्रियाओं कि मानचित्रण परत का उपयोग अलग स्तंभ नाम हो सकता है बनाने के लिए मेरे मानचित्रण परत में उपयोग के नीचे है; यदि कॉलम नहीं मिला है और डिफ़ॉल्ट (टी) या शून्य वापस लौटाता है तो आप इसे अपवाद फेंकने के लिए संशोधित कर सकते हैं।

मैं समझता हूँ कि यह सबसे खूबसूरत या सर्वोत्कृष्ट समाधान नहीं है (और वास्तव में, यदि आप इसे तो आपको चाहिए बच सकते हैं), हालांकि, विरासत संग्रहित प्रक्रियाओं या SQL क्वेरी एक काम के आसपास वारंट सकता है।

/// <summary> 
    /// Grabs the value from a specific datareader for a list of column names. 
    /// </summary> 
    /// <typeparam name="T">Type of the value.</typeparam> 
    /// <param name="reader">Reader to grab data off of.</param> 
    /// <param name="columnNames">Column names that should be interrogated.</param> 
    /// <returns>Value from the first correct column name or an exception if none of the columns exist.</returns> 
    public static T GetColumnValue<T>(IDataReader reader, params string[] columnNames) 
    { 
     bool foundValue = false; 
     T value = default(T); 
     IndexOutOfRangeException lastException = null; 

     foreach (string columnName in columnNames) 
     { 
      try 
      { 
       int ordinal = reader.GetOrdinal(columnName); 
       value = (T)reader.GetValue(ordinal); 
       foundValue = true; 
      } 
      catch (IndexOutOfRangeException ex) 
      { 
       lastException = ex; 
      } 
     } 

     if (!foundValue) 
     { 
      string message = string.Format("Column(s) {0} could not be not found.", 
       string.Join(", ", columnNames)); 

      throw new IndexOutOfRangeException(message, lastException); 
     } 

     return value; 
    } 
+1

यह खराब प्रदर्शन करेगा – DalSoft

-3

मैं इस दृष्टिकोण (मुझे लगता है कि जब डेटा तक पहुँचने, तो आप पहले से आकार पता होना चाहिए) से असहमत है, मैं समझता हूँ कि अपवाद देखते हैं कि।

तुम हमेशा पाठक के साथ एक datatable लोड कर सकता है और फिर इसे माध्यम से पुनरावृति। फिर आप यह देखने के लिए जांच सकते हैं कि कॉलम मौजूद है या नहीं। यह कम प्रदर्शनकारी होगा, लेकिन आपको कोशिश/पकड़ने वाले ब्लॉक की आवश्यकता नहीं होगी (इसलिए शायद यह आपकी आवश्यकताओं के लिए अधिक सक्षम है)।

9

इस चाल करना चाहिए:

Public Shared Function ReaderContainsColumn(ByVal reader As IDataReader, ByVal name As String) As Boolean 
     For i As Integer = 0 To reader.FieldCount - 1 
      If reader.GetName(i).Equals(name, StringComparison.CurrentCultureIgnoreCase) Then Return True 
     Next 
     Return False 
    End Function 

या (सी # में)

public static bool ReaderContainsColumn(IDataReader reader, string name) 
{ 
    for (int i = 0; i < reader.FieldCount; i++) { 
     if (reader.GetName(i).Equals(name, StringComparison.CurrentCultureIgnoreCase)) return true; 
    } 
    return false; 
} 

: ओ)

7

तुम भी IDataReader.GetSchemaTable उपयोग कर सकते हैं के सभी कॉलम की एक सूची प्राप्त करने के लिए पाठक।

http://support.microsoft.com/kb/310107

+0

GetSchemaTable लौटाए गए डेटा संरचना के कॉलम सूचीबद्ध नहीं करता है। – JamesEggers

+2

हां यह करता है। पाठक से प्रत्येक पंक्ति में पहला आइटम। GetSchemaTable() पंक्तियां कॉलम नाम हैं। यानी पाठक। GetSchemaTable() पंक्तियां [0] [0] मुझे पहला कॉलम नाम देता है। –

+0

पिछली बार मैंने ऐसा करने की कोशिश की, उसने मुझे कॉलमा जैसे schema_info दिया। मुझे फिर से कोशिश करनी होगी हालांकि यह थोड़ी देर हो चुकी है। – JamesEggers

0

सबसे अच्छा समाधान मैं का उपयोग किया है इस तरह यह क्या कर रहा है:

DataTable dataTable = new DataTable(); 
dataTable.Load(reader); 
foreach (var item in dataTable.Rows) 
{ 
    bool columnExists = item.Table.Columns.Contains("ColumnName"); 
} 

पाठक [ "columnName"] के माध्यम से प्राप्त करना चाहते हैं और अशक्त या DBNull के लिए जाँच फेंक होगा एक अपवाद।

4
Enumerable.Range(0, reader.FieldCount).Any(i => reader.GetName(i) == "ColumnName") 
संबंधित मुद्दे