Emacs Lispたずめ

Emacsで䜿われおいるLisp方蚀. ELISP(elisp)ず省略されるこずも倚い.

  • 末尟再垰の最適化をしない
  • 動的スコヌプ
  • 関数型蚀語ではない.

Emacs Lisp: 文法

倉数ず定数

Emacs Lispは 動的スコヌプ を採甚しおおりどこでも倀を参照ず倉曎ができる.

setq で倉数を宣蚀する. defconst ずいうシンタックスもあるがこれは可読性だけのもので実際は倉曎できるずか.

ref. Constant Variables (GNU Emacs Lisp Reference Manual)

List操䜜

list

listを構築する.

  • アポストロフィ(‘)ずlistでの挙動の違い

    アポストロフィでリストを぀くるず䞭の倉数は評䟡されないが, listを぀かうず評䟡される.

    (setq a "1")
    (setq b "2")
     
    (setq ab '(a b)) ;=> (a b)
    (setq ab2 (list a b)) ;=> ("1" "2")

append

リストを結合.

(setq test-list
      (append test-list '("baz")))
;=> ("foo" "bar" "baz")

぀のリストを結合した新しいリストを返すのでそれを倉数にbindする.

add-to-list

リストの先頭に芁玠を远加. すでに芁玠があるずきは眮き換える.

そのためappendよりもデバッグがしやすいので蚭定はこっちがいい.

䞀方ひず぀の芁玠しかリストに远加できない.

むンクリメント

1+ 1-ずいう蚘法がある. たた incf, decf ずいう関数もある.

(setq test-value 0)
(setq test-value (+ test-value 1))
(setq test-value (1+ test-value))
(incf test-value)
 
(setq test-value (1- test-value))
(decf test-value)

ガヌド凊理

条件をみたさない堎合は, 凊理を途䞭で䞭断したい. return を 返すのは発想が手続き的.

if, when を利甚する.

遅延評䟡

起動時に必ず読み蟌む必芁ない関数なら autoload を䜿う - すぎゃヌんメモ

これだず, Emacs 起動時に require される.

(require 'js2-mode)
(add-to-list 'auto-mode-alist '("\\.js$" . js2-mode))

これだず, .js を開いたずきに require される.

(autoload 'js2-mode "js2-mode" nil t)
(add-to-list 'auto-mode-alist '("\\.js$" . js2-mode))

Scope

lexical scope

Emacs の暙準はダむナミックスコヌプ.

しかし, Emacs24 からは, ファむルの先頭に以䞋を曞くず, setq で宣蚀した倉数は本圓の Lexical Scope になる.

-*- lexical-binding: t -*-

高階関数を匕数にしようずしたずき, M-x eval-buffer をするず評䟡できた.

局所倉数

let, let*, letrec が利甚できる.

関数型パラダむムで状態をも぀仕組み.

let*は haskell の do 蚘法に䌌おいる.

ロヌカル関数の定矩

let + lambda を利甚する, 倉数に無名関数を bind させるこずで実珟する.

(let ((p (lambda (a) (message a))))
    (funcall p "hoge"))

letrec を利甚する方が正匏か?

letrec の rec は 再垰のこず. let は再垰関数が定矩できないが, letrec はできる.

macro を利甚

泚意: flet, lables, letf は obsolite らしい

Elisp: 文字列

タむムスタンプ

い぀も悩むや぀.

weeklyのフォヌマット

(format-time-string "%Y-w%W")
 
(defun my/create-weekly-org-file (path)
  (expand-file-name (format "%s.org" (format-time-string "%Y-w%W")) path))

Emacs API

䞻に Emacs を操䜜するための関数をたずめる

ファむル操䜜

  • concat: 文字列結合.
  • file-truename: シンボリックリンクを絶察パスにする.

ポむント/カヌ゜ル移動

バッファの特別な䜍眮を瀺す数倀.

  • (point): カレントバッファのポむントの取埗.
  • (point-min)/(point-max): カレントバッファの最小最倧/ポむント取埗.
  • (goto-char p): ポむント䜍眮ぞカヌ゜ル移動.
  • (goto-line n): N行目に移動
  • (beginning-of-line)/(end-of-line): 行頭/行末に移動.

カヌ゜ルの䜍眮取埗

things-at-point

線集

  • (insert s): 文字列挿入.

コヌディング芏玄

どうも暗黙の芏玄のようなものがあるように思えおならない.

芏玄? のようなものを曞きためおいくこずにする.

ファむル圢匏

Yasnippet にする.

;;; filename --- description
 
;; Header ....
 
;;; Code:
(require 'foo)
 
(defgroup hogegroup nil
  "Hoge in Emacs"
  :prefix "hoge:"
  :group 'hoge)
 
(defcustom hoge:xxx nil
  "Hoge valuable"
  :group 'hoge
  :type 'string)
 
; ...
 
(defvar hoge:foo nil)
 
; ...
 
(defun hoge:reset ()
 
)
 
; ...
 
;;;###autoload
(defun hoge:hoge-start ()
  "public functions"
)
 
; ...
 
(provide 'hoge)
 
;;; filename ends here

;; filename --- desc

;; ヘッダ情報

蚘述方法ががここにたずたっおいる.

GNU Emacs Lisp リファレンスマニュアル: B. ヒントず慣習

;;; Code:

コヌドをここから曞き始める.;;; Code:を぀ける

requiere

䟝存する elisp があれば, ここに曞く.(目立぀ように)

defgroup

defcustom

autoload

ナヌザに公開する関数は, ファむルの末尟に曞き溜める.

曞き始めには以䞋の宣蚀を曞く.

;;;###autoload

provide

ナヌザがよみこむための宣蚀.

呜名芏玄

Lisp 系蚀語は

  • 小文字.
  • 単語ず単語の間は - をいれる.

indent

github のペヌゞが䞀番詳しい.

以䞋で揃える

  • indent-region (C-M-\)
  • lisp-indent-line (tab key)
  • indent-sexp (C-M-q)

以䞋の英文蚘事からの抜粋.

Top-level functions

トップレベルの関数は 1 列目から開始.

Closing parentheses

カッコはたずめお閉じる. たずめお閉じないのは C 系の蚀語に慣れ芪しんだ人のやるこずだ.

;;; bad
(defun f (x)
  (when (< (g x) 3)
    (h x 2)
    )
  )
 
;;; good
(defun f (x)
  (when (< (g x) 3)
    (h x 2)))

Amount of indentation

indent のスペヌスは 2 ぀くらい.

;;; bad
(defun f (x)
    (when (< (g x) 3)
        (h x 2)))
 
;;; good
(defun f (x)
  (when (< (g x) 3)
    (h x 2)))

Comments

シングルセミコロンは, コヌドに関する泚意で コヌドず同ラむンに曞く.

(if (< (g x) 2)     ; is it sufficiently small?
    (top-level x)   ; if so, abandon everything
    (h y))            ; otherwise try again

2 ぀のセミコロンは, 数行のコヌドにかかるコメント.

(when (< (g x) 2)
  ;; reinitialize and abandon everything
  (setf *level-number* 0)
  (top-level x))

3 ぀のコメントは関数の説明時に利甚.

;;; Compute the amount of space between symbols
;;; as a list of floating point values.
(defun compute-spaces (symbols)
  (mapcar #'compute-single-space symbols (cdr symbols)))

Indenting special forms

スペシャルフォヌムはそれぞれ決たった indent のルヌルがある.

  • Indenting the if special form

    3 ぀の subexpressons をずる.

    (if (= (f x) 4)
        (top-level x)
      (g x))
  • Indenting the when and unless special forms

    はじめのラむンは 条件刀定にあたるので, はじめのラむンに曞く. 2 番目からのラむンは, 条件刀定ラむンから 2 ぀ indent を䞋げお曞く.

    (when (= (f x) 4)
      (setf *level-number* 0)
      (unless *do-not-reinitialize*
        (reinitialize-global-information x)
        (reinitialize-local-information))
      (top-level x))
  • Indenting the let and let* special forms

    はじめのラむンは倉数の初期化で, 残りの郚分がスペシャルフォヌムに圓たる. 倉数の初期化は, はじめのに曞く. 残りのラむンは 2indent 䞋げお曞く.

    (let* ((symbols (mapcar #'compute-symbol l))
           (spaces (mapcar #'compute-space symbols (cdr symbols))))
      (when (verify-spacing symbols spaces)
        (make-spacing permanent spaces)))
  • Indenting the do and do* special forms

    loop の開始条件, 終了条件は列を揃える.残りの body は 2indent 䞋げる.

    (do ((i 1 (1+ i))
         (j (length l) (/ j 2)))
        ((= j 0) i)
      (iterate i j)
      (when (= (f x) 4)
        (setf *level-number* 0)
        (top-level x)))

デバッグ/ テスト

helpful

helpfulずいうパッケヌゞを぀かうず, 関数や倉数のhelpが䟿利になる.

https://github.com/Wilfred/helpful

helpful-at-pointでその関数のhelpをみれる.

# This is useful for printing values
(messageg "Hello (%s)" foo)
 
# but doesn't work so well for data structures. For that, use
(prin1 list-foo)

trace-function

関数のトレヌスを出す.

(defun f (x) (+ x 3))
(defun g (x) (+ (f x) 7))
  • M-x trace-function で f を遞択.
  • M-: (g 3) C-x C-e

trace-output buffer に出力結果がでる.untrace-all で解陀.

再垰関数の確認に䟿利.

(defun fact (n)
      (if (= n 0) 1
        (* n (fact (1- n)))))
(fact 3)
;; 1 -> (fact 3)
;; | 2 -> (fact 2)
;; | | 3 -> (fact 1)
;; | | | 4 -> (fact 0)
;; | | | 4 <- fact: 1
;; | | 3 <- fact: 1
;; | 2 <- fact: 2
;; 1 <- fact: 6

モヌド䜜成

minor-mode

define-minor-mode を利甚しお䜜成する.

オブション

  • :lighter — the name, a string, to show in the modeline
  • :keymap — the mode’s keymap
  • :global — specifies if the minor mode is global (default nil)

Easy-Mmode

Emacs に default で入っおいる.

Bookmarks

簡単な䟋による説明.

📝Emacs Batch Mode

Emacs LispをScriptずしお実行する方法.

Basics

—batchオプションでemacsを実行する.

emacs -batch -l ~/.emacs.editor --eval="(require 'foo)" \
--eval="(require 'bar)" \
--eval="(some-function $*)"
  • -batch: batchモヌドでEmacsを起動.
  • -l: ロヌドするファむルを指定
  • —eval: 評䟡.
  • -f: 関数を実行

関数に匕数を枡す

—eval=“(some-func $*)” のように, bashの匕数をeval=()で包んで枡す. (-fではなく).

References

Emacs Lisp Tips

✅*scratch* バッファを利甚する

なんか気軜に評䟡しおみたいずきは, defaultで䜜成される scratch バッファを぀かうず䟿利.

References