Как проверить, реализует ли класс методы интерфейса в NRefactory
У меня есть два файла. Одним из них является объявление класса, а другим - объявление интерфейса. Класс должен реализовывать интерфейс. Как я могу проверить в NRefactory, реализует ли класс методы интерфейса?
Я должен дать больше деталей.
Первый файл - например:
class Test : IF
{
}
а второй
interface IF
{
void Foo();
}
Я должен прочитать эти файлы и разобрать с NRefactory. Мне нужно проверить, реализует ли класс Test метод из интерфейса IF.
Без компиляции и загрузки скомпилированной сборки.
3 ответа
Я нашел решение в NRefactory коде. Я изменил это, чтобы достичь своих целей. Сначала мы должны реализовать посетителя, который проверяет, реализует ли класс каждый метод из интерфейса:
public class MissingInterfaceMemberImplementationVisitor : DepthFirstAstVisitor
{
private readonly CSharpAstResolver _resolver;
public bool IsInterfaceMemberMissing { get; private set; }
public MissingInterfaceMemberImplementationVisitor(CSharpAstResolver resolver)
{
_resolver = resolver;
IsInterfaceMemberMissing = false;
}
public override void VisitTypeDeclaration(TypeDeclaration typeDeclaration)
{
if (typeDeclaration.ClassType == ClassType.Interface || typeDeclaration.ClassType == ClassType.Enum)
return;
base.VisitTypeDeclaration(typeDeclaration);
var rr = _resolver.Resolve(typeDeclaration);
if (rr.IsError)
return;
foreach (var baseType in typeDeclaration.BaseTypes)
{
var bt = _resolver.Resolve(baseType);
if (bt.IsError || bt.Type.Kind != TypeKind.Interface)
continue;
bool interfaceMissing;
var toImplement = ImplementInterfaceAction.CollectMembersToImplement(rr.Type.GetDefinition(), bt.Type, false, out interfaceMissing);
if (toImplement.Count == 0)
continue;
IsInterfaceMemberMissing = true;
}
}
}
И теперь мы должны прочитать все файлы, проанализировать их и вызвать вышеупомянутый класс следующим образом:
var solutionFiles = new List<FileInfo>();
var trees = new Dictionary<FileInfo, SyntaxTree>();
IProjectContent projectContent = new CSharpProjectContent();
foreach (var file in solutionFiles.Where(f => f.Extension == ".cs").Distinct())
{
var parser = new ICSharpCode.NRefactory.CSharp.CSharpParser();
SyntaxTree syntaxTree;
using (var fs = new FileStream(file.FullName, FileMode.Open, FileAccess.Read, FileShare.Read, 4096, FileOptions.SequentialScan))
{
syntaxTree = parser.Parse(fs, file.FullName);
}
trees.Add(file, syntaxTree);
var unresolvedFile = syntaxTree.ToTypeSystem();
projectContent = projectContent.AddOrUpdateFiles(unresolvedFile);
}
var compilation = projectContent.CreateCompilation();
foreach (var sharpFile in trees)
{
var originalResolver = new CSharpAstResolver(compilation, sharpFile.Value, sharpFile.Value.ToTypeSystem());
var visitor = new MissingInterfaceMemberImplementationVisitor(originalResolver);
sharpFile.Value.AcceptVisitor(visitor);
if (visitor.IsInterfaceMemberMissing)
return false;
}
return true;
Использовать is
ключевое слово
http://msdn.microsoft.com/en-us/library/scekt9xw.aspx
if (myObj is IMyInterface) {
....
}
Кроме того, вы также можете использовать as
ключевое слово.
class MyClass { }
interface IInterface { }
MyClass instance = new MyClass();
var inst = instance as IInterface;
if (inst != null)
{
// your class implements the interface
}
Используйте это, когда вам нужно использовать этот экземпляр позже, без дополнительной ссылки. С помощью is
вам все еще нужно сделать
if (instance is IInterface)
{
var inst = (IInterface) instance;
}
или же
(instance as IInterface).InstanceProperty
так почему бы не сделать это прямо сейчас.