作为这个问题的后续,它试图找出如何做这样的事情,这应该很容易,这尤其阻止我更习惯于使用Emacs,而是启动我已经熟悉的编辑器。在编辑多个文件时,我经常使用这个示例。

在Ultraedit中,我用Alt+s然后p来显示一个带有选项的对话框:查找(包括跨多行使用正则表达式),替换,在文件/类型中,目录,匹配大小写,仅匹配整个单词,列出更改的文件和搜索子目录。通常我会先用鼠标点击-拖动选择我想要替换的文本。

只使用Emacs本身(在Windows XP上),不调用任何外部实用程序,如何替换所有的foo\nbar与bar\nbaz在*.c和*.h文件中的一些文件夹和它下面的所有文件夹。也许Emacs并不是最好的工具,但是如何用最少的命令就能轻松完成呢?


当前回答

我想推荐一个还没有被提及的伟大工具,即Helm。

它是许多标准Emacs操作的一个很好的替代品,包括补全、搜索等。特别是,helm-find-files允许在多个选定文件中执行查询替换(包括regexp)。

只需打开helm-find-files,用M-SPC标记相关文件,然后使用F6或F7在所选文件中运行query replace或query replace regexp。

其他回答

M-x find-name-dired RET it may take some time for all the files to appear in the list, scroll to bottom (M->) until "find finished" appears to make sure they all have loaded Press t to "toggle mark" for all files found Press Q for "Query-Replace in Files...": you will be prompted for query/substitution regexps. Proceed as with query-replace-regexp: SPACE or y to replace and move to next match, n to skip a match, etc. Type ! to replace all occurrences in current file without asking, N to skip all possible replacement for rest of the current file. (N is emacs 23+ only) To do the replacement on all files without further asking, type Y. Call “ibuffer” (C-x C-b if bound to ibuffer, or M-x ibuffer RET) to list all opened files. Type * u to mark all unsaved files, type S to save all marked files * * RET to unmark all marks, or type D to close all marked files

这个答案结合了这个答案、这个网站和我自己的笔记。使用Emacs 23+。

M-X Dired, t标记所有文件,Q查询替换所有文件中的文本。 在执行query-replace命令之前,可以使用i命令展开子目录。 我添加的关键信息是,如果你给I命令加一个前缀(control-u), 它会提示你输入arg,而-R参数会递归展开所有子dirs 进入干燥的缓冲器。所以现在您可以查询搜索整个目录中的每个文件。

我想推荐一个还没有被提及的伟大工具,即Helm。

它是许多标准Emacs操作的一个很好的替代品,包括补全、搜索等。特别是,helm-find-files允许在多个选定文件中执行查询替换(包括regexp)。

只需打开helm-find-files,用M-SPC标记相关文件,然后使用F6或F7在所选文件中运行query replace或query replace regexp。

find-name-dired是可以的,但是:

您得到的所有文件都匹配同一个regexp。 find- dred在这方面更灵活,但它也是为使用通用规则而设计的(即使这些规则可能非常复杂)。当然find也有自己复杂的语言。 如果您希望只对find(name)-dired缓冲区中收集的文件名中的一些文件进行操作,则需要标记它们,或者删除/省略不想操作的那些行。

另一种替代方法是使用dred +命令,作用于(a)标记的文件和(b)标记子目录中所有标记的文件(或所有文件,如果没有标记)…发现递归。这为您提供了对文件选择的通用性和轻松控制。这些“here-and-below”命令都在dred模式下的前缀键M-+上。

例如,M-+ Q与Q——query-replace相同,但目标文件都是在当前目录和任何标记的子目录中标记的文件,down, down, down…

是的,使用这种“这里和下面”命令的另一种方法是递归地插入所有子dirs及其子dirs,然后使用q之类的顶级命令。但是,不使用插入的子dirs通常更方便。

要做到这一点,你无论如何都需要一种快速递归插入所有子dirs的方法。在这方面,Dired+也能提供帮助。M-+ M-i递归地插入所有标记的子dirs和它们自己标记的子dirs。也就是说,它类似于M-i(在Dired+中插入标记的subdirs),但它对subdirs递归地起作用。

(所有这样的“这里和下面”干燥+命令在菜单多个>标记在这里和下面。)

您还可以对Emacs文件集执行Dired操作,Emacs文件集是一组保存的位于任何位置的文件名。如果您使用Icicles,那么您可以为文件集或其他类型的保存文件列表中的文件打开一个Dired缓冲区。

You can also bookmark any Dired buffer, including one that you create using find(-name)-dired. This gives you a quick way to return to such a set (e.g. a project set) later. And if you use Bookmark+ then bookmarking a Dired buffer records (a) its ls switches, (b) which files are marked, (c) which subdirectories are inserted, and (d) which (sub)directories are hidden. All of that is restored when you "jump" to the bookmark. Bookmark+ also lets you bookmark an entire tree of Dired buffers --- jumping to the bookmark restores all of the buffers in the tree.

所提供的答案都很棒,但是我认为我应该添加一个稍微不同的方法。

这是一个更具交互性的方法,需要wgrep, rgrep和iedit。iedit和wgrep都必须通过MELPA或Marmalade安装(使用M-x package-list-packages)

首先运行M-x rgrep查找要查找的字符串。

您将能够指定文件类型/模式和递归的文件夹。

接下来你需要运行wgrep,用C-s C-p启动它。

Wgrep会让你编辑rgrep结果,所以在字符串上设置一个区域来匹配并使用C-启动iedit-mode;(取决于你的终端,你可能需要重新绑定这个)

所有事件都可以立即编辑。C-x C-s来提交wgrep。然后是c x s !保存已更改的文件。

这种方法的主要好处是可以使用iedit-mode关闭某些匹配M-;。您还可以使用rgrep中的结果跳转到文件中,例如,如果您有一个意外的匹配。

我发现它对于在项目中进行源代码编辑和重命名符号(变量,函数名等)非常有用。

如果你还不知道/不使用iedit模式,这是一个非常方便的工具,我强烈建议你去看看。