Зарегистрируйте C # COM Object, не имея реального файла для RegAsm - PullRequest
0 голосов
/ 19 апреля 2011

У меня есть странный механизм загрузки приложений.У меня есть исполняемый exe-файл со всеми остальными DLL и самим приложением внутри его ресурсов.Эти файлы (сборки) извлекаются из ресурсов и загружаются по требованию, потому что я присоединяюсь к событию Assemblyresolve текущего AppDomain.

 [STAThread]
        static void Main()
        {
            // if the command line contains extract then extract the dlls, next run will resolve assemblies from disk
            bool saveDllsToDisk = new List<string>(Environment.GetCommandLineArgs()).Contains("extract");
            // if the command line contains bin then use the bin folder instead of temp to extract dlls
            bool useBinFolder = new List<string>(Environment.GetCommandLineArgs()).Contains("bin");

            if (!Directory.Exists(tempFolder)) {
                Directory.CreateDirectory(tempFolder);
            }

            // the assembly resolver will get here because it will not find the dlls in the bin folder
            // we load assemblies in our specific way:
            // - if exists in our temp/bin folder load from there
            // - else load from resources
            // - if specified extract DLLs to the temp/bin folder
            AppDomain.CurrentDomain.AssemblyResolve += (sender, args2) =>
            {
                string name = new AssemblyName(args2.Name).Name;
                Debug.WriteLine("START LOADING " + name);

                Assembly assembly = null;
                string folder = useBinFolder ? binFolder : Path.Combine(tempFolder, APP_NAME);
                string fileName = name.Replace(".","_").Replace("#EXE#", ""); // in resources we use _ instead of .
                string extension = name.Contains("#EXE#") ? "exe" : "dll"; // hack for our embedded exe files
                name = name.Replace("#EXE#", ""); // hack for our embedded exe files

                if (File.Exists(Path.Combine(folder,String.Format("{0}.{1}", name, extension))))
                {
                    // load from file in app temp folder
                    assembly = Assembly.LoadFile(Path.Combine(folder, String.Format("{0}.{1}", name, extension)));
                }
                else
                {
                    // extract assembly from resources
                    byte[] assemblyBytes = (byte[])resMan.GetObject(fileName, CultureInfo.InvariantCulture);
                    assembly = Assembly.Load(assemblyBytes);

                    // if selected save to file so the next run JIT will resolve from disk
                    if (saveDllsToDisk)
                    {
                        string outDll = Path.Combine(folder, String.Format("{0}.{1}", name, extension));
                        using (var fs = File.Create(outDll))
                        {
                            fs.Write(assemblyBytes, 0, assemblyBytes.Length);
                        }
                    }
                }

                Debug.WriteLine("LOADED " + name);
                return assembly;
            };

            splashScreen = new frmSplash();
            // as soon as splashscreen starts animating assembly preloading will be launched on this eventhandler
            splashScreen.Started += new EventHandler(splash_Started);
            // splashscreen has finished fadein we must now wait for all libraries to be preloaded and set CanContinue
            splashScreen.Finishing += new EventHandler(splash_Finishing);
            splashScreen.CanContinue = false;

            // run splashscreen while preloading of DLLs is going on
            Application.Run(splashScreen);
        }

Этот загрузчик имеет только одну ссылку и это основное приложение,но он не ссылается на него в основном методе, чтобы избежать поиска сборки при запуске.То, что я делаю, это принудительная загрузка сборок во время анимации заставки:

ObjectHandle instance;

                string[,] assemblies = {
                     {"WindowsBase, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35","dummy"}, // will go from GAC, not our resolver
                     {"PresentationCore, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35","dummy"}, // will go from GAC, not our resolver
                     {"PresentationFramework, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35","dummy"}, // will go from GAC, not our resolver
                     {"WindowsFormsIntegration, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35","dummy"}, // will go from GAC, not our resolver
                     {"PresentationFramework.Aero, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35","dummy"}, // will go from GAC, not our resolver
                     {"System.Core, Version=3.5.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089","dummy"}, // will go from GAC, not our resolver                     
                     {"AvalonDock, Version=1.3.3585.0, Culture=neutral, PublicKeyToken=85a1e0ada7ec13e4", "dummy"}, // out reference inside resources
                     {"AvalonDock.Themes, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null", "dummy"}, // out reference inside resources
                     {"ICSharpCode.AvalonEdit, Version=4.0.0.5950, Culture=neutral, PublicKeyToken=9cc39be672370310", "dummy"}, // out reference inside resources
                     {"WPG, Version=2.0.4120.37542, Culture=neutral, PublicKeyToken=null","dummy"}, // out reference inside resources
                     {"WPGBrushEditor, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null","dummy"}, // out reference inside resources
                     {"HLSLEditor, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null","dummy"} // preload even our main file that will launch after the splashscreen
                };

                for (int i = 0; i <= assemblies.GetUpperBound(0); i++)
                {
                    try
                    {
                        instance = AppDomain.CurrentDomain.CreateInstance(assemblies[i,0], assemblies[i,1]);
                    }
                    catch (Exception ex) {
                        /* must not raise errors, it will fail because we are not actually asking for a
                         *  valid type and we only need this assembly loaded right now*/
                    }
                }

Теперь вы видите, что в любой папке нет фактических dll-файлов, но это сборки, непосредственно загруженные из ресурсов.

Моя проблема сейчас: Как зарегистрировать COM-объект, отключенный в одном из моих DLL-файлов?

RegAsm использует filepaths для регистрации COM-объекта ...: (*

Любая помощьоценили!

Ответы [ 2 ]

1 голос
/ 27 апреля 2011

Я делаю нечто очень похожее в Excel-DNA .Вам не нужно десериализовать типы COM на диск или регистрировать их заранее.Что я сделал, это определил класс, который реализует IClassFactory, а затем зарегистрировал его в качестве текущей фабрики классов для конкретного CLSID (типа), вызвав CoRegisterClassObject.Когда COM пытается создать объект, вызывается ваша фабрика классов, которая может вернуть экземпляр вашего типа .NET.

В зависимости от контекста вам, возможно, придется работать с регистрацией ProgID / CLSID на лету.

(Если вы свяжетесь со мной, я буду рад помочь вам получить правильные фрагменты кода Excel-DNA, чтобы начать работу.)

0 голосов
/ 20 апреля 2011

Я немного покопался и, возможно, реализовал DllRegisterServer () в своем основном исполняемом файле, а затем зарегистрировал бы все типы COM во всех встроенных DLL-библиотеках ...

Кто-нибудь может подтвердить?

РЕДАКТИРОВАТЬ: Да, это может быть сделано, но основной исполняемый файл должен быть COM-прокси для DLL внутри.

...