Как проверить, реализует ли класс методы интерфейса в 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

так почему бы не сделать это прямо сейчас.

Другие вопросы по тегам