于是有了另一种方式,拆分到文件,每种模式用一个单独的文件。在主配置文件里写上 (load 'xxx)
但是这挺麻烦的,你要新建一个文件名为 xxx,然后主配置文件里又得写 xxx,这显然是重复劳动……这是我最痛恨的……因为我一年级就上了好几次
于是我弄了一个文件夹,按顺序自动 load 里面的文件,当然不是随便 load,得符合条件才行。比如我设定的条件是扩展名为 el。
这样一来,不想 load 的文件改下扩展名就可以了……我还弄了一个函数,专门给文件添加/删除扩展名 .bak
但是时间长了也挺麻烦的,因为要手动改,而且下次才能生效…… 你还得自己 load-file
……其实一般情况下,这不是什么问题,你就是给每种模式写上几万行的配置,都写在一个文件里,对于现在的机器而言,压力也不是太大……但是咱追求的不是方便,而是折腾

好了,开始动手
[lisp]
(let (name
(dir (expand-file-name "_major-mode/" *init-dir*)))
(mapcar
(lambda(x)
(setq name
(file-name-sans-extension (file-name-nondirectory x)))
(add-hook (concat-symbol name "-mode-hook")
`(lambda()(load ,x))))
(directory-files dir t "\\.el\\'")))
[/lisp]
简单的说,假设 ~/init.emacs/_major-mode/ 目录下有 3 个文件 emacs-lisp.el nxml.el js2.el.bak ,则会:
在 emacs-lisp-mode-hook 里添加 (lambda nil (load "~/init.emacs/_major-mode/emacs-lisp.el"))
在 nxml-mode-hook 里添加 (lambda nil (load "~/init.emacs/_major-mode/nxml.el"))
用到的 concat-symbol
[lisp](defun concat-symbol (&rest lst)
(read (apply 'concat (mapcar (lambda(x)(if (symbolp x) (symbol-name x) x)) lst))))
[/lisp]
需要注意的是,这简直就是一个特大号的 hook ,所以不用在里面再写什么 add-hook 'xxx-mode-hook 了,里面所有的东西都相当于在 xxx-mode-hook 里。
如此一来,就产生了另一个问题,每打开一个 a.xxx 文件,xxx-mode-hook 就要运行一次,这比较二,实际上只要运行一次就可以了。
这个问题可以用 require 来解决,但缺点是,你要在新建一个文件为 xxx.el ,还要在里面写 provide 'xxx ,并且还要把当前目录加到 load-path,重复劳动还是比较多。
另一种方案就是 load-once
[lisp]
(defmacro load-once (&rest s)
(let* ((name (file-name-sans-extension
(file-name-nondirectory
(or load-file-name (buffer-file-name)))))
(a (concat-symbol "*load-once--" name "*")))
`(if (boundp ',a)
nil
,@s
(setq ,a t))))
[/lisp]
在 xxx.el 里面只运行一次的部分可以写在 load-once 里面,像这样
[lisp]
(load-once
(foo)
(bar)
)
[/lisp]
基本上,你把所有内容都写在 load-once 里面就对了,不管这个文件被 load 了多少次,都只运行一次。
这种方法的缺点是,需要在文件开头和结尾分别写 (load-once 和 ),而 provide 只要写在一处;
每种模式被 load 后,自动生成了一个值为 t 的变量 *load-once--xxx* 、*load-once--yyy* ……
