VSIX - Невозможно загрузить файл или сборку ссылочной библиотеки DLL

Мой вопрос очень похож на этот, только то, что ответ и обходные пути не работают для меня. Также я в Visual Studio 2012.

У меня есть VSPackage, который ссылается на другой проект, который зависит от других библиотек. Каждый раз, когда я запускаю свой пакет в отладке, я получаю исключение, что другие библиотеки не могут быть найдены. Они находятся в выходном каталоге, и они подписаны.

Я пытался ссылаться на них напрямую в проекте VSPackage, но безрезультатно.

Мысли?

1 ответ

Решение

Эта проблема существует, потому что Visual Studio не ищет сборки в папке вашего расширения, если ваше расширение не имеет явной зависимости от этих сборок. Например, зависимости, установленные в файле конфигурации (IoC config) или в коде xaml. Я знаю три решения этой проблемы:

  1. Вы можете развернуть эти сборки в GAC, и Visual Studio загрузит их. Этот метод хорош, если вы используете стороннюю библиотеку, созданную для использования в GAC (например, MS Enterprise Library). Но VSIX Deployment Package не позволяет устанавливать сборки в GAC, вы можете использовать установщик MSI.

  2. Для VSPackages для Visual Studio 2010/2012 вы можете использовать атрибут ProvideBindingPath. Путь, в котором находится ваше расширение, будет добавлен в пути, которые Visual Studio использует для поиска зависимых сборок. Если ваше расширение не включает VSPackage, вы можете добавить этот атрибут к любому общедоступному классу ( см. Здесь).

    [ProvideBindingPath] 
    public class MyVsPackage : Package 
    { /* ... */ }
    
  3. Вы можете вручную разрешить имена сборок. Для этого вам нужно подписаться на событие AssemblyResolve и вам необходимо вернуть требуемые сборки из обработчика. Это наиболее гибкий способ, если вы не можете использовать предыдущие методы, это специально для вас.

    В моем проекте IntelliDebugger я написал для него класс ManualAssemblyResolver:

using System;
using System.Reflection;

namespace IntelliEgg.Debugger.Utility
{
    public class ManualAssemblyResolver : IDisposable
    {
        public ManualAssemblyResolver(Assembly assembly)
        {
            if (assembly == null)
                throw new ArgumentNullException("assembly");

            _assemblies = new[] {assembly};
            AppDomain.CurrentDomain.AssemblyResolve += OnAssemblyResolve;
        }

        public ManualAssemblyResolver(params Assembly[] assemblies)
        {
            if (assemblies == null)
                throw new ArgumentNullException("assemblies");

            if (assemblies.Length == 0)
                throw new ArgumentException("Assemblies should be not empty.", "assemblies");

            _assemblies = assemblies;
            AppDomain.CurrentDomain.AssemblyResolve += OnAssemblyResolve;
        }

        public void Dispose()
        {
            AppDomain.CurrentDomain.AssemblyResolve -= OnAssemblyResolve;
        }

        private Assembly OnAssemblyResolve(object sender, ResolveEventArgs args)
        {
            foreach (Assembly assembly in _assemblies)
            {
                if (args.Name == assembly.FullName)
                {
                    return assembly;
                }
            }

            return null;
        }

        private readonly Assembly[] _assemblies;
    }
}

Этот класс должен быть создан до первого вызова проблемной сборки (например, в методе Package::Initialize())

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