Windows核心编程句柄和伪句柄 CHRIS

摘要:
句柄的使用起源于早期的Windows,当时只允许在内存有限的机器中使用。使用此方法,Windows可以根据需要安全地移动对象,然后更新引用中的指针,从而允许进程安全地访问对象。此时,Windows将返回对象的实际地址,以便进程可以安全地读取和写入对象。只要锁定计数器大于0,Windows就不会移动对象。当Windows从单一的分时程序变成多任务操作系统时,需要在新的操作系统上运行这么多遗留代码,所以在这次更改中保留了句柄。
GetCurrentProcess(), DuplicateHandle()
Window中为什么会有句柄的概念:
从Visual C++的头文件来看,HANDLE被typedef为void的指针,那是指向未确定数据结构的指针:typedef void* HANDLE;
但是这并不说明任何问题,因为句柄远远不只是指向任意数据类型的指针。它是指向数据对象指针的指针。句柄的使用来源于早期的Windows,当时它只能在有限内存的机器中允许(因为当时内存昂贵也存储小)。为了留出足够的空间内存以运行其他程序,Windows经常将对象在内存中移动。但是如果进程已经有了指向该对象的指针,移动该对象就将使得指针无效。
为了处理这一问题,Microsoft使用系统指针以跟踪对象。程序不是直接使用指针,而是使用句柄,让它引用包含真实对象的地址。通过这一方法,Windows可以安全按照需要移动对象,然后更新参考中的指针,从而允许进程安全地访问对象。
当进程需要使用句柄访问对象时,它调用GlobalLocl()以锁定内存的对象。这时Windows将返回对象的实际地址,从而进程可以安全读写该对象。只要对象被进程锁定,Windows就不会去移动它。句柄包含的不只是对象的地址;比如,它还包含锁计数器,标识着多少进程已经请求过对象的地址。只要这个锁计数器大于0,Windows就不会移动该对象。
这个思想扩展到大部分对象上,包括文件,也由其句柄标识。
这时,句柄就是指向对象的指针,而该对象包含指向另一个对象的指针。这对于现在的机器有些奇怪。当Windows从单个分时共享程序向多任务操作系统转变时,有如此之多的遗留代码需要在新的操作系统上运行,于是句柄在这一转变中保留下来了。 
 
GetCurrentProcess():

GetCurrentProcess得到得到的称之为"伪句柄"
只是一个标识,你可以发现,其实就是返回$FFFFFFFF,
每个进程得句柄都是一样
得,只是实用于进程内部得使用.
如果你想得到实际得句柄,在进程间进行通讯,必需要进行转化,
调用DuplicateHandle,注意,得实句柄使用完成以后,你必须要调用CloseHandle去关闭.
其实,你应该明白了为何"伪句柄"得存在,就是使用简单,不用关闭,
不会造成内存泄漏.

同样道理,GetCurrentThread也是伪句柄,其值永远是$FFFFFFFE,只是适用于线程内部得使用.



DuplicateHandle() :

在系统中,对象分两类:核心对象和用户对象.如进程对象,线程对象,文件映射
对象等就是核心对象;而向窗口,菜单等都是用户对象.
两者是有差别的,用于标示用户对象的句柄是系统唯一的,也就是说,一个进程
完全可以对另外一个进程中的用户对象进行操作,比如两个进程间通信的方法之一,
就是发送消息.正是由于窗口是用户对象,所以句柄是系统唯一,通过FindWindow(),
得到另外一个进程的窗口句柄,然后用SendMessage(),让hWnd的窗口过程来处理消
息,实现了进程间的通信.因此,对于用户对象,你根本不用DuplicateHandle(),直接
把句柄拿来用就行了.
而核心对象则不一样.核心对象是为了加强系统的稳定性,因此,
核心对象句柄是
进程相关的
,在每一个进程中都有一个核心对象表,每一个对象的索引(不完全是)作为内和对象的句柄,从而实现进程相关.同一个对象在不同的进程中可能有不同的索引,即句柄.对核心对象进行操作时,系统还要进行安全检验,看一下你是否有权来操作这个对象.因此你不能同用户对象一样,直接把句柄拿过来用.比方说,你想操作另一个进程中的文件映射对象,这个文件映射对象句柄在那个进程中假设是0x000001,但在你的进程中,很有可能0x00000001时表示另一个核心对象,此时的操作就永远不会成功,甚至会产生灾难性的后果.此时,就有必要用DuplicateHandle().

免责声明:文章转载自《Windows核心编程句柄和伪句柄 CHRIS》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇浏览器的渲染原理干了3年程序员,我开窍了下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Linux内核分析总结

周子轩 原创作品转载注明出处 《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000 笔记: 冯诺依曼体系结构的核心思想是存储程序计算机。在计算机中有两种指令,一是用户指令,一是系统调用。 Linux是一个基于POSIX和UNIX的多用户、多任务、支持多线程和多CPU的操作系统...

Linux 进程间通信(一)

Linux 进程间通信   进程是一个独立的资源分配单位,不同进程之间的资源是相互独立的,没有关联,不能在一个进程中直接访问另一个进程中的资源。但是,进程不是孤立的,不同的进程之间需要信息的交换以及状态的传递,因此需要进程间数据传递、同步与异步的机制。 分类 统一主机间进程通信 Unix进程间通信方式 无名通道 有名通道 信号 System V进...

Win32汇编操作注册表

1.打开和关闭子键 注册表函数对注册表的操作是通过句柄来完成的,与文件操作一样,在对某个键下的子键或者键值项进行操作之前,需要先将这个键打开,然后使用键句柄来引用这个键,在操作完毕以后再将键句柄关闭。注册表的根键不需要打开,它们的句柄是固定不变的,要使用根键的时候只要把这些句柄直接拿来用就是了,Windows.inc中已经预定义了它们的数值: HKEY_C...

Domain Socket本地进程间通信

socket API原本是为网络通讯设计的,但后来在socket的框架上发展出一种IPC机制,就是UNIX Domain Socket。虽然网络socket也可用于同一台主机的进程间通讯(通过loopback地址127.0.0.1),但是UNIX Domain Socket用于IPC更有效率:不需要经过网络协议栈,不需要打包拆包、计算校验和、维护序号和应答...

Java关于数组对象赋值与指针

在实现PageRank算法中犯了两个错误, 1:在对double类型赋值时,除法中没有将分母设置为double类型,而是int类型,导致出现商为0的结果错误,导致推迟 2:当定义两个数组时,对数组赋值时,忘记了,数组是对象的特点直接nowRank=resultRank; 其中resultRank又重新赋值,所以导致nowRank中元素值也发生变化,因为数组...

VC 串口通讯基本原理,讲的很是详细

目 录打开串口........................................................................................................................ 2关闭串口.................................................