当前位置:首页 > c++ > 正文内容

为什么64位windows上访问系统目录时64位程序转到System32,而32位程序转到了SysWOW64?

xuwenyan11个月前 (07-13)c++868

我们通常使用SHGetKnownFolderPathGetSystemDirectory来获取System目录,这里使用GetSystemDirectory演示。

我们会发现GetSystemDirectory无论是32位程序调用还是64位程序调用,所得到的目录都是C:WindowsSystem32

 

 

这看起来32位程序和64位程序访问系统目录并没有什么区别,但是当需要操作文件时就会不同了,我们在系统目录创建一个文件演示一下(注意操作系统需要管理员权限)

 

 

我们可以看到32位程序和64位程序同样在系统目录创建test.text文件,但最终所处的目录是不同的,这是因为windows文件重定向器把32位程序重定向到了SysWOW64目录,关于windows文件重定向器详情参考微软文档:文件系统重定向器

概述文件系统重定向器

%windir%System32 目录是为 64 位 Windows 上的 64 位应用程序保留的,而32位程序就会被重定向到%windir%SysWOW64,为了兼容性,应用程序不应直接使用路径。相反,应该调用API获取路径。

原始路径32 位 x86 进程的重定向路径32 位 ARM 进程的重定向路径
%windir%System32%windir%SysWOW64%windir%SysArm32
%windir%lastgoodsystem32%windir%lastgoodSysWOW64%windir%lastgoodSysArm32
%windir% egedit.exe%windir%SysWOW64 egedit.exe%windir%SysArm32 egedit.exe

如何禁用文件重定向

应用程序可以使用Wow64DisableWow64FsRedirectionWow64RevertWow64FsRedirection函数来控制 WOW64 文件系统重定向器。禁用文件系统重定向会影响调用线程执行的所有文件操作,因此只有在需要单个CreateFile调用时才应禁用它,并在函数返回后立即重新启用它。长时间禁用文件系统重定向会阻止 32 位应用程序加载系统 DLL,从而导致应用程序失败。

 

文件系统重定向器避坑

如果执行进程导致系统显示 UAC 提示,则不会发生重定向。而是启动所请求文件的 64 位版本。为防止出现此问题,请指定 SysWOW64 目录以避免重定向并确保访问该文件的 32 位版本,或者以管理员权限运行 32 位应用程序,以便不显示 UAC 提示。关于如何获取SysWOW64 目录,请使用GetSystemWow64DirectorySHGetKnownFolderPath

某些子目录免于重定向。对这些子目录的访问不会重定向到 %windir%SysWOW64:%windir%system32catroot
%windir%system32catroot2
%windir%system32driverstore
%windir%system32driversetc
%windir%system32logfiles
%windir%system32spool

    文章作者:xuwenyan
    版权声明:本文为本站原创文章,转载请注明出处,非常感谢,如版权漏申明或您觉得任何有异议的地方欢迎与本站取得联系。

    扫描二维码推送至手机访问。

    版权声明:本文由艺文笔记发布,如需转载请注明出处。

    本文链接:https://www.xuwenyan.com/archives/2772

    分享给朋友:

    “为什么64位windows上访问系统目录时64位程序转到System32,而32位程序转到了SysWOW64?” 的相关文章

    C++从dll导出lib

    C++从dll导出lib

    一、使用VC++的工具DUMPBIN将DLL中的导出函数表导出到一定义(.DEF)文件EXAMPLE: DUMPBIN VideoDeCoder.dll /EXPROTS /OUT:VideoDeCoder.def 二、将导出的.DEF文件整理为一符合.DEF个数的函数导出文件EX...

    C++指针*为什么靠后会比较好?

    C++指针*为什么靠后会比较好?

    大多数书中和大神的代码里,往往指针的*都是靠变量而不是靠类型的,这主要是为了不造成我们第一眼对变量类型的误解和对指针类型的误解,比如: int* p1,p2 我们一眼看上去是不是通常会觉得p1、p1都是一个int*的指针呢?因为我们通常会误把int*当作一个类型,然而无论int*还是i...

    C++如何获取控制台程序的输出内容?

    C++如何获取控制台程序的输出内容?

    很多工具程序(如ffmpeg)的进度显示往往都是以控制台字符显示的方法,我们可能需要调用这种控制台工具去完成工作,但同时又希望以友好的ui界面去显示当前的工作状态(如进度)。此时我们能想到的就是运行控制台程序,然后以某种方式去获取到控制台程序的输出,然后转换到我们的ui界面上去显示。 有多种...

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    现象:程序在xp系统上面总是莫名其妙的崩溃,检查代码看不出任何问题,感觉代码都很好。即使你远程调试,找到了崩溃的点,当你注释了崩溃点之后,还是会崩溃到别的地方。当你遇到了这种情况的时候,不妨参照一下下面的方法看看,说不定可以解决问题。如何解决?将崩溃程序相关的所有工程代码全部关闭全局变量的线程安全检...

    ATL实现windows右键菜单扩展(ContextMenu)

    ATL实现windows右键菜单扩展(ContextMenu)

    右键菜单,即用户右击shell对象时弹出的上下文菜单(context menu)。本文记录了如何创建右键菜单的基本过程,跟着步骤一步一步来,即可创建出一个右键菜单工程。第一步,新建一个ATL工程Visual Studio—>新建项目—>ATL—>使用默认配置(一直按下一步即可)。注...

    uafxcwd.lib(afxmem.obj) : error LNK2005:

    uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)"解决办法

    如果在编译MFC程序的时候出现下列及类似的错误: 1˃uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)" (??2@YAPAXI@Z) 已经在 LIBCMTD.lib(new...