Вопрос по c# – Установка свойства с помощью отражения со строковым значением

283

Я хотел бы установить свойство объекта через Reflection со значением типаstring. So, for instance, suppose I have a Ship класс, со свойствомLatitude, который являетсяdouble.

Вот что я хотел бы сделать:

Ship ship = new Ship();
string value = "5.5";
PropertyInfo propertyInfo = ship.GetType().GetProperty("Latitude");
propertyInfo.SetValue(ship, value, null);

Как это, это бросаетArgumentException:

Object of type 'System.String' cannot be converted to type 'System.Double'.

Как я могу преобразовать значение в правильный тип, основанный наpropertyInfo?

Вопрос для вас: это часть заказного решения ORM? user3308043

Ваш Ответ

11   ответов
-7

Вы хотите поиграть с Reflection или создать производственное программное обеспечение? Я хотел бы спросить, почему вы используете отражение для установки свойства.

Double new_latitude;

Double.TryParse (value, out new_latitude);
ship.Latitude = new_latitude;
Вы должны уважать то, что люди пытаются делать, а не то, что, по вашему мнению, они должны делать. Downvoted. (ОтGenericProgramming.exe:ReflectionBenefits() )
6

Я попробовал ответ отLBushkin и он работал отлично, но он не будет работать для нулевых значений и пустых полей. Поэтому я изменил это на следующее:

propertyName= "Latitude";
PropertyInfo propertyInfo = ship.GetType().GetProperty(propertyName);
if (propertyInfo != null)
{
     Type t = Nullable.GetUnderlyingType(propertyInfo.PropertyType) ?? propertyInfo.PropertyType;
     object safeValue = (value == null) ? null : Convert.ChangeType(value, t);
     propertyInfo.SetValue(ship, safeValue, null);
}
471

Ты можешь использоватьConvert.ChangeType() - Это позволяет использовать информацию времени выполнения на любомIConvertible Тип, чтобы изменить форматы представления. Однако не все преобразования возможны, и вам может потребоваться написать специальную логику, если вы хотите поддерживать преобразования из типов, которые не являютсяIConvertible.

Соответствующий код (без обработки исключений или логики специального случая) будет:

Ship ship = new Ship();
string value = "5.5";
PropertyInfo propertyInfo = ship.GetType().GetProperty("Latitude");
propertyInfo.SetValue(ship, Convert.ChangeType(value, propertyInfo.PropertyType), null);
Прочтите ответ @AliKaraca ниже. И этот, и приведенный ниже, быстрые и свободные, но выполняют работу для распространенных типов.
Не работает для гидов.
18

Я заметил, что многие люди рекомендуютConvert.ChangeType - Однако в некоторых случаях это работает, как только вы начинаете привлекатьnullable типы вы начнете получатьInvalidCastExceptions:

http://weblogs.asp.net/pjohnson/archive/2006/02/07/Convert.ChangeType-doesn_2700_t-handle-nullables.aspx

Оболочка была написана несколько лет назад, чтобы справиться с этим, но это тоже не идеально.

http://weblogs.asp.net/pjohnson/archive/2006/02/07/Convert.ChangeType-doesn_2700_t-handle-nullables.aspx

2

Если вы пишете приложение Metro, вы должны использовать другой код:

Ship ship = new Ship();
string value = "5.5";
PropertyInfo propertyInfo = ship.GetType().GetTypeInfo().GetDeclaredProperty("Latitude");
propertyInfo.SetValue(ship, Convert.ChangeType(value, propertyInfo.PropertyType));

Замечания:

ship.GetType().GetTypeInfo().GetDeclaredProperty("Latitude");

вместо

ship.GetType().GetProperty("Latitude");
5

С помощьюConvert.ChangeType и получить тип для преобразования изPropertyInfo.PropertyType.

propertyInfo.SetValue( ship,
                       Convert.ChangeType( value, propertyInfo.PropertyType ),
                       null );
2

Я отвечу на это общим ответом. Обычно эти ответы не работают с гидами. Вот рабочая версия с направляющими тоже.

var stringVal="6e3ba183-89d9-e611-80c2-00155dcfb231"; // guid value as string to set
var prop = obj.GetType().GetProperty("FooGuidProperty"); // property to be setted
var propType = prop.PropertyType;

// var will be type of guid here
var valWithRealType = TypeDescriptor.GetConverter(propType).ConvertFrom(stringVal); 
Это должен быть принятый ответ. Он также работает с GUID & lt; 3. Спасибо, Али (это прозвище моей дочери)
6

Вы, вероятно, ищетеConvert.ChangeType метод. Например:

Ship ship = new Ship();
string value = "5.5";
PropertyInfo propertyInfo = ship.GetType().GetProperty("Latitude");
propertyInfo.SetValue(ship, Convert.ChangeType(value, propertyInfo.PropertyType), null);
11

Вы можете использовать конвертер типов (без проверки ошибок):

Ship ship = new Ship();
string value = "5.5";
var property = ship.GetType().GetProperty("Latitude");
var convertedValue = property.Converter.ConvertFrom(value);
property.SetValue(self, convertedValue);

С точки зрения организации кода, вы можете создатьсвоего рода миксин это приведет к следующему коду:

Ship ship = new Ship();
ship.SetPropertyAsString("Latitude", "5.5");

Это будет достигнуто с помощью этого кода:

public interface MPropertyAsStringSettable { }
public static class PropertyAsStringSettable {
  public static void SetPropertyAsString(
    this MPropertyAsStringSettable self, string propertyName, string value) {
    var property = TypeDescriptor.GetProperties(self)[propertyName];
    var convertedValue = property.Converter.ConvertFrom(value);
    property.SetValue(self, convertedValue);
  }
}

public class Ship : MPropertyAsStringSettable {
  public double Latitude { get; set; }
  // ...
}

MPropertyAsStringSettable может быть многократно использован для разных классов.

Вы также можете создать свой собственныйпреобразователи типа прикрепить к вашим свойствам или классам:

public class Ship : MPropertyAsStringSettable {
  public Latitude Latitude { get; set; }
  // ...
}

[TypeConverter(typeof(LatitudeConverter))]
public class Latitude { ... }
Есть ли какая-то особая причина, по которой вы добавили маркерный интерфейс вместо того, чтобы просто использоватьobject?
Да, интерфейс маркера служит заполнителем для добавления методов расширения. С помощьюobject добавил бы методы расширения ко всем классам, что обычно нежелательно.
3

Или вы можете попробовать:

propertyInfo.SetValue(ship, Convert.ChangeType(value, propertyInfo.PropertyType), null);

//But this will cause problems if your string value IsNullOrEmplty...
32

Как уже говорили несколько других, вы хотите использоватьConvert.ChangeType:

propertyInfo.SetValue(ship,
    Convert.ChangeType(value, propertyInfo.PropertyType),
    null);

На самом деле, я рекомендую вам взглянуть на весьConvert Class.

Этот класс и многие другие полезные классы являются частьюSystem Namespace, Я считаю полезным сканировать это пространство имен каждый год или около того, чтобы увидеть, какие функции я пропустил. Попробуйте!

Хорошая точка зрения. Я отредактирую и укажу на настоящих ответчиков, или удалю мой, если кто-то добавит то, что я сказал об остальном пространстве имен.
ОП, вероятно, хочет получить общий ответ для установки свойства любого типа, которое имеет явное преобразование из строки.

Похожие вопросы