虽然handle和handler只有一个字符之差,但在计算机的世界里,含义却大相径庭。

1. 先说说handle

北京话说"一边儿玩儿去,玩勺子把儿去","勺子把儿"说的就是handle。而将handle译成"句柄"绝对是一个相当文雅相当阳春白雪的翻译,因为太文绉绉啦,很多文化底蕴不够的码农就看不大懂了或者望而生畏。为了弄明白为什么这么整,我费了点儿周折。 句柄者,弯杆杆儿,弯把手儿也。注意: "句"为"勾"的通假字,句柄应该读作gou柄才是。《说文解字》对"句"的解释是"句, 曲也"。《说文解字注》(作者:清代学者段玉裁,简称"段注")里是这么说的"凡曲折之物,侈为倨,敛为句。考工记多言倨句。" 因此,如果将handle翻译成大白话"弯把手儿",进不得教科书,也写不进那些晦涩乏味的计算机图书。那么,程序员当如何理解handle呢?简单来说,handle就是一个"带把儿"的物件的那个"把儿"。

1.1 handle的本来含义

A handle is a part of, or attachment to, an object that can be moved or used by hand.

例如:

o 带橡胶handle的现代拔钉锤(A modern claw hammer with rubber handle 【图片来源: https://en.wikipedia.org/wiki/File:Claw-hammer.jpg】)

o 带handle的平底锅

1.2 handle在计算机世界里的含义

A handle is an abstract reference to a resource.

+1: A handle is a unique identifier for an object managed by Windows.
+2: A handle can be anything from an integer index to a pointer to a resource
in kernel space. The idea is that they provide an abstraction of a resource,
so you don't need to know much about the resource itself to use it.

注: +1 is from "what-is-a-windows-handle"; +2 is from "what-is-a-handle-in-c"

例如: (在Unix/Linux系统中) 【鉴于个人对windows了解甚少,故不谈windows】

进程号pid就是一个handle,

文件描述符(fd)也是一个handle,

系统调用号(syscall num)仍然是一个handle,

... 不胜枚举。

在操作系统中,一切对用户来说是透明(注:这里的"透明"指的是"看不见摸不着就如空气一样"而不是"一览无余毫无秘密可言")的但是操作系统内核看得懂的无符号整数(unsigned int)都可以被看作是handle。

在操作系统设计与实现中,联系内核态和用户态,靠的就是一个个无符号整数。因为用数字来做通信密码(比如:操作码,错误码等)实在是太方便了。而且,一个unsigned int占4个字节,可以表征的通信密码总数为2^32(=4G, 约40亿)。 如果不用无符号整数来做通信密码,而是采用可读性很好的明文(字符串"string")来做通信,那是何等的情何以堪?! 因为,计算机做字符串比较的代价要远远大于无符号整数的比较。

好啦,扯远了,一句话,下次看到"句柄",不用害怕啦。因为它就是handle, 说白了就是跟一个黑盒子进行通信的密码。一旦通信密码传给了黑盒子,黑盒子具体怎么操作,对持有handle的用户来说,完全不用关心。"不看过程,只看结果"就得了。 古人云"微曲为倨,甚曲为句",将"handle"翻译成"句柄",还是有一定道理的,因为用户程序拿到的handle,通常并不能够径直通向真实的内核资源,而是需要"绕个弯儿",也就是被内核映射成一个指向内核资源的首地址的pointer才能够访问真实的内核资源。

2. 什么是handler

在编程中使用过信号(signal)的朋友一定跟handler不会陌生。 例如:

$ man -s2 signal
NAME
signal - ANSI C signal handling SYNOPSIS
#include <signal.h> typedef void (*sighandler_t)(int); sighandler_t signal(int signum, sighandler_t handler);
...

hanlder就是一个回调函数(callback)。当某个事件到达时,事先注册的handler会被接收到事件的主体所调用。 示例代码:

o foo.c

 #include <stdio.h>
#include <signal.h>
#include <unistd.h> unsigned int g_flag = ; static void foo_handler(int signum)
{
printf("signal %d is caught, %s is called\n", signum, __func__);
g_flag++;
} int main(int argc, char *argv[])
{
signal(SIGUSR1, foo_handler); while (!g_flag)
sleep();
printf("good bye\n"); return ;
}

o 编译并测试

T1$ gcc -g -Wall -m32 -o foo foo.c

T1$ ./foo

T2$ ps -ef | grep foo | grep -v grep
veli : pts/ :: ./foo T2$ kill -SIGUSR1 The output from T1 looks like: T1$ ./foo
signal 10 is caught, foo_handler is called
good bye

维基百科对handler的解释是这样的,

Handler, an asynchronous callback (computer programming) subroutine in computing
...
Event handler, a routine for processing a programming event
Interrupt handler, a routine for processing CPU interrupts
Signal handler, a routine for handling signals sent to a process
Exception handler, a routine for handling software exceptions

而维基百科对handle的解释是这样的,

In computer programming, a handle is an abstract reference to a resource. 
Handles are used when application software references blocks of memory or
objects managed by another system, such as a database or an operating system.
A resource handle can be an opaque identifier, in which case it is often an
integer number (often an array index in an array or "table" that is used to
manage that type of resource), or it can be a pointer that allows access to
further information. Common resource handles are file descriptors, network sockets,
database connections, process identifiers (PIDs), and job IDs.
Process IDs and job IDs are explicitly visible integers, while file descriptors
and sockets (which are often implemented as a form of file descriptor) are
represented as integers, but are typically considered opaque. In traditional
implementations, file descriptors are indices into a (per-process) file
descriptor table, thence a (system-wide) file table.

3. 总结

  • A handle  is an abstract reference to a resource. Handle是对某个资源的抽象引用。
  • A handler is an asynchronous callback subroutine. Handler则是一个异步的回调函数(子程序)。

附注: 《柯林斯高阶英语学习词典》对handle和handler的解释(供参考并帮助理解其在计算机世界里的含义)

A handle  is the part of an object such as a tool, bag, or cup that you hold in order to be able to pick up and use the object.
A handler is someone whose job is to deal with a particular type of object.

闲话handle和handler的更多相关文章

  1. Jquery报错:Uncaught TypeError: ((m.event.special[e.origType] || (intermediate value)).handle || e.handler).apply is not a function

    页面中出现了Jquery报错:Uncaught TypeError: ((m.event.special[e.origType] || (intermediate value)).handle || ...

  2. jquery , find the event handler,找到jquery中的event handler

    找到 dispatch: function (e) { e = b.event.fix(e); var n, r, i, s, o, u = [], a = d.call(arguments), f ...

  3. Go Web:Handler

    Multiplexer根据URL将请求路由给指定的Handler.Handler用于处理请求并给予响应.更严格地说,用来读取请求体.并将请求对应的响应字段(respones header)写入Resp ...

  4. Android: ListView数据的分批加载 以及 Handle 总结

    这是效果图: activity_main.xml 01 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/ ...

  5. Android中的Handler及它所引出的Looper、MessageQueue、Message

    0.引入 0.1.线程间通信的目的 首先,线程间通信要交流些什么呢? 解答这个问题要从为什么要有多线程开始,需要多线程的原因大概有这些 最早也最基本:有的任务需要大量的时间,但其实并不占用计算资源,比 ...

  6. I/O Handler的管理(3)

    另外连接地址:http://blogs.readthedocs.org/zh_CN/latest/Handler_mgr.html 本章目录 I/O Handler的管理 IO句柄与Select_Re ...

  7. Golang 接口型函数和http.Handler接口

    一.接口型函数 参考Golang必备技巧:接口型函数 1.原始接口实现 type Handler interface { Do(k, v interface{}) } func Each(m map[ ...

  8. Handle详解

    首先通过一个函数启动一个服务器,只提供一个方法并返回Hello World!,当你在浏览器输入http://127.0.0.1:8080,就会看到Hello World. 对于http.ListenA ...

  9. JavaScript事件详解-jQuery的事件实现(三)

    正文 本文所涉及到的jQuery版本是3.1.1,可以在压缩包中找到event模块.该篇算是阅读笔记,jQuery代码太长.... Dean Edward的addEvent.js 相对于zepto的e ...

随机推荐

  1. Dom4j的一个小例子,用于解析xml文件的元素获取方式(转)

    import java.io.File; import java.io.IOException; import javax.xml.parsers.ParserConfigurationExcepti ...

  2. 在JS中简单实现Formatter函数

    JS原生并没有提供方便使用的Formatter函数,用字符拼接的方式看起来混乱难读,而且使用起来很不方便.个人感觉C#里提供的语法比较好用,如: String.Format("Welcome ...

  3. [51单片机] nRF24L01 无线模块 串口法命令 通过无线控制另一个的灯

    >_<!概述: 这是在上一个的基础上通过按键发送4种不同命令来控制接收端的LED灯亮的改进版(上一个:http://www.cnblogs.com/zjutlitao/p/3840013. ...

  4. PostgreSQL递归查询

    原料 --创建组织架构表 create table "Org"( "OrgId" ) primary key, "ParentId" ), ...

  5. 对象初始化的完整过程(C#)

    1.静态构造函数 在引入本文的主题之前,我们先来铺垫一下吧,看看静态构造函数的概念及用途. C#中允许创建无参数构造函数,该函数仅执行一次.它一般被用来初始化静态字段.CLR不能保证在某个特定时刻执行 ...

  6. ASP.NET 页面执行顺序详解

    今天整理了一下ASP执行过程,从.net页码的执行周期开始做一个详细的了解.我重写了页面的绝大多数方法.然后加载执行.所得的顺序如下. 方法是每个重写的事件中都输出一个字符,按字符打印出来的先后判断事 ...

  7. ASP.NET Core使用NLog记录日志

    1.根目录新建nlog.config配置文件 <?xml version="1.0"?> <nlog xmlns="http://www.nlog-pr ...

  8. GO学习笔记 - 用":="实现短声明变量!

    对于Delphi程序员,":="这个符号再熟悉不过了,表示对变量赋值!在GO语言中,同样表示赋值,但是和Delphi有些不同,不同就在于不需要var变量名称了!GO语言中的&quo ...

  9. java学习笔记—使用HttpSession实现QQ的访问记录(31)

    1. 编写QQ空间数据类(QQS.java) public class QQS { private static LinkedHashMap<Integer, String> qqs = ...

  10. “全栈2019”Java多线程第十章:Thread.State线程状态详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...