Мы используем Linqpad для наших распорядителей данных в качестве инструмента отчетности, альтернативного SSMS. Большинство наших управляющих все еще застряли в использовании SQL (мы постепенно переводим некоторых из них на LINQ, но постепенно). Однако мы столкнулись с ограничением в LINQPad, с которым я не знаю, как с ним бороться. Для больших наборов результатов, поскольку LINQPad сначала загружает запрос в память, а затем выводит его на экран, у нас заканчивается память для больших наборов результатов. Есть ли способ отправить SQL-запрос из LINQPad прямо в CSV?

1
Elsimer 7 Янв 2013 в 22:30

1 ответ

Лучший ответ

LINQPad не имеет встроенного метода для этого (я должен добавить его), но его достаточно легко написать. Поместите следующее в «Мои расширения»:

public static class MyExtensions
{
    public static void WriteCsv<T> (this IEnumerable<T> elements, string filePath)
    {
        var fields = typeof (T).GetFields();

        var props = typeof (T).GetProperties()
            .Where (p => IsSimpleType (p.PropertyType))
            .ToArray();

        using (var writer = new StreamWriter (filePath))
        {
            string header = string.Join (",", 
                fields.Select (f => f.Name).Concat (props.Select (p => p.Name)))

            // Work around bug in Excel
            if (header.StartsWith ("ID")) header = " " + header;

            writer.WriteLine (header);
            foreach (var element in elements)
            {
                var values =
                    fields.Select (f => Format (f.GetValue (element))).Concat (
                    props.Select (p => Format (p.GetValue (element, null))));

                writer.WriteLine (string.Join (",", values));
            }
        }
    }

    static string Format (object value)
    {
        if (value == null) return "";

        // With DateTimes, it's safest to force a culture insensitive format:
        if (value is DateTime) return ((DateTime)value).ToString ("s");
        if (value is DateTimeOffset) return ((DateTimeOffset)value).ToString ("s");

        string result = value.ToString();
        result = result.Replace ("\"", "\"\"");
        if (result.Contains (",") || result.Contains ("\"") || result.Any (c => c < 0x20))
            result = "\"" + result + "\"";

        return result;
    }

    public static bool IsSimpleType (Type t)
    {
        if (t.IsGenericType && t.GetGenericTypeDefinition () == typeof (Nullable<>))
            t = t.GetGenericArguments() [0];

        return t.IsPrimitive || Type.GetTypeCode (t) != TypeCode.Object ||
               typeof (IFormattable).IsAssignableFrom (t);
    }
}

Обратите внимание, что это форматирует значения с использованием текущего языка и региональных параметров (за исключением дат). Если вы хотите использовать инвариантную культуру, это достаточно легко изменить.

Перед его вызовом необходимо отключить отслеживание объектов, чтобы избежать кеширования объектов в памяти, установив для ObjectTrackingEnabled значение false:

ObjectTrackingEnabled = false;
Customers.WriteCsv (@"c:\\temp\\customers.csv");
2
Joe Albahari 10 Янв 2013 в 11:09
Это не выводит свойства, допускающие значение NULL (поскольку typeof (IFormattable) .IsAssignableFrom (p.PropertyType)) возвращает false)
 – 
sgmoore
9 Янв 2013 в 19:52
Похоже, что Util.WriteCSV бета-сборки - это еще один способ (я предполагаю, что вы встроили это в результате этого и подобных сообщений), хотя в любом случае он, похоже, не работает на вкладке SQL. Я собираюсь отметить это как ответ, хотя мне придется потрудиться и потренироваться, чтобы заставить его работать в моей ситуации.
 – 
Elsimer
11 Янв 2013 в 18:00