未能加载文件或程序集,Windows结构化异常处理浅

图片 1图片 2

近期一直被一个问题所困扰,就是写出来的程序老是出现无故崩溃,有的地方自己知道可能有问题,但是有的地方又根本没办法知道有什么问题。更苦逼的事情是,我们的程序是需要7x24服务客户,虽然不需要实时精准零差错,但是总不能出现断线丢失数据状态。故刚好通过处理该问题,找到了一些解决方案,怎么捕获访问非法内存地址或者0除以一个数。从而就遇到了这个结构化异常处理,今就简单做个介绍认识下,方便大家遇到相关问题后,首先知道问题原因,再就是如何解决。废话不多说,下面进入正题。

图片 3

  1 /*---------------------------------------------
  2 CHECKER3.C -- Mouse Hit-Test Demo Program No.3
  3               (c) Charles Petzold, 1998
  4 ---------------------------------------------*/
  5 
  6 #include <Windows.h>
  7 
  8 #define DIVISIONS 5
  9 
 10 LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
 11 LRESULT CALLBACK ChildWndProc(HWND, UINT, WPARAM, LPARAM);
 12 
 13 TCHAR szChildClass[] = TEXT("Checker3_Child");
 14 
 15 int WINAPI WinMain( __in HINSTANCE hInstance
 16                     , __in_opt HINSTANCE hPrevInstance
 17                     , __in LPSTR lpCmdLine
 18                     , __in int nShowCmd )
 19 {
 20     static TCHAR szAppName[] = TEXT("Checker3");
 21     HWND hwnd;
 22     MSG msg;
 23     WNDCLASS wndclass;
 24 
 25     wndclass.style = CS_HREDRAW | CS_VREDRAW;
 26     wndclass.lpfnWndProc = WndProc;
 27     wndclass.cbClsExtra = 0;
 28     wndclass.cbWndExtra = 0;
 29     wndclass.hInstance = hInstance;
 30     wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
 31     wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
 32     wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
 33     wndclass.lpszMenuName = NULL;
 34     wndclass.lpszClassName = szAppName;
 35 
 36     if (!RegisterClass(&wndclass))
 37     {
 38         MessageBox(NULL, TEXT("Program requires Windows NT!")
 39             , szAppName, MB_ICONERROR);
 40         return 0;
 41     }
 42 
 43     wndclass.lpfnWndProc = ChildWndProc;
 44     wndclass.cbWndExtra = sizeof(long);
 45     wndclass.hIcon = NULL;
 46     wndclass.lpszClassName = szChildClass;
 47 
 48     RegisterClass(&wndclass);
 49 
 50     hwnd = CreateWindow(szAppName, TEXT("Checker3 Mouse Hit-Test Demo")
 51         , WS_OVERLAPPEDWINDOW
 52         , CW_USEDEFAULT, CW_USEDEFAULT
 53         , CW_USEDEFAULT, CW_USEDEFAULT
 54         , NULL, NULL, hInstance, NULL);
 55 
 56     ShowWindow(hwnd, nShowCmd);
 57     UpdateWindow(hwnd);
 58 
 59     while (GetMessage(&msg, NULL, 0, 0))
 60     {
 61         TranslateMessage(&msg);
 62         DispatchMessage(&msg);
 63     }
 64 
 65     return msg.wParam;
 66 }
 67 
 68 LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
 69 {
 70     static HWND hwndChild[DIVISIONS][DIVISIONS];
 71     int cxBlock, cyBlock, x, y;
 72 
 73     switch (message)
 74     {
 75     case WM_CREATE:
 76         for (x = 0; x != DIVISIONS; ++x)
 77             for (y = 0; y != DIVISIONS; ++y)
 78             {
 79                 hwndChild[x][y] = CreateWindow(szChildClass, NULL
 80                     , WS_CHILDWINDOW | WS_VISIBLE
 81                     , 0, 0, 0, 0
 82                     , hwnd, (HMENU)(y << 8 | x)
 83                     , (HINSTANCE)GetWindowLong(hwnd, GWL_HINSTANCE)
 84                     , NULL);
 85             }
 86         return 0;
 87 
 88     case WM_SIZE:
 89         cxBlock = LOWORD(lParam) / DIVISIONS;
 90         cyBlock = HIWORD(lParam) / DIVISIONS;
 91 
 92         for (x = 0; x != DIVISIONS; ++x)
 93             for(y = 0; y != DIVISIONS; ++y)
 94             {
 95                 MoveWindow(hwndChild[x][y]
 96                     , x * cxBlock, y * cyBlock
 97                     , cxBlock, cyBlock, TRUE);
 98             }
 99         return 0;
100 
101     case WM_LBUTTONDOWN:
102         MessageBeep(0);
103         return 0;
104 
105     case WM_DESTROY:
106         PostQuitMessage(0);
107         return 0;
108     }
109 
110     return DefWindowProc(hwnd, message, wParam, lParam);
111 }
112 
113 LRESULT CALLBACK ChildWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
114 {
115     HDC hdc;
116     PAINTSTRUCT ps;
117     RECT rect;
118 
119     switch (message)
120     {
121     case WM_CREATE:
122         SetWindowLong(hwnd, 0, 0);    // on/off flag
123         return 0;
124 
125     case WM_LBUTTONDOWN:
126         SetWindowLong(hwnd, 0, 1 ^ GetWindowLong(hwnd, 0));
127         InvalidateRect(hwnd, NULL, FALSE);
128         return 0;
129 
130     case WM_PAINT:
131         hdc = BeginPaint(hwnd, &ps);
132 
133         GetClientRect(hwnd, &rect);
134         Rectangle(hdc, 0, 0, rect.right, rect.bottom);
135 
136         if (GetWindowLong(hwnd, 0))
137         {
138             MoveToEx(hdc, 0, 0, NULL);
139             LineTo(hdc, rect.right, rect.bottom);
140             MoveToEx(hdc, 0, rect.bottom, NULL);
141             LineTo(hdc, rect.right, 0);
142         }
143 
144         EndPaint(hwnd, &ps);
145         return 0;
146     }
147 
148     return DefWindowProc(hwnd, message, wParam, lParam);
149 }

什么是结构化异常处理

结构化异常处理(structured exception handling,下文简称:SEH),是作为一种系统机制引入到操作系统中的,本身与语言无关。在我们自己的程序中使用SEH可以让我们集中精力开发关键功能,而把程序中所可能出现的异常进行统一的处理,使程序显得更加简洁且增加可读性。

使用SHE,并不意味着可以完全忽略代码中可能出现的错误,但是我们可以将软件工作流程和软件异常情况处理进行分开,先集中精力干重要且紧急的活,再来处理这个可能会遇到各种的错误的重要不紧急的问题(不紧急,但绝对重要)

当在程序中使用SEH时,就变成编译器相关的。其所造成的负担主要由编译程序来承担,例如编译程序会产生一些表(table)来支持SEH的数据结构,还会提供回调函数。

注:
不要混淆SHE和C++ 异常处理。C++ 异常处理再形式上表现为使用关键字catchthrow,这个SHE的形式不一样,再windows Visual C++中,是通过编译器和操作系统的SHE进行实现的。

在所有 Win32 操作系统提供的机制中,使用最广泛的未公开的机制恐怕就要数SHE了。一提到SHE,可能就会令人想起 *__try__finally* 和 *__except* 之类的词儿。SHE实际上包含两方面的功能:终止处理(termination handing)异常处理(exception handing)

revit二次开发中遇到的问题

CHECKER3.C

终止处理

终止处理程序确保不管一个代码块(被保护代码)是如何退出的,另外一个代码块(终止处理程序)总是能被调用和执行,其语法如下:

__try
{
    //Guarded body
    //...
}
__finally
{
    //Terimnation handler
    //...
}

**__try __finally** 关键字标记了终止处理程序的两个部分。操作系统和编译器的协同工作保障了不管保护代码部分是如何退出的(无论是正常退出、还是异常退出)终止程序都会被调用,即**__finally**代码块都能执行。

RevitAPIUI.dll 只能 Native Library 中执行; 脱离了Native Library,API是跑不起来的 。

本文由澳门新葡亰平台官网发布于操作系统,转载请注明出处:未能加载文件或程序集,Windows结构化异常处理浅

TAG标签:
Ctrl+D 将本页面保存为书签,全面了解最新资讯,方便快捷。