事先验证是否可以找到组装

问题描述

| 我的.net程序正在使用程序集,应将其安装在GAC中。 如果该组件安装正确,则在我第一次使用该组件时将立即加载它,这很棒。例如,如果我使用如下所示的程序集类型:
ESRI::ArcGIS::esriSystem::AoInitialize^  aoi = gcnew ESRI::ArcGIS::esriSystem::AoInitializeClass();
程序集将被加载。 现在,有时该程序集将不存在于GAC中,我需要程序知道这一点,但不会崩溃。 我尝试将程序集的用法包装到try / catch块中,如下所示:
try
{
  ESRI::ArcGIS::esriSystem::AoInitialize^  aoi = gcnew  ESRI::ArcGIS::esriSystem::AoInitializeClass();
}
catch (System::Exception^ )
{
  // assembly not found
}
但是程序不会引发异常,而是会崩溃。 我如何预先检查组件是否在GAC中并且可以正常使用而不会崩溃?或者如何捕捉崩溃并禁用程序中的相应菜单项?     

解决方法

        您对程序集的加载方式有错误的思维模型。在程序集中创建类型的实例时,不会发生这种情况。 JIT编译器是首先需要汇编的程序。为该方法生成机器代码。如果查看异常的调用堆栈,则会看到该异常是在调用方法中引发的。 至少,这是当您使用Microsoft抖动时通常发生的情况,它会在最后可能的时刻按需生成机器代码。与其他抖动(例如Mono抖动)不同,它的编译更加积极。 您可以在调用方法中捕获异常,但这非常脆弱。除了抖动依赖性之外,您还必须使用[MethodImpl(MethodImplOptions.NoInlining)]来赋予该方法属性,以确保该代码永远不会内联。因为那将再次导致过早需要该类型。 正确的解决方法是使用插件体系结构。您必须使用接口类型来声明要在主程序中使用的属性和方法。该接口类型需要在单独的程序集中声明,同时由主程序和插件引用。并与您的主程序一起部署,无论客户是否可以使用插件选项。使用Assembly.Load()加载插件程序集,并使用Assembly.CreateInstance()创建实现接口的类型的具体实例。而且永远不要在代码中引用具体类型,而只引用接口类型。您会在Google查询中找到许多此类示例。     ,        最好的方法是在安装过程中添加所有必需的程序集。 但是,可以在创建类型之前通过简单的类型检查来实现所需的功能。 像这样:
var t = System.Type.GetType(\"TypeName,Assembly\");
if (t == null) throw CannotLoadTypeException();