Emacs高手修炼手册

视频讲解版本

Emacs初体验

0 教程简介

  • 带领认识Emacs这款传说中的编辑器
  • 每节课五分钟左右,没有废话,干货
  • 每节课一个技巧或知识点,轻松学习
  • 对笔者也是个学习的过程,共同进步
  • 新手视角,向老鸟学习,向高手迈进

1 Emacs简介

  • 讲解的版本是GNU Emacs
  • GNU Emacs于1984年,Richard Stallman发起并维护
  • GNU顶级项目
  • 竞争对手VIM,VSCode等
  • 跨平台,内核解释器C写成,通过ELisp语言进行扩展

2 Emacs安装

2.1 Windows

建议使用Scoop或Chocolatey等包管理器安装。

scoop bucket add extras scoop install emacs
choco install emacs

或者也可以直接在Emacs官网进行下载解压使用。

https://www.gnu.org/software/emacs/

2.2 Linux

建议使用分发版自带的包管理器进行安装,如:

# Ubuntu 
sudo apt install emacs 
# Fedora 
sudo dnf install emacs 
# Arch 
sudo pacman -S emacs

或者也可以通过官网下载源代码编译安装。

https://www.gnu.org/software/emacs/

2.3 macOS

建议使用Homebrew包管理器进行安装。

brew cask install emacs

或者也可以在官网通过下载后拖动到Applications目录中即可使用。

https://www.gnu.org/software/emacs/

3 Emacs版本

  • 最新版本(2020年3月):26.3
  • 即将发布版本:27
  • 开发中不稳定版本:28

教程以26.3为演示版本,同时兼容27版本。安装前可查看包管理器安装的是否是对应版本。

# Windows 
scoop info emacs
 # Linux 
apt info emacs 
dnf info emacs 
# macOS 
brew cask info emacs

启动Emacs,然后查看版本。
M-x emacs-version

或者在命令行中查看版本。
emacs --version

4 Emacs初识

clipboard.png

绿色:菜单栏
红色:工具栏
黄色:编辑区域
蓝色:状态栏
紫色:交互区域(输出信息,M-x操作等)

5 基本操作速记

别太多,先学习一点点光标的移动。

  • C,代表CTRL键
  • M,代表Meta键,往往是ALT(部分键盘是ESC)

光标移动,形成肌肉记忆之前可通过速记法来记忆:

  • C-n,下一行(速记:Nextline)
  • C-p,前一行(速记:Previous line)
  • C-f,向前移动一个字符(速记:Forward)
  • C-b,向后退一个字符(速记:Backforward)
  • C-k,从光标位置到末尾删掉(速记:Kill)
  • C-a,回到行首(速记:a是字母表的开始)
  • C-e,去往行尾(速记:End of line)
  • M-<,回到编辑区域最开始位置(速记:<)
  • M->,去往编辑区域最后的位置(速记:>)
  • C-v,向下翻一屏
  • M-v,向上翻一屏

6 自带文档

C-h t ;速记 Help Tutorial

查看快捷键的含义:

C-h k ;速记 Help Keybind

查看函数的定义以及快捷键绑定:

C-h f ;速记 Help Function

7 对外观做点改变

图形化配置
M-x customize

  • 菜单栏
    menu-bar-mode

  • 工具栏
    tool-bar-mode

  • 滚动条
    scroll-bar-mode

优势:不用写ELisp代码

劣势:

  • 搜索
  • 生成的配置代码只能是单一文件配置,显得凌乱
  • 扩展的配置不容易进行
  • 涉及到复杂逻辑的(如条件判断)不方便配置

8 体验下用配置文件配置

创建文件 ~/.emacs 并写入一下内容:

(menu-bar-mode -1) 
(tool-bar-mode -1)
(scroll-bar-mode -1)

重启Emacs看看实际的效果。

Emacs配置初体验

9 认识配置文件

Emacs配置文件的位置,会按照一下顺序去查找:

  • ~/.emacs
  • ~/.emacs.d
  • ~/.config/emacs/init.el

第一个是一个单一文件配置;第二个更符合工程化;第三个仅适用于≥27的版本。教程会从第一个入手,逐渐变为第二种的模式。

10 为什么不用大牛配置

  • 大牛配置
    • Spacemacs
    • Centaur Emacs
    • Doom Emacs
    • Steve Purcell
    • redguardtoo(陈斌,一年成为emacs高手作者)
    • ……
  • 当然建议用大牛配置

    • 兼容性好
    • 代码质量高
    • 功能齐全
    • 开箱即用
  • 为什么不用

    • 体验配置Emacs的乐趣
    • 对Emacs工作多一些了解
    • 体验大牛成长的过程
    • 追求轻量,追求速度,追求新
  • 我的配置
    https://github.com/cabins/.emacs.d

11 第一行emacs配置代码

已经体验了关掉菜单栏、工具栏、滚动条等。现在尝试关掉启动界面:

(setq inhibit-startup-screen t)

重启Emacs看看?

12 软件源

  • 什么是软件源
  • 为什么要用软件源
  • 为什么要改成国内的软件源
(setq package-archives '(
    ("melpa" . "http://mirrors.tuna.tsinghua.edu.cn/elpa/melpa/") 
    ("gnu" . "http://mirrors.tuna.tsinghua.edu.cn/elpa/gnu/")
    ("org" . "http://mirrors.tuna.tsinghua.edu.cn/elpa/org/")))

13 安装第一个emacs扩展

安装扩展前,需要先刷新软件源的索引。

(setq package-check-signature nil) ;个别时候会出现签名校验失败
 (require 'package) ;; 初始化包管理器
 (unless (bound-and-true-p package--initialized) 
    (package-initialize)) ;; 刷新软件源索引
 (unless package-archive-contents
     (package-refresh-contents))

(unless (package-installed-p 'use-package) 
    (package-refresh-contents) 
    (package-install 'use-package))

14 使用use-package管理扩展

  • 什么是use-package

    • 简单理解,是一个宏
    • 用更简单统一的方式去管理插件
  • 怎么用

    • 基本格式,并举个例子
;; 最简洁的格式
 (use-package restart-emacs)

;; 常用的格式
 (use-package smooth-scrolling 
    :ensure t ;是否一定要确保已安装
    :defer nil ;是否要延迟加载 
    :init (setq smooth-scrolling-margin 2) ;初始化参数 
    :config (smooth-scrolling-mode t) ;基本配置参数 
    :bind ;快捷键的绑定 
    :hook) ;hook模式的绑定

建议添加的配置(部分来自use-package官方建议):

(eval-and-compile 
    (setq use-package-always-ensure t) ;不用每个包都手动添加:ensure t关键字 
    (setq use-package-always-defer t) ;默认都是延迟加载,不用每个包都手动添加:defer t 
    (setq use-package-always-demand nil) 
    (setq use-package-expand-minimally t) 
    (setq use-package-verbose t))

15 更换个主题

教程演示采用的主题是 gruvbox-theme

(use-package gruvbox-theme 
    :init (load-theme 'gruvbox-dark-soft t))

顺便配置一个好看一点的Mode Line。

(use-package smart-mode-line 
    :init 
    (setq sml/no-confirm-load-theme t) 
    (setq sml/theme 'respectful) 
    (sml/setup))

16 工程化管理配置(1)功能拆分

到此为止,在~/.emacs 中配置了很多的内容,开始显得很乱:

  • 初始化Elpa
  • 安装扩展
  • 添加主题

是时候分类处理了。就像一个开发项目那样。尝试着把配置按照功能不同拆分成不同的文件。如:

  • lisp/init-elpa.el用于存放Elpa和Package初始化
  • lisp/init-package.el用于存放安装的扩展
  • lisp/init-ui.el用于存放界面主题相关配置

如何在初始化文件中把它们联系在一起(调用)?

17 工程化管理配置(2)文件加载与调用

先设置加载的目标目录到 load-path 中。

(add-to-list 'load-path 
    (expand-file-name (concat user-emacs-directory "lisp")))

各个文件通过 provide 暴露对外调用的名称。如:

(provide 'init-ui)

然后在 init.el 文件中通过 require 调用:

require 'init-ui

18 关于自定义的配置

建议写入一个单独的文件,对外开源的时候,该文件不被提交到Git中。如,custom.el

(setq custom-file 
    (expand-file-name "custom.el" user-emacs-directory))

这样就会把通过图形界面做的一些配置,写入到这里了。

19 操作系统判断

后面的部分配置会因操作系统不同而产生不同配置代码,所以有必要对操作系统进行判断。

(defconst *is-mac* (eq system-type 'darwin)) 
(defconst *is-linux* (eq system-type 'gnu/linux)) 
(defconst *is-windows* (or (eq system-type 'ms-dos) (eq system-type 'windows-nt)))

20 macOS平台将Command键映射为Meta

对于Mac键盘来说,Command键距离x的位置与Windows键盘更相近,所以习惯上可以修改这个按键映射为Meta,如果你不需要这个设置,可以不用管。

(when *is-mac* 
    (setq mac-command-modifier 'meta)
    (setq mac-option-modifier 'none))

21 通过修改字体解决Windows上Emacs的卡顿

(use-package emacs 
  :if (display-graphic-p) 
  :config 
  ;; Font settings 
  (if *is-windows* 
    (progn 
      (set-face-attribute 'default nil :font "Microsoft Yahei Mono 9") 
      (dolist (charset '(kana han symbol cjk-misc bopomofo)) 
        (set-fontset-font (frame-parameter nil 'font) charset (font-spec :family "Microsoft Yahei Mono" :size 12)))) 
  (set-face-attribute 'default nil :font "Source Code Pro for Powerline 11")))

22 建议增加的一点启动配置

设置系统的编码,避免各处的乱码。

(prefer-coding-system 'utf-8)
(set-default-coding-systems 'utf-8) 
(set-terminal-coding-system 'utf-8) 
(set-keyboard-coding-system 'utf-8) 
(setq default-buffer-file-coding-system 'utf-8)

设置垃圾回收阈值,加速启动速度。

(setq gc-cons-threshold most-positive-fixnum)

23 测试启动耗时

通过 benchmark-init 包进行启动耗时的测量。

(use-package benchmark-init 
  :init (benchmark-init/activate) 
  :hook (after-init . benchmark-init/deactivate))

M-x benchmark-init/show-durations-tree 或者 M-x benchmark-init/show-durations-tabulated

24 中断命令输入

C-g在输入任何命令的时候,可随时通过该快捷键放弃命令的继续键入。

25 用y/n来代替yes/no

(defalias 'yes-or-no-p 'y-or-n-p)

为了项目管理的统一化,可以用如下的use-package写法:

(use-package emacs :config (defalias 'yes-or-no-p 'y-or-n-p))

26 显示行号

26.1 行号的类型

从Emacs 26开始,自带了行号显示功能,可以通过设置display-line-numbers-type变量的值,来改变行号的类型。可取值有:

  • relative,相对行号
  • visual,可视化行号
  • t

26.2 行号配置

(use-package emacs 
    :config 
    (setq display-line-numbers-type 'relative) 
    (global-display-line-numbers-mode t))

注:
有些时候我发现在Windows上开启了行号会让屏幕滚动的时候闪烁,此时,我们可以让其在Windows上不开启行号。

(use-package emacs 
    :unless *is-windows* 
    :config 
    (setq display-line-numbers-type 'relative) 
    (global-display-line-numbers-mode t))

Emacs文本编辑

27 文本编辑之选中/复制/剪切/粘贴

选中

  • 方法一,使用鼠标拖动选中
  • 方法二,标记开始位置,然后通过移动光标到结束位置
    • 默认使用C-SPC或C-@进行开始位置的标记
    • 对于很多场景下,C-SPC快捷键被占用(输入法,Spotlight等)

复制
M-w

剪切
C-w

粘贴
C-y ;; 速记 Yank

28 文本编辑之删除

场景一:从光标位置到行尾的删除
C-k

场景二:删除当前行

默认没有这样的快捷键,可以安装一个扩展来实现,crux,该扩展提供了包含该场景在内的一系列优化快捷命令。

(use-package crux 
    :bind ("C-c k" . crux-smart-kill-line))

后面补充一节课来介绍这个扩展。

场景三:一次性删除到前/后面的第一个非空字符(删除连续的空白)

通过hungry-delete扩展实现,该功能。

(use-package hungry-delete 
    :bind (("C-c DEL" . hungry-delete-backward)
           ("C-c d" . hungry-delete-forward)))

29 文本编辑之行/区域上下移动

在VSCode或Jetbrains家族的编辑器中,上下移动行/块是非常容易的,但原生的Emacs没有提供这样的功能。可通过插件drag-stuff来辅助实现。

(use-package drag-stuff 
  :bind (("<M-up>". drag-stuff-up) 
         ("<M-down>" . drag-stuff-down)))

30 文本编辑之原生搜索/替换

Emacs原生提供了强大的搜索和替换功能。

;; 当前位置到文档结尾的搜索 
C-s
;; 当前位置到文档开始的搜索 
C-r 
;; 替换 
M-%

替换时按下y确认替换,n跳过本处的替换,!全部替换。

31 文本编辑之强化搜索

著名的ivy-counsel-swiper三剑客,就是为了强化搜索(同时优化了一系列Minibuffer的操作)。

(use-package ivy 
  :defer 1 
  :demand 
  :hook (after-init . ivy-mode) 
  :config 
  (ivy-mode 1) 
  (setq ivy-use-virtual-buffers t 
        ivy-initial-inputs-alist nil 
        ivy-count-format "%d/%d " 
        enable-recursive-minibuffers t 
        ivy-re-builders-alist '((t . ivy--regex-ignore-order))) 
  (ivy-posframe-mode 1))) 

(use-package counsel 
  :after (ivy) 
  :bind (("M-x" . counsel-M-x) 
         ("C-x C-f" . counsel-find-file) 
         ("C-c f" . counsel-recentf)
         ("C-c g" . counsel-git))) 

(use-package swiper 
  :after ivy 
  :bind (("C-s" . swiper) 
         ("C-r" . swiper-isearch-backward)) 
  :config (setq swiper-action-recenter t 
                swiper-include-line-number-in-search t))

32 文本编辑之自动补全

Emacs界补全两大垄断插件,Auto-CompleteCompany(Complete Anything)。时代和特性原因,Company呈现压倒性优势。

(use-package company 
  :config 
  (setq company-dabbrev-code-everywhere t 
        company-dabbrev-code-modes t 
        company-dabbrev-code-other-buffers 'all 
        company-dabbrev-downcase nil 
        company-dabbrev-ignore-case t 
        company-dabbrev-other-buffers 'all 
        company-require-match nil 
        company-minimum-prefix-length 2 
        company-show-numbers t 
        company-tooltip-limit 20 
        company-idle-delay 0 
        company-echo-delay 0 
        company-tooltip-offset-display 'scrollbar 
        company-begin-commands '(self-insert-command)) 
  (push '(company-semantic :with company-yasnippet) company-backends) 
  :hook ((after-init . global-company-mode)))

但,注意下认识到Company的不足:文档。其文档都是写在源码中的。

33 文本编辑之语法检查

默认Emacs自带了Flymake,但Flycheck是一个更优的方案。

建议全局启用:

(use-package flycheck 
  :hook (after-init . global-flycheck-mode))

如果你只想在编程语言的模式下启用:

(use-package flycheck 
  :hook (prog-mode . flycheck-mode))

34 文本编辑之行排序

M-x sort-lines

35 文本编辑之顺序更换

光标前后两个字符互换
C-t

光标前后两个单词互换
M-t

36 文本编辑之字数统计

整个Buffer统计

M-= 
;; 或者 
M-x count-words-region

选中区域统计

;; 先选中 
M-x count-words

Emacs常用插件

37 crux优化Emacs常用操作

  • 优化版的回到行首
  • 快速打开Emacs配置文件
  • 快速连接两行等
(use-package crux 
  :bind (("C-a" . crux-move-beginning-of-line) 
         ("C-c ^" . crux-top-join-line)
         ("C-x ," . crux-find-user-init-file)
         ("C-c k" . crux-smart-kill-line)))

38 如何记忆越来越长的快捷键

答案:不用记忆。

如果你不喜欢用鼠标在菜单栏点击的化,你其实可以用Which-Key这款插件来辅助你“记忆”。

(use-package which-key 
  :defer nil 
  :config (which-key-mode))

当你按下一个按键时,它会提示你下一个按键的含义。并等待你去按下。

Emacs窗口管理

39 窗口管理之Buffer管理

Buffer切换

C-x b

杀死当前Buffer

C-x k

批量管理Buffer

C-x C-b ;; 进入Buffer列表 
d ;; 标记删除 
u ;; 取消当前行标记 
U ;; 取消全部标记 
x ;; 执行操作 
? ;; 查看按键帮助

40 窗口管理之MiniBuffer交互优化

显示在左下角的MiniBuffer移动视线范围大,移动到中央位置,更合适一些。

(use-package ivy-posframe
  :init
  (setq ivy-posframe-display-functions-alist 
    '((swiper . ivy-posframe-display-at-frame-center)
      (complete-symbol . ivy-posframe-display-at-point)
      (counsel-M-x . ivy-posframe-display-at-frame-center)
      (counsel-find-file . ivy-posframe-display-at-frame-center)
      (ivy-switch-buffer . ivy-posframe-display-at-frame-center)
      (t . ivy-posframe-display-at-frame-center)))

41 窗口管理之分屏

开启横向分屏
C-x 3

开启纵向分屏
C-x 2

只保留当前分屏
C-x 1

关闭当前分屏
C-x 0

42 窗口管理之分屏宽度调整

增加高度
C-x ^

增加/减少宽度
C-x {
C-x }

43 窗口管理之快速切换分屏

默认可通过以下快捷键进行窗口的循环跳转:
C-x o

当分屏很多的时候效率非常低。通过ace-window可快速进行窗口间的跳转。类似的插件较多,但该款为最佳方案。

(use-package ace-window 
    :bind (("M-o" . 'ace-window)))
image.png

44 窗口管理之Tab标签页管理

C-x t 2 ;; 新建Tab 
      1 ;; 关闭其它Tab 
      0 ;; 关闭当前Tab 
      b ;; 在新Tab中打开Buffer

未完待续……

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容