VI的竞争对手Emacs快速入门

发表于:2007-07-04来源:作者:点击数: 标签:
Emacs 启动: 直接打emacs, 如果有X-windows 就会开视窗. 如果不想用X 的版本, 就用 emacs -nw (No windows )起动. 符号说明 C-X 表示按住CTRL键, 然后按X, 再把CTRL, X一起放开. M-X META META 在没有META键的电脑 上, M-X 等于先按 ESC键, 接著按 X键. Sun

  Emacs 启动:
  直接打emacs, 如果有X-windows就会开视窗. 如果不想用X 的版本, 就用 emacs -nw   (No windows)起动.
  
  符号说明
  C-X 表示按住CTRL键, 然后按X, 再把CTRL, X一起放开.
  M-X     META         META
  
  在没有META键的电脑上, M-X 等于先按 ESC键, 接著按 X键.
  Sun上面META键就是菱形的那个键.
  有些系统META键就是ALT键.(或者某一边的ALT键)
  
  C-X或 M-X的X没有大小写分别.
  
  Emacs按键命令基本上是一串C-<chr>和M-<chr>组成的.
  超过两个以上的按键命令, Emacs会在萤幕最下面一行显示你按过什么.
  这一行叫作mini buffer
  
  结束Emacs按 C-x C-c
  
  取消执行 C-g
  有些Emacs命令会跑很久, 可以用C-g中断之. 按错键也可以按C-g取消.
  
  上下移动  C-p 向上 (previous line) C-n 向下(next line)
  左右移动  C-f 向右 (forward)    C-b 向左 (backward)
  
  其实Emacs内部没有行的概念, 把一篇文章放在一个大buffer
  里面, 所以C-f (forward)就是向档尾移动, C-b (backward)
  是移回去的意思, 一次一个字.
  
  翻页 下一页 C-v (view next screen)
  上一页 M-v
  
  翻页时,上一页末尾会留一点在萤幕最上面,以维持连续性.
  
  Emacs在游标接近萤幕最下方时会自动跳半页, 把档案往前挪一点, 方便阅读.
  
  重画萤幕 C-L
  
  Emacs里面游标的专有名词叫point. point == 游标目前的 点 游标一次跳一个字(word) M-f 往后跳 M-b
  
  注意 C-f 与 M-f, C-b 与 M-b的对称性.
  
  移到行头 C-a 行尾 C-e
  移到句首 M-a 到句尾 M-e
  (M-a 到上一个句点后面,一个句子的起头.
  M-e 到句点后面)
  
  移到档头 M-< 档尾 M->
  
  删除游标目前指的/后面的字 C-d
  前面的字 DEL (Delete键)
  DEL的正名叫Rubout (Rub out)
  
  M-DEL 往回删一个字(word)
  M-d  往前删 (游标后面)
  C-k  删至行尾 (kill)
  M-k  删到一句子结尾(删到句点) (kill)
  
  注意Backspace = C-h 在Emacs下是help的意思
  后面有(kill)的, 表示此删除的动作是kill, 不太等于delete.
  emacs会把kill掉的东西放到kill ring去, 算是一种暂存的地方, 以后可以叫出来.见 yank说明.
  
  Undo: C-x u
  C-_ 等于 C-x u  有些DEC终端机, C-/就是C-_
  有时等于C-Shift- -
  重复执行
  
  举例, 向右移 8个字, C-u 8 C-f
  C-u 在Emacs里是蛮特别的,用来设定一些引数(argument/repeat count)
  给其后的命令.
  
  C-u 2 0 C-n 向下移 20行
  
  有一个特别的例外, C-u 3 C-v 不是翻三页, 而是整个萤幕向上移三行.
  据说这比较有意义.
  
  C-u 1 0 C-x u  UNDO 10次
  
  给C-L一个引数会怎么样:
  C-u 0 C-l 会重画萤幕,并且把目前的行移到萤幕第一行.
  
  另外, C-u 100  等于 M-100
  C-u 数字 等于 M-数字
  
  X windows 下,
  C-left C-right 一次移一个字(word).
  C-up C-down 移动一段 (paragraphs/C语言的话是block)
  Home = C-a
  End = C-e
  C-Home = M-<
  C-end = M->
  PgUp PgDn = M-v C-v
  设定重覆次数更加简单,
  比如要向右移10个字 C-1 C-0 right-arrow
  就是按住CTRL, 然后打10就对了, 比 C-u 1 0 简单.
  
  Mouse中键用来选取有hi-light的地方.
  右键是menu-button
  
  如果不小心按两次ESC, 等于 M-ESC, 会有一个讯息跑出来说你按到一个被disable的命令. 这是高级指令, 作者认为 初学者用不道,所以会问你要不要启动它, 一般回答no.
  
  如果某一行太长, 萤幕显示不下, Emacs会在萤幕最右边打个$,
  表示此行未完,右边还有.
  
  把一行拆成两行: 在想拆处按Enter即可.
  合并两行为一行: 在行尾按C-d (或行首按DEL)
  
  Yank: 吐出被删掉的(killed)东西.
  
  只要用kill (C-k, M-k等) 删除, 超过一个字的资料, emacs就会把它存起来, 然后C-y 可以把它叫出来. 功能跟Cut & Paste一样. Kill 和delete不一样, 只有被 kill掉的东西才能用yank吐回来.
  
  游标在同一地方不动, 连续kill掉的资料会被当成一次kill掉的, yank时会一起回来.
  
  被Kill掉的资料是放在称作 kill ring的资料结构上面, ring就是个圆圈, 被kill掉的东西会依序摆在圆圈上. yank 会放回最近一次kill掉的资料. 如果不是你想要的话, 用M-y 可以换. (M-y就是告诉emacs, 不对不对, 我不是要这一个,换前一个给我).
  
  M-y 要紧接在C-y之后.
  
  拷贝文字的方法== 连续 C-k 几次, 把要拷贝的行全部删掉, 然后按 C-y 弄回来. 再到想复制的地方按一次C-y, 就成了.
  
  把要拷贝的资料kill掉在yank回来好像很笨. 是有比较文明的 方法, 那就是M-w, 不过较麻烦.
  
  首先,要先设标记. Mark 用 C-SPC 或 C-@ 设. 然候把游标移到另一端, 按 M-w 就可以把 mark 到 point间的字存到kill ring上. point 就是游标的意思.
  
  Emacs不会把Mark起来的地方用highlight表示, 除非在X下. 在X下, 可以用M-w 来拷贝用滑鼠反白的文字.
  
  kill & yank 就是 cut & paste的意思.
  
  以上大部份指令对Bash的命令列编辑也有效
  
  档案操作
  读档: Emacs术语叫 finding a file.
  C-x C-f 然后在mini-buffer输入档名. 输入档名时, SPC键有auto-complete的功能,或者会秀出到目前为止档名前几个字和输入一样的. (TAB键也有类似功能)
  
  C-x C-f 叫 find-file
  
  C-x C-s 存档 (save current file, save current buffer)
  C-x s  存所有的档
  
  C-x i  插入档案 把另外的档案的内容读入目前编辑区内
  
  视窗
  
  Emacs把档案读进来,存在buffer中.
  我们透过window来看/编辑buffer.
  
  两个视窗会把萤幕切成两部份, 他们可以同时显示相同的, 或不同的档案.
  
  对初学者而言, 最需要的是记住怎样让不想要的视窗消失:
  
  C-x 0 关掉目前的视窗
  C-x 1 会让目前的视窗占满整个萤幕 (One Window), 取消/关掉其他的视窗.
  
  Emacs里面有许多功能都会开一个小视窗来和使用者沟通, 显示讯息.
  有时候不会自动消失很讨厌, C-x 1 就很有用.
  
  另一个功能是如何跳到另一个视窗.
  C-x o (other-window)
  
  C-x 2 把目前的视窗切成两个 (水平分割)
  C-x 3           (垂直分割)
  C-x 4 是一串与视窗有关的指令.
  C-x 5 则是扩展到X的视窗, 称为frame.
  C-x 5 2 就是再开另一个X视窗 (frame).
  
  多档编辑
  C-x C-b 看目前有那些buffer (buffer就是emacs放开起的档案的地方).
  C-x b 然后在minibuffer输入buffer的名字,可以切换编辑buffer. TAB键也有作用. 有些内部的buffer (就是没有档案的buffer), 是用*开头和结束, 这个也要打, 如*scratch*
  
  最后提醒:
  C-x 1 可以把多余的视窗关掉.
  
  Emacs扩充指令
  
  前面介绍的emacs按键大部份都是C-<chr> 或者 M-<chr>的形式.
  这是最简单的按法, 由一对按键构成一个指令.
  
  Emacs的按键可以超过2个以上. 如 C-x 1 或 C-x C-b.
  一般超过一个按键组合的命令都是用C-x 开头.
  
  另外你也可以直接下命令. 按 M-x 之后就可以打一个Emacs命令来执行. 一般这些命令名字都很长, 不过都不常用. 等一下 我们会介绍一些. 还有介绍怎么把这些命令设成按键指令.
  
  C-x C-c 就是结束Emacs. 不过一般Emacs很笨重, 一旦起动就不轻易退出. 所以比较常用的是C-z
  
  C-z 把Emacs暂停, 回到命令列. 当你下次再需要编辑时,打fg %emacs或者fg就可以把Emacs唤醒.
  
  在X下, C-z会把emacs缩成icon
  
  mode line
  
  emacs编辑画面由 编辑区(buffer) 状态列 (modeline) 和对话区 (minibuffer)构成. 这里解释 modeline 显示的讯息.
  
  以下是个范例:
  
  --**-XEmacs: xemacs.qs   (Fundamental)----74%-------
  
  由后面往前解释, 74% 表示游标的位置.
  
  (Fundamental)表示编辑模式.这是最原始的模式. 编辑不同种类的文章可能希望用不同的模式, 比如说C-mode, lisp-mode, tex-mode, text-mode等等. 在不同模式下可能多一些按键出来. 举例text-mode.
  M-x text-mode
  可以切入text-mode, 这是一般人编辑文字使用的模式. 和Fundamental mode没什么差异. 不过游标移动时, Emacs对一个字的定义就有所不同, 因而M-f M-b 等移动一个字, 一个段落的指令就可能会停在标点符号的前面. 此时状态列变为... (Text)----70%---
  
  以上说的是Major mode. 另外还有minor mode, 其实就是一些额外的功能.
  比如说, M-x auto-fill-mode 则状态列显示 (Text Fill).
  auto-fill就是自动断行, 让文章每行固定有70个字.
  
  M-X fundamental-mode 可以变回来.
  
  这里要说明一下, emacs在 minibuffer下有auto-completion的功能,也就是打M-x fund 然后按 SPC, 它会自动补全 fundamental-mode,不用全打. 如果有两个以上的选择, 它会告诉你. 这个功能对find-file (C-x C-f)等等档案编辑功能也有效. 前面提过.另外, minibuffer下面M-n和 M-p 可以取回上次呼叫的命令.
  
  最后解释两个**号. 右边的*表示文章被修改过了.
  左边的* 表示这个编辑区(buffer)可以修改.
  有一些emacs的buffer是read-only buffer, 就会标成%%%表示档案是read-only.
  
  C-x C-q 可以解开read-only的锁定, 无论如何你要改这个编辑区.这是个toggle指令, 如果原来是可以修改的, C-x C-q会把它切成read-only.
  
  Search
  
  没有Search 功能的编辑器简直就是小朋友的玩具. Search是一项很重要的功能, 所以emacs也提供的很完善.
  C-s
  C-r
  M-x re-search-forward
  M-x re-search-backward
  M-x search-forward
  M-x search-backward
  
  以上这些指令是基本的search指令. C-s, C-r是increamental search,就是你打字的同时, emacs就直接帮你找. 一个是forward, 一个是backward.找到了怎么办? 按C-g可以取消搜寻, 跳回原来的位置. 按Enter就让游标停在找到的地方 -- 此时minibuffer显示:Mark saved where search started什么意思? 就是isearch帮你在原来的位置设了一个mark, 然后把point(cursor) 移到新的位置.
  想跳回去原先的地方?
  C-x C-x 就可以了.(exchange-point-and-mark)
  C-u C-SPC 可以依序跳回前几次设mark的地方.
  (C-SPC是设mark, 给它一个argument, 就是反动作) (还记不记得C-u 可以给后面的指令设一些参数. 有些指令拿这个参数来当作repeat count, 有些指令就只拿来当作on/off, true/false, set/clear而已)
  
  M-x re-search-forward可以让你用regular expression搜寻.
  M-x search-forward则没有increamental的功能.
  
  另外一个指令, 作用和grep很像:
  M-x oclearcase/" target="_blank" >ccure
  
  和search相提并论的就是replace.
  
  M-x replace 然后按 SPC, 就知道了.
  
  Emacs的设定:
  
  Emacs的设定档是 $HOME/.emacs
  你应该多少知道, emacs是用lisp写成的编辑器, .emacs档也都是要用lisp的语法设定. emacs用的lisp称为elisp, 和一般的lisp差一点点.
  
  有一个info page, emacs-lisp-intro, 深入浅出的介绍emacs lisp. 如果你还不会, 不懂programming, 强烈建议你看这份文件. 如果你会texinfo, 你可以把它很漂亮的印出来. (内容一点点而已, 两三天就看完了)
  
  如果你把.emacs搞砸了, 进emacs很奇怪, 怎么办?
  1. 用 vi 改 .emacs :>
  2. emacs -q 进 emacs
  
  Major Modes
  
  一般常见的emacs major mode有
  fundamental-mode
  text-mode
  lisp-mode  有自动对括号/重排, 直接执行lisp code功能.
  c-mode/cc-mode c-mode是比较旧的c-mode, cc-mode应该是目前新的c-mode. 有自动重排/对括号的功能. 也可以在emacs内compile, 跳到compiler error 修正错误. 执行程式时debug. (配合dbx/gdb)compile是透过Makefile进行.
  tex-mode  Tex/Latex编辑模示. 可能是打一些奇怪的标点符号比较方便.
  <programming-language>-mode 同lisp/cc-mode. 如果是interpreter的话, emacs通常都可以直接执行/debug.
  <programming-language>-mode还有tags的功能, 后述.
  
  html-mode, texinfo-mode, sgml-mode: 编写html, texi, sgml之用.
  w3-mode   WWW browser. 在x-win上不满意,但可以接受...
  
  Tags
  
  Tags 是一个显为人知的功能? 所以我想提一下. 这不是emacs发明的,而是vi 原本的特异功能. emacs只是发扬光大而已.
  
  假设你有一个目录, 里面是一个程式的原始码, 比如说, tin 的原始码,放在 ~/tin-1.3beta 下面. 你想看它们.
  
  首先, 叫emacs cd到该目录:
  M-x cd
  
  然后, 建立tag table.
  tag table 就是一张对照表, 记录哪个符号(variable/function call)对映到哪个档案的哪个地方. 有这张表, emacs可以让我们快速的在程式码内游走. 一般这张表是一个档案, 叫作TAGS (大写)
  
  M-! etags *.[ch]
  
  M-! 是执行external shell command的意思. etags就是emacs的建表程式.你只要告诉它你的source code在那里即可.
  
  vi的话是使用ctags这个程式, 它建出来的档名叫tags (小写). 因为我们介绍emacs, 所以不管它.
  
  然后, 怎么看程式? 你知道所有的C 程式都是由main()开始, 所以你想找到main()在哪个档案. 这时只要按 M-. 然后emacs会问你tag table在哪里. 因为我们已经cd到该目录, 直接按enter就好了. 然后输入main,emacs就会把你带到main(){ ... }去.
  
  如果 你看到某个程式片断呼叫一个你没看过的函式, 你可以把游标移到该函式的名字上, M-. ENTER 就搞定了.
  
  如果 emacs找错了 (比如有变数和函式同名, emacs跳到变数去),那你可以用 C-u M-. 找下一个.
  
  在编辑程式码的时候, M-SPC 很有用, 它会把游标附近的空白缩成一个.在其它地方也有效.
  
  Emacs的一些package:
  
  M-x dired (或C-x d)
  游走/编辑 目录, 就是档案总管的意思
  
  M-x man  就是man page
  M-x shell 开个command prompt, 不过不能跑vi,elm, tin...
  M-x gnus 读新闻/读信
  M-x rmail 读信
  M-x vm  view mail
  M-x mh-rmail 读信 (package mh-e)
  M-x mh-smail 送信 (package mh-e)
  
  强列建议改用emacs读news/bbs. 世界会更美好!
  
  读信的话就要看你的感觉. 这些读信程式都会把信从系统的mail folder搬到自己的目录下, 占用quota, 我不喜欢 建议elm或mutt.
  除非参加mailling list配合procmail. 不然不实用.
  用mh-e 须要装mh 这个外部程式, 不太好. 建议vm 或 gnus.
  
  如果你的资料用rcs/sccs作版本管理, emacs自动会起动version control(minor mode.), c-x c-q 变成check-in/check-out.
  
  如何取得更多的资讯:
  
  Emacs的lisp 经过多年的发展,已成为完整的self-documenting系统.
  很多东西都可以线上找到你要的资讯.
  
  前面说过,或者你已经不小心按backspace遇到了, C-h (就是backspace的ascii码) 在emacs里面是help的意思, 它可以带出一串指令.常用的有:
  
  C-h F  Emacs FAQ
  C-h t  Emacs Tutorial
  C-h n  Emacs NEWS file, 介绍最近改版的新功能
  C-h i  Info system. Info是gnu用来取代man page的系统, 基本上和文字模示的WWW差不多. 有许多重要的资讯 可以在这边找到. 如果你是新手, 建议你在x-win下 看. 不然, 按键 m (menuitem), SPC next page l (last node: node就是章节的意思) u (up node) d (directory, 索引). BS (Backspace, back a page). 如果全部只按SPC, 就跟man 一样.
  C-h k  describe key, 告诉你按这个键执行那个lisp function.
  C-h f  describe function. 告诉你function在作什么. 如果只按SPC, emacs会给你所有lisp 函数的列表, 和说明.
  C-h v  describe variable 同function.
  C-h a  apropos的意思(approximate). 给lisp function的部份 字串, emacs帮你找.
  C-h b  列出目前所有的keybinding
  C-h m  mode help. 列出目前的mode的特殊说明.
  C-c C-h 列出以C-c 开头的所有key-binding. 虽然说Emacs 可以定义按键, 可是Ctrl- 开头的所有组合大概都用光了, 只有C-c算是可以自定指令. 不过有些mode也侵犯这个空间. 目前的convention是C-c <chr> 留给user, C-c C-<chr> 留给package.
  
  有以上这些help, 你的emacs/elisp功力会随著时间成长.
  
  Elisp 简介:
  
  Emacs有三份手册.第一份是使用手册, 第二份是Elisp 手册, 第三份是Elisp 简介. 第三份的程度是入门级, 值得看. Elisp手册其实也写的很简单, 还教你lisp, 不过有点长, 适合参考.
  
  因为我lisp没有仔细学过, 所以:
  以下所言, 如有巧合, 那才是真的.
  
  Basic data type
  字串 (string) "Hello, World"
  字元 (char) ?a        ; 问号开头
  atom & list:
  (1 2 3 4) 是一个list, 由 4个 atom 组成.
  pair: 中间是句点.
  (apple . 2)
  alist (associated list)
  就是一堆 pair的集合,就像perl/tcl的associative array.
  或者说是一个资料库, 一堆 (key, value) pair.
  '((Apple   . 1)
  (Orange  . 2)
  (PineApple . 3))
  vector (?)
  emacs 19用vector 来表示按键(key strok sequence)
  [f1] [f2] [f1 a]
  
  nil 就是空的list, 或者表示 false
  t  true
  Forms
  我们写程式最好有样版让我们填空最简单了.
  Form 就是样版, 不过意义不太一样.
  Form 就是Elisp 可以接受的句型.
  lisp 解译器 预设是对list的每个元素求值(evaluate),
  除非是 special form, 有特殊的定义. 比如说
  
  (defun FUNC (ARG-LIST)
  BODY ...)
  
  就是一个special form, 用来定义函式, 所以FUNC 不会被 求值, 被当成symbol, ...
  
  (quote (LIST))
  
  这也是个special form, 叫 lisp 把 (LIST)当做symbol就好了, 不要 evaluate.
  
  quote 很常用, 所以有个缩写:
  '(LIST) 等于 (quote (LIST))
  'Asymbol 可以表示一个Atom, 名称叫Asymbol
  
  set 可以产生/定义新的变数.
  (set 'hello 1)
  ; hello = 1
  ; 注意我们用 'hello, 所以lisp不会evaluate hello的值.
  
  这家伙很常用, 也有简写.
  (setq hello 1)
  setq 就是set quote 的缩写. 这是个 special form, 不会对 第二个元素求值.
  
  Evaluation
  
  在Emacs下, C-x C-e 可以执行(evaluate, 求值)游标左边的叙述. 结果会出现在minibuffer.
  
  lisp-interaction-mode中 C-j 可以evaluate, 并且把结果append到 buffer.
  
  lisp 程式由一堆list 构成.称为expression.
  每个expression 都回传回一个值.
  有些expression有副作用, 如删掉一个字.
  (这跟C 的int delete_char() 意思一样, 它传回int, 并且删掉某个char)
  
  定义函式:
  
  (defun NAME (ARGS-LIST)
  "注解"         ; optional
  (interactive)      ; optional
  BODY)
  
  定义一个叫NAME的函式. BODY 是一堆expression.
  注解是用来给C-h f显示的.
  (interactive) 表示这个函示会和user/buffer作用.
  (interactive "B") 表示执行此函式先问user一个buffer的名字, 然后当作参数传给它. (如, 当user透过key-binding 或者 M-x 呼叫此函式时)
  (interactive "BAppend to buffer: \nr")
  问user buffer name时, 提示号 Append to buffer: 此function有两个引数,第一个是B, 就是buffer 第二个是r, region 用\n 隔开.
  (interactive "p") 用C-u 设的prefix 把它当作参数传给我.
  预设值==4. C-u C-f 向右移四个字
  
  一些lisp 函式:
  
  (list 1 2 3 4)  产生 '(1 2 3 4)
  (car '(1 2 3 4)    1
  (cdr '(1 2 3 4)    '(2 3 4)
  (cons 1 '(2 3 4))   '(1 2 3 4)
  (cons 1 2)      (1 . 2)
  (cons 0 (cons 1 (cons 2 nil)))
  等于 '(0 1 2)
  {list 是用 pair 串起来的, 用C 表示:
  pair: {Object *first, Object *second};
  *(pair[i].first) == i;
  pair[i].second == pair[i+1]; }
  
  (cons '(1 2) '(3 4)) '((1 2) 3 4)
  (setq a 1)
  (1+ a)    ; a+1
  (+ 2 a)    ; a+2
  (* 1 2 3 4)
  (current-buffer) ; 传回目前buffer的资料物件
  (switch-to-buffer (other-buffer))
  (set-buffer)
  (buffer-size)
  (setq current-pos (point))
  (point-min)
  (point-max)
  (message "Hello")  ; 在minibuffer显示Hello
  (if (test)
  (then-part)
  (else-part))
  (cond ((test1) BODY1)
  ((test2) BODY2)
  (t OTHER-WISE)
  (let ((var1 value) ; local variable
  var2       ; no value
  (var3 value)
  ...)
  BODY ...)
  (lambda (ARG-LIST) ...) 同 defun, 但是没有名字 (anonymous).
  可以存到变数去:
  (setq hello (lambda () (message "Hello,World")))
  (funcall hello)
  (goto-char (point-max))
  (defvar VAR VALUE "*注解") 如果VAR 不存在才定义. 有注解可以用C-h v 看. 注解打*号表是使用者可以直接改/ 这个变数本来就是给使用者设定用的.
  可以用 M-x edit-options 来线上设定 (emacs结束就没有了, 不过edit-options可以给你所有可修改的变数的列表,你可以 放到.emacs档内.
  
  (directory-files "./" t "\\..*")
  return a list of files under directory X
  (load "xxxx.el") 同#include <stdio.h>
  (setq load-path (cons "~/emacs" load-path)) ; load的search path.
  (autoload ...) 不像load会直接evaluate 整个档案, 而是需要时再load.
  (local-unset-key [(control c)])
  (local-set-key [(control c) a] 'forward-sexp)
  sexp 就是一个expession, n个expression如果用括号括起来就算一个.(expression的定义随语言的不同而有不同, 在C, lisp tex, html, fortran 下皆有差异)

原文转自:http://www.ltesting.net