Прежде чем я попытаюсь ответить, я вынужден указать, что то, что вы делаете, кажется излишним.Предполагая, что вы помещаете этот код в конструктор, генерировать что-то вроде:
public class Foo
{
private int a;
private bool b;
private SomeType c;
public Foo()
{
this.a = default(int);
this.b = default(bool);
this.c = default(SomeType);
}
}
не нужно.Это уже происходит автоматически при создании класса.(Фактически, некоторые быстрые тесты показывают, что эти назначения даже не оптимизируются, если они выполняются явно в конструкторе, хотя я полагаю, что JITter мог бы позаботиться об этом.)
Во-вторых, default
Ключевое слово было разработано в значительной степени для того, чтобы делать именно то, что вы делаете: чтобы предоставить способ присвоить значение «по умолчанию» переменной, тип которой неизвестен во время компиляции.Я предполагаю, что он был введен для использования универсальным кодом, но автоматически сгенерированный код, безусловно, также правильно его использует.
Имейте в виду, что значение default
ссылочного типа равно null
,поэтому
this.list = default(List<int>);
не создает новый List<int>
, он просто устанавливает this.list
в null
.Я подозреваю, что вы хотите сделать вместо этого, использовать свойство Type.IsValueType
, чтобы оставить типы значений в их значениях по умолчанию, и инициализировать ссылочные типы, используя new
.
Наконец, я думаю, что выздесь ищем свойство IsGenericType
класса Type
и соответствующий метод GetGenericArguments()
:
foreach (PropertyInfo property in properties)
{
if (property.Type.IsGenericType)
{
var subtypes = property.Type.GetGenericArguments();
// construct full type name from type and subtypes.
}
else
{
code += "this." + property.Name + " = default(" + property.PropertyType.Name + ")";
}
}
РЕДАКТИРОВАТЬ:
Что касается конструирования чего-то полезного для ссылочного типа, то, как я видел, используемый сгенерированный код распространенный метод - требовать конструктор без параметров для любого класса, который вы ожидаете использовать.Достаточно просто увидеть, есть ли у класса конструктор без параметров, вызвав Type.GetConstructor()
, передав пустой Type[]
(например, Type.EmptyTypes
), и посмотреть, вернет ли он ConstructorInfo
или null
.Как только это будет установлено, просто заменив default(typename)
на new typename()
, вы получите то, что вам нужно.
В более общем случае вы можете предоставить любой массив типов этому методу, чтобы увидеть, есть ли соответствующий конструктор, или вызвать GetConstructors()
чтобы получить их всех.Здесь нужно обратить внимание на поля IsPublic
, IsStatic
и IsGenericMethod
поля ConstructorInfo
, чтобы найти поле, которое вы действительно можете вызвать из любого места, где генерируется этот код.
Проблемавы пытаетесь решить, однако, он станет произвольно сложным, если вы не можете наложить на него некоторые ограничения.Одним из вариантов может быть поиск произвольного конструктора и создание вызова, который выглядит следующим образом:
var line = "this." + fieldName + " = new(";
foreach ( var param in constructor.GetParameters() )
{
line += "default(" + param.ParameterType.Name + "),";
}
line = line.TrimEnd(',') + ");"
(Обратите внимание, что это только для иллюстративных целей, я бы, вероятно, использовал здесь CodeDOM или, по крайней мере, StringBuilder:)
Но, конечно, теперь у вас есть проблема определения подходящего имени типа для каждого параметра, которые сами могут быть обобщенными.И все параметры ссылочного типа будут инициализированы нулем.И нет никакого способа узнать, какой из произвольно большого количества конструкторов, из которых вы можете выбрать, на самом деле производит полезный объект (некоторые из них могут делать плохие вещи, например, предположить, что вы собираетесь устанавливать свойства или вызывать методы сразу после создания экземпляра).
То, как вы решаете эти проблемы, не является техническим: вы можете рекурсивно применять эту же логику к каждому параметру настолько низко, насколько захотите.Для вашего случая использования необходимо решить, насколько сложными вам должны быть и какие ограничения вы хотите установить для пользователей.