在传递给使用泛型的 class 时修改 IEnumerable<T> 或 ICollection<T>
Modifying an IEnumerable<T> or ICollection<T> when passed to a class using generic types
我在使用泛型类型时修改我的 IEnumerable<> 值时遇到了一些麻烦。
下面是一些代码:
public class ParameterArray<TEnumerable, TType> : IParameterArray<TEnumerable, TType>
where TEnumerable : ICollection<TType> // Note that I have tried with IEnumerable<TType> instead of ICollection<TType>, but did not really help
{
public ParameterArray(TEnumerable value, TType minValue, TType maxValue)
{
MinValue = minValue;
MaxValue = maxValue;
Value = value;
}
private TEnumerable _value;
public TEnumerable Value
{
get => _value;
set
{
ICollection<TType> checkedValue;
// Here I want to check the range before setting the value
// Note that Value can be of type int[], float[], bool[], byte[]
// I have tried many work around, but for some reasons, I am never able to update _value
for (int idx = 0; idx < value.Count(); idx++)
{
// Backup value after range limiting
checkedValue.Add(CheckRange(value.ElementAt(idx)));
}
// Would like to save these new values in _value, but not sure how to
_value = checkedValue as TEnumerable; // Or (TEnumerable)checkedValue; // !!! Breaking here at runtime - Cannot convert from one type to the other
}
}
public TType CheckRange(TType value)
{
TType retValue = value;
if (value != null)
{
// Restrict range to Min/Max
if (Comparer<TType>.Default.Compare(MaxValue, value) < 0)
retValue = MaxValue;
else if (Comparer<TType>.Default.Compare(MinValue, value) > 0)
retValue = MinValue;
}
else
{
throw new ArgumentOutOfRangeException($"Value cannot be null: Range is {MinValue} to {MaxValue}");
}
return retValue;
}
}
不知何故,我无法将我的 ICollection 备份到我的 _value(这是一个 TEnumerable)中。但是,考虑到我已声明 TEnumerable 派生自 ICollection(即 TEnumerable : ICollection)
,我不太明白原因
我尝试了多种方法,但它们都会导致相同的问题,即无法将我的结果分配给 _value。
欢迎提出任何建议。
也许我需要的不是:where TEnumerable : ICollection
编辑和解决方案
按照@nvoigt 的建议,我重新格式化了我的 class,这样它就不会将 TEnumerable 作为输入。这种额外的抽象层产生了额外的复杂性,使一切都难以处理。
这是最终的解决方案:
public class ParameterArray<TType> : ParameterBase, IParameterArray<TType>
{
public ParameterArray(IEnumerable<TType> value, TType minValue, TType maxValue)
{
MinValue = minValue;
MaxValue = maxValue;
Value = value;
}
private IEnumerable<TType> _value;
public IEnumerable<TType> Value
{
get => _value;
set
{
IList<TType> checkedValue = new List<TType>();
for (int idx = 0; idx < value.Count(); idx++)
{
// Range limiting
checkedValue.Add(CheckRange(value.ElementAt(idx)));
}
// Back up the updated value
_value = checkedValue.ToArray();
}
}
public TType CheckRange(TType value)
{
// Unchanged
...
}
}
ckeckedValue as TEnumerable
让我们看看。假设我使 TEnumerable
成为 int[]
并且 ckeckedValue
成为 ICollection<int>
那么您的编译器是正确的...您不能使用 as
运算符从相同基类型的 ICollection
。这不管用,不管是否是泛型。
我真的不能推荐一个好的方法。如果是我,我会放弃通用的可枚举部分,将其设为 List<>
或 Array<>
并完成它。如果您有明确的约束 要求 该部分是通用的,您可能应该 post 他们提出问题,因此我们可以根据这些约束提出不同的解决方案。
我在使用泛型类型时修改我的 IEnumerable<> 值时遇到了一些麻烦。
下面是一些代码:
public class ParameterArray<TEnumerable, TType> : IParameterArray<TEnumerable, TType>
where TEnumerable : ICollection<TType> // Note that I have tried with IEnumerable<TType> instead of ICollection<TType>, but did not really help
{
public ParameterArray(TEnumerable value, TType minValue, TType maxValue)
{
MinValue = minValue;
MaxValue = maxValue;
Value = value;
}
private TEnumerable _value;
public TEnumerable Value
{
get => _value;
set
{
ICollection<TType> checkedValue;
// Here I want to check the range before setting the value
// Note that Value can be of type int[], float[], bool[], byte[]
// I have tried many work around, but for some reasons, I am never able to update _value
for (int idx = 0; idx < value.Count(); idx++)
{
// Backup value after range limiting
checkedValue.Add(CheckRange(value.ElementAt(idx)));
}
// Would like to save these new values in _value, but not sure how to
_value = checkedValue as TEnumerable; // Or (TEnumerable)checkedValue; // !!! Breaking here at runtime - Cannot convert from one type to the other
}
}
public TType CheckRange(TType value)
{
TType retValue = value;
if (value != null)
{
// Restrict range to Min/Max
if (Comparer<TType>.Default.Compare(MaxValue, value) < 0)
retValue = MaxValue;
else if (Comparer<TType>.Default.Compare(MinValue, value) > 0)
retValue = MinValue;
}
else
{
throw new ArgumentOutOfRangeException($"Value cannot be null: Range is {MinValue} to {MaxValue}");
}
return retValue;
}
}
不知何故,我无法将我的 ICollection 备份到我的 _value(这是一个 TEnumerable)中。但是,考虑到我已声明 TEnumerable 派生自 ICollection(即 TEnumerable : ICollection)
,我不太明白原因我尝试了多种方法,但它们都会导致相同的问题,即无法将我的结果分配给 _value。
欢迎提出任何建议。
也许我需要的不是:where TEnumerable : ICollection
编辑和解决方案
按照@nvoigt 的建议,我重新格式化了我的 class,这样它就不会将 TEnumerable 作为输入。这种额外的抽象层产生了额外的复杂性,使一切都难以处理。
这是最终的解决方案:
public class ParameterArray<TType> : ParameterBase, IParameterArray<TType>
{
public ParameterArray(IEnumerable<TType> value, TType minValue, TType maxValue)
{
MinValue = minValue;
MaxValue = maxValue;
Value = value;
}
private IEnumerable<TType> _value;
public IEnumerable<TType> Value
{
get => _value;
set
{
IList<TType> checkedValue = new List<TType>();
for (int idx = 0; idx < value.Count(); idx++)
{
// Range limiting
checkedValue.Add(CheckRange(value.ElementAt(idx)));
}
// Back up the updated value
_value = checkedValue.ToArray();
}
}
public TType CheckRange(TType value)
{
// Unchanged
...
}
}
ckeckedValue as TEnumerable
让我们看看。假设我使 TEnumerable
成为 int[]
并且 ckeckedValue
成为 ICollection<int>
那么您的编译器是正确的...您不能使用 as
运算符从相同基类型的 ICollection
。这不管用,不管是否是泛型。
我真的不能推荐一个好的方法。如果是我,我会放弃通用的可枚举部分,将其设为 List<>
或 Array<>
并完成它。如果您有明确的约束 要求 该部分是通用的,您可能应该 post 他们提出问题,因此我们可以根据这些约束提出不同的解决方案。