LINQ .Cast()扩展方法失败,但(type)object 有效?

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (2)
  • 关注 (0)
  • 查看 (77)

为了在一些LINQtoSQL对象和DTO之间进行转换,我们在DTO上创建了显式的强制转换操作符。这样我们就可以做以下工作:

DTOType MyDTO = (LinqToSQLType)MyLinq2SQLObj;

这个效果很好。

但是,当您尝试使用LINQ进行强制转换时。它控制一个无效的强制转换异常,不能将Linq2SQLType类型转换为DTOType。也就是说,下面的内容不起作用

List<DTO.Name> Names = dbContact.tNames.Cast<DTO.Name>()
                                               .ToList();

但是,下面的方法很好:

DAL.tName MyDalName = new DAL.tName();
DTO.Name MyDTOName = (DTO.Name)MyDalName;

下面的内容也很好

List<DTO.Name> Names = dbContact.tNames.Select(name => (DTO.Name)name)
                                               .ToList();

为什么。CAST()扩展方法引发无效的强制转换异常?我用过。CAST()扩展方法在过去多次使用,当您将类似于基类型的东西转换为派生类型时,它工作得很好,但是当对象有显式CAST运算符时,它就会掉下来。

提问于
用户回答回答于

Cast<>扩展方法不应用用户定义的转换。它只能转换到接口或提供类型的类继承中。

用户定义的转换是根据表达式中涉及的静态类型在编译时标识的。它们不能作为运行时转换应用,因此下列操作是非法的:

public class SomeType
{
  public static implicit operator OtherType(SomeType s) 
  { 
    return new OtherType(); 
  }
}

public class OtherType { }

object x = new SomeType();
OtherType y = (OtherType)x; // will fail at runtime

UDC是否存在于SomeTypeOtherType-不能通过引用类型来适用object尝试运行上述代码将在运行时失败,报告如下:

System.InvalidCastException: 
    Unable to cast object of type 'SomeType' to type 'OtherType'

Cast<>()只能执行保持表示的转换...这就是为什么不能使用它应用用户定义的转换。

用户回答回答于

如果对Linq程序集进行反编译,则会得到类似于以下代码的代码。最终的转换是从“对象”到目标类型,对于自定义类型总是失败的。

private static IEnumerable<TResult> CastIterator<TResult>( IEnumerable source )
{
    foreach(object current in source)
    {
        yield return (TResult)( (object)current );
    }
    yield break;
}

public static IEnumerable<TResult> DCast<TResult>( this IEnumerable source )
{
    IEnumerable<TResult> enumerable = source as IEnumerable<TResult>;
    if(enumerable != null)
    {
        return enumerable;
    }
    if(source == null)
    {
        throw new ArgumentNullException( "source" );
    }
    return CastIterator<TResult>( source );
}

TFISH

扫码关注云+社区

领取腾讯云代金券