LSP 是当前使用最广泛的一套协议,用于给文本编辑器提供类似 IDE 的功能,比如:自动补全、定义跳转等。对于 Emacs 来说,主要有两个实现:
笔者本人在使用 lsp-mode 多年后转到了 eglot,主要觉得 lsp 太占用内存,有很多华而不实的功能,导致使用时能明显感到卡顿。
下面是笔者在使用 lsp-mode 几天后,执行 memory-report
后的数据(完整版):
Largest Variables
1.8 GiB lsp-clients
1.8 GiB lsp--session
1.8 GiB lsp--last-active-workspaces
4.6 MiB package-archive-contents
4.5 MiB elfeed-db
Largest Buffers
3.6 GiB database.rs
3.6 GiB source.rs
3.6 GiB history-backup/main.rs
1.8 GiB datafusion-5.0.0/src/logical_plan/expr.rs
1.8 GiB prometheus.rs
1.8 GiB libsqlite3-sys-0.23.2/src/error.rs
可以看到,占内存最高的变量都与 lsp-mode 有关,而且在用 lsp-mode 进行 Rust 开发时,能明显感到卡顿,根本不敢用 rust-analyzer 来进行补全,之前笔者都是用 tabnine 来进行 Rust 代码的补全,只用 lsp 来进行『查找定义』。
在替换成 eglot 后,内存使用就没有这么夸张了,用 rust-analyzer 进行补全时,之前的卡顿感没有了,和在 VSCode 中的体验无异。而且 eglot 的依赖很少,会尽量复用 Emacs 内置的模块,比如采用 flymake,而不是 flycheck,也有相关 issue 讨论如何用在 eglot 中使用 flycheck:
笔者使用 eglot 的配置如下,主要进行了下面几点的改进:
- eldoc 高度限制为 3 行,太大了影响阅读代码
- 修改高亮『当前变量』的字体,默认的不是很明显
- 修改
eldoc-documentation-functions
,支持直接在 minibuffer 中显示 flymake 错误信息 - 修改 rust-analyzer 启动参数,开启所有 features,这样可以分析那些只在特定 feature 下开启的文件
- 增加 Rust 宏展开的命令,lsp-mode 默认支持,这里给出了 eglot 的实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
| (use-package eglot
:defer t
:commands (eglot-ensure my/rust-expand-macro)
:hook (eglot-managed-mode . my/eglot-hook)
:config
(progn
(defun my/eglot-hook ()
;; Show flymake diagnostics first.
;; https://github.com/joaotavora/eglot/discussions/898#discussioncomment-2609402
(setq eldoc-documentation-functions
(cons #'flymake-eldoc-function
(remove #'flymake-eldoc-function eldoc-documentation-functions))))
(setq eldoc-echo-area-use-multiline-p 3
eldoc-echo-area-display-truncation-message nil)
(set-face-attribute 'eglot-highlight-symbol-face nil
:background "#b3d7ff")
;; https://github.com/joaotavora/eglot/pull/901
;; 2022-05-28 后支持
(add-to-list 'eglot-server-programs `(rust-mode . ("rust-analyzer"
:initializationOptions (:cargo (:features "all")))))
(defun my/rust-expand-macro ()
"Expand macro at point, same as `lsp-rust-analyzer-expand-macro'.
https://rust-analyzer.github.io/manual.html#expand-macro-recursively"
(interactive)
(jsonrpc-async-request
(eglot--current-server-or-lose)
:rust-analyzer/expandMacro (eglot--TextDocumentPositionParams)
:error-fn (lambda (msg) (error "Macro expand failed, msg:%s." msg))
:success-fn
(lambda (expanded-macro)
(cl-destructuring-bind (name format expansion result) expanded-macro
(let* ((pr (eglot--current-project))
(buf (get-buffer-create (format "*rust macro expansion %s*" (project-root pr)))))
(with-current-buffer buf
(let ((inhibit-read-only t))
(erase-buffer)
(insert result)
(rust-mode)))
(switch-to-buffer-other-window buf))))))
))
|
最后,读者可以根据自身需求,通过添加 hook 的方式来自动打开 eglot:
1
| (add-hook 'rust-mode-hook 'eglot-ensure)
|