С# generics, когда T может быть массивом
Я пишу обертку С# для сторонней библиотеки, которая считывает как одиночные значения, так и массивы с аппаратного устройства, но всегда возвращает массив object [] даже для одного значения. Это требует повторных вызовов объекта [0], когда я хочу, чтобы конечный пользователь мог использовать generics для получения массива или одного значения.
Я хочу использовать generics, чтобы вызывающий мог использовать оболочку следующими способами:
MyWrapper<float> mw = new MyWrapper<float>( ... );
float value = mw.Value; //should return float;
MyWrapper<float[]> mw = new MyWrapper<float[]>( ... );
float[] values = mw.Value; //should return float[];
В MyWrapper у меня есть свойство Value в настоящее время следующим образом:
public T Value
{
get
{
if(_wrappedObject.Values.Length > 1)
return (T)_wrappedObject.Value; //T could be float[]. this doesn't compile.
else
return (T)_wrappedObject.Values[0]; //T could be float. this compiles.
}
}
Я получаю ошибку компиляции в первом случае:
Невозможно преобразовать тип 'объект []' to 'T'
Если я изменю MyWrapper.Value на T [], я получаю:
Невозможно преобразовать тип 'object []' to 'T []'
Любые идеи о том, как достичь моей цели? Спасибо!
Ответы
Ответ 1
Изменить: обновленный ответ. Библиотека возвращает массив объектов, вы не сможете просто вернуть его в виде T, будь то массив или один элемент, не выполняя некоторую работу с ним. Нижеприведенная функция является примером взятия массива объектов и возврата его как массива или одного элемента.
public static T GetValue<T>(object[] inputs)
{
if (typeof(T).IsArray)
{
Type elementType = typeof(T).GetElementType();
Array array = Array.CreateInstance(elementType, inputs.Length);
inputs.CopyTo(array, 0);
T obj = (T)(object)array;
return obj;
}
else
{
return (T)inputs[0];
// will throw on 0-length array, check for length == 0 and return default(T)
// if do not want exception
}
}
Пример потребления:
object[] inputs = { 1f, 2f, 3f }; // what the library is returning
float[] array = GetValue<float[]>(inputs); // what you want?
float singleValue = GetValue<float>(inputs); // what you want?
Ответ 2
Вам нужно обмануть компилятор, сначала наведя массив в объект
public T Value
{
get
{
if(_wrappedObject.Values.Length > 1)
return (T)(object)_wrappedObject.Value; //T could be float[]. this doesn't compile.
else
return (T)_wrappedObject.Values[0]; //T could be float. this compiles.
}
}