32位程序如何在WIN64位下运行

本文探讨了在64位Windows系统环境下,如何使32位程序能够正常运行并连接ODBC数据库及ASP。通过分析系统结构和路径配置,提出了解决方案包括切换IIS运行模式、调整ASP.net运行模式、修改Oracle客户端为32位模式等步骤,最终实现程序与数据库的正常连接。

 

环境

  32位程序如何在windows 64位下运行(64位下ODBC与ASP数据库连接)

       一般系统部署的服务器若是windows系统,就会采用64位win2003的结构。可是我们编写的程序绝大多数都是在x86下32位cpu架构中编译的,要正常移植到64位机器还真的是很麻烦,不仅要求应用程序是64位模式编译,还需要数据库也得是64位,iis64位,framework64位,好在相应的厂商都提供这些支持组件。不知道有人遇到过像我这样的问题么,应用中有需要增加一个Access数据库导入功能,这就需要连接ODBC的mdb驱动,可是Microsoft OLE DB Provider for Jet不支持64位系统,即使安装了office2007后,发现ODBC也没有任何变化,这下尴尬了。

  分析

  对64位系统结构进行分析。

  c盘下有两个程序安装目录:Program Files和Program Files (x86);

  windows文件夹下有两个目录:System32和SysWOW64;

  在System32中的odbcad32.exe中没有其他驱动(这也是控制面板中ODBC直接启动的程序),但是在SysWOW64下的odbcad32.exe却存在mdb之类的所有驱动,这说明ODBC驱动在操作系统中是存在的,只是因为运行模式导致调用的odbc程序不同,造成找不到对应的驱动。了解到这一点,就能找到解决方案:切换64位运行模式为32位。

  解决步骤

  切换IIS运行模式为32位

  开始→运行→cmd,运行脚本:cscript %SYSTEMDRIVE%\inetpub\adminscripts\adsutil.vbs SET W3SVC/AppPools/Enable32bitAppOnWin64 1,即可将IIS由64位切换为32位模式;若想恢复为64位模式,运行脚本:cscript %SYSTEMDRIVE%\inetpub\adminscripts\adsutil.vbs SET W3SVC/AppPools/Enable32bitAppOnWin64 0。若脚本报错,切换路径为C:\Inetpub\AdminScripts,因为adsutil.vbs文件在该路径下。

  Asp.net运行模式为32位

  重新注册asp.net,开始→运行→cmd,运行脚本:%SYSTEMROOT%\Microsoft.NET\Framework\v2.0.40607\aspnet_regiis.exe -i。若要注册64位,脚本为:%SYSTEMROOT%\Microsoft.NET\Framework64\v2.0.40607\aspnet_regiis.exe -i。

  重启机器,IIS即可运行32位的程序,ODBC的连接也生效了。

  Oracle64位切换为32位的方法

  若以前系统运行在64位模式下使用的是64位的oracle,修改为32位模式后程序的数据库连接就失败,会报:“如果在安装 32 位 Oracle 客户端组件的情况下以 64 位模式运行……”这个错误。想到的方法自然是再安装32位的oracle客户端了。其实最简单的方法是下载ORALCE工具包Instant Client Package。将其解压到任意文件夹下,如D:\oracle\instantclient_11_2。

  设置对应的环境变量:右击”我的电脑” – “属性” – “高级” – “环境变量” – “系统环境变量”:

  1、新建NLS_lANG=SIMPLIFIED CHINESE_CHINA.ZHS16GBK(注意中间有空格,这个如果不设,toad查询中文会是乱码)

  2、新建或修改TNS_ADMIN=D:\oracle\instantclient_11_2

  3、修改Path增加D:\oracle\instantclient_11_2,同时删除原来64位oracle的路径

  其中第一条很重要,解决运行时报:Ora-12705: cannot access nls data files or invalid environment specified的错误。

  新建文本文件,修改后缀名为reg,内容为:

  Windows Registry Editor Version 5.00

以下是代码片段:
    [HKEY_LOCAL_MACHINE\SOFTWARE\ORACLE\KEY_OraClient10g_home1]
  ”NLS_LANG”=”SIMPLIFIED CHINESE_CHINA.ZHS16GBK”
  ”ORACLE_BUNDLE_NAME”=”Standard”
  ”ORACLE_GROUP_NAME”=”Oracle – OraClient10g_home1″
  ”ORACLE_HOME”=”D:\oracle\instantclient_11_2”
  ”ORACLE_HOME_KEY”=”SOFTWARE\\ORACLE\\KEY_OraClient10g_home1″
  ”ORACLE_HOME_NAME”=”OraClient10g_home1″

  双击导入注册表。

  将64位oracle的tns文件拷入D:\oracle\instantclient_11_2中,否则运行会报:ORA-12154: TNS:could not resolve the connect identifier specified错误。

函数功能初始化一个临界资源对象。该函数无返回值。单进程的各个线程可 以使用临界资源对象来解决同步互斥问题,该对象不能保证哪个线程能够获得到 临界资源对象,该系统能公平的对待每一个线程。lpCriticalSection 临界资源对象 指针,该参数是一个out类型。 EnterCriticalSection(__inout LPCRITICAL_SECTION lpCriticalSection); 该函数用以获得指定的临界区的对象的所有权,该函数等待临界区对象的所 有权,如果该所有权赋予了调用线程,则该函数返回,否则该函数会一直等待, 从而导致线程等待。 LeaveCriticalSection(__inout LPCRITICAL_SECTION lpCriticalSection); 线程使用完所保护的资源后,需要调用该函数,释放指定的临界区对象的所 有权,这时候其他想要获得该临界区对象所有权的线程就可以获得该所有权,从 而进入关键代码段,访问保护的资源。 DeleteCriticalSection(__inout LPCRITICAL_SECTION lpCriticalSection); 该函数释放一个没有被任何线程所拥有的临界区对象的所有资源。 LeaveCriticalSection(&g_cs); EnterCriticalSection(&g_cs);  三、实验代码: #include <windows.h>//必要的头文件,使用Windows API函数 #include <stdio.h>  int tickets = 100; //票数 HANDLE hMutex; //使用全局的相互排斥对象来保证对同一资源 的相互排斥訪问与操作这里是tickets CRITICAL_SECTION g_cs; //使用关键代码方法 //线程1,2 的入口函数 DWORD WINAPI Fun1Proc(LPVOID lpParameter); DWORD WINAPI Fun2Proc(LPVOID lpParameter); void main() {   HANDLE hThread1;   DWORD thread1ID;   //创建线程1   hThread1 = CreateThread(NULL, 0, Fun1Proc, NULL, 0, &thread1ID);   HANDLE hThread2;   DWORD thread2ID;   //创建线程2   hThread2 = CreateThread(NULL, 0, Fun2Proc, NULL, 0, &thread2ID);   CloseHandle(hThread1); //关闭线程的句柄,这样当线程结束时,线程内核对象 被释放   CloseHandle(hThread2); //创建一个相互排斥对象,假设成功返回相互排斥对象的句柄,否则返回NULL   hMutex = CreateMutex(NULL, FALSE, "tickets");   if (hMutex)     {       if(ERROR_ALREADY_EXISTS == GetLastError())         {  puts("only one instance can run!");           return;         }     }   InitializeCriticalSection(&g_cs); //初始化临界资源对 象 Sleep(4000);//主线程睡眠4秒钟,给其他线程执行的时间   system("pause");//暂停防止程序退出 }  DWORD WINAPI Fun1Proc(LPVOID lpParameter) {   while(TRUE)   {     Sleep(1);            if(tickets > 0)       {       printf("Thread1 sell tickets : %d\n", tickets );       tickets=tickets-1;            }     else       break;        }   return 0; } DWORD WINAPI Fun2Proc(LPVOID lpParameter)        {   while(TRUE)   {     Sleep(1);     if (tickets > 0)     {         printf("Thread2 sell tickets : %d\n", tickets );       tickets=tickets-1;            }     else       break;     }   return 0; }  说明:初始化代码以及主函数已经给出,(主函数中,红色的代 码是第二种同步方法的定义及初始化方法,如果采用第一种方 法,可以删除)请在Fun1proc和Fun2proc函数中填入合适的 置填同步方法(说明中的红色代码可以直接使用)使进程同步 四、实验的运行结果 运行结果截图,不同步和同步的效果图
最新发布
12-03
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值