我已成功通过使用:
从SyntaxNode收到了一个ITypeSymbolSemanticModel.GetTypeInfo(sytaxNode).ConvertedType
现在我想知道这个ITypeSymbol
是否与我执行代码中存在的System.Type
实例相对应,例如typeof(IEnumerable<int>)
或someObject.GetType()
。
我试过
typeInfo.ConvertedType.ToString() == type.ToString()
但是这些不使用相同的格式规则,例如IEnumerable<int>
TypeInfo.ToString() == "System.Collections.Generic.IEnumerable<int>"
而
typeof(IEnumerable<int>).ToString() == "System.Collections.Generic.IEnumerable`1[System.Int32]"
此外,我认为最好比较AssemblyQualifiedNames而不仅仅是命名空间和类型名称,以避免可能的名称冲突。
理想情况下,我希望能够在执行代码中获取与我从语义模型中获得的ITypeInfo相对应的实际System.Type实例(假设所需的程序集已加载和/或可用)。这将允许检查类型是否可以从其他类型等分配。
答案 0 :(得分:7)
您可以使用INamedTypeSymbol
获取Compilation.GetTypeByMetadataName()
类型名称。
所以试试这个:
semanticModel.GetTypeInfo(sytaxNode).ConvertedType.Equals(
semanticModel.Compilation.GetTypeByMetadataName(typeof(WhateverType).FullName))
这不适用于封闭的泛型类型,对于那些你需要做更多的事情。例如:
var ienumerableType = semanticModel.Compilation.GetTypeByMetadataName("System.Collections.Generic.IEnumerable`1");
var intType = semanticModel.Compilation.GetTypeByMetadataName("System.Int32");
var type = ienumerableType.Construct(intType);
答案 1 :(得分:4)
基于@Tamas的答案,我创建了以下递归解决方案,适用于封闭的泛型类型。
static bool TypeSymbolMatchesType(ITypeSymbol typeSymbol, Type type, SemanticModel semanticModel)
{
return GetTypeSymbolForType(type, semanticModel).Equals(typeSymbol);
}
static INamedTypeSymbol GetTypeSymbolForType(Type type, SemanticModel semanticModel)
{
if (!type.IsConstructedGenericType)
{
return semanticModel.Compilation.GetTypeByMetadataName(type.FullName);
}
// get all typeInfo's for the Type arguments
var typeArgumentsTypeInfos = type.GenericTypeArguments.Select(a => GetTypeSymbolForType(a, semanticModel));
var openType = type.GetGenericTypeDefinition();
var typeSymbol = semanticModel.Compilation.GetTypeByMetadataName(openType.FullName);
return typeSymbol.Construct(typeArgumentsTypeInfos.ToArray<ITypeSymbol>());
}