264

From question How to replace a character for a newline in Vim?. You have to use \r when replacing text for a newline, like this

:%s/%/\r/g

But when replacing end of lines and newlines for a character, you can do it like:

:%s/\n/%/g 

What section of the manual documents these behaviors, and what's the reasoning behind them?

Community
  • 1
  • 1
Vinko Vrsalovic
  • 244,143
  • 49
  • 315
  • 361
  • 1
    What is the `%` for when searching and replacing? – qed Oct 21 '13 at 20:12
  • 8
    @qed: by default `:s` will only apply the substitution to the current line. `:%s` makes it address the whole file. [More](http://vim.wikia.com/wiki/Search_and_replace). – c24w Jan 06 '14 at 14:09
  • 3
    +c24w I think he is referring to the '%' between the slashes, the first example the '%' being in the search location, and in the replace location in the second example, but @qed to answer your question, those '%'s are matched literally. – RastaJedi Aug 24 '16 at 02:57

5 Answers5

219

From http://vim.wikia.com/wiki/Search_and_replace :

When Searching

...

\n is newline, \r is CR (carriage return = Ctrl-M = ^M)

When Replacing

...

\r is newline, \n is a null byte (0x00).

lmat - Reinstate Monica
  • 6,119
  • 4
  • 44
  • 55
157

From vim docs on patterns:

\r matches <CR>

\n matches an end-of-line - When matching in a string instead of buffer text a literal newline character is matched.

iconoclast
  • 17,897
  • 10
  • 91
  • 122
pjz
  • 38,171
  • 5
  • 45
  • 60
124

Another aspect to this is that \0, which is traditionally NULL, is taken in s//\0/ to mean "the whole matched pattern". (Which, by the way, is redundant with, and longer than, &).

  • So you can't use \0 to mean NULL, so you use \n
  • So you can't use \n to mean \n, so you use \r.
  • So you can't use \r to mean \r, but I don't know who would want to add that char on purpose.

—☈

rking
  • 1,432
  • 1
  • 9
  • 8
  • 1
    Good point! So this is a hack around \0 - backreference to 0-th nested group - whole regexp. This is a very good answer, but it makes me dissapointed, that I can't use \r in replacement... (meaning: insert #(13), for example in a "normal" file (== unix EOLs). – Tomasz Gandor Nov 06 '13 at 09:54
  • 3
    OK, the answer is in the Vim docs. Just escape \^M or \, with a normal backslash. Nice. ----- split line in two at this point (Type the as CTRL-V ) *s* \r idem -------- *s/\r* \ insert a carriage-return (CTRL-M) (Type the as CTRL-V ) *s/\* – Tomasz Gandor Nov 06 '13 at 10:03
  • 18
    "Another aspect"? You're too modest! Explaining precisely_why_ this happens makes for a superior answer to all the ones just saying _how_ to work around it. +1 – underscore_d Oct 20 '15 at 13:23
  • 1
    This is honestly *really* great insight into why `\n` has different meanings in search and replace patterns. Fantastic answer. – ZeroKnight Apr 30 '19 at 10:51
49

:help NL-used-for-Nul

Technical detail:

<Nul> characters in the file are stored as <NL> in memory. In the display they are shown as "^@". The translation is done when reading and writing files. To match a <Nul> with a search pattern you can just enter CTRL-@ or "CTRL-V 000". This is probably just what you expect. Internally the character is replaced with a <NL> in the search pattern. What is unusual is that typing CTRL-V CTRL-J also inserts a <NL>, thus also searches for a <Nul> in the file. {Vi cannot handle <Nul> characters in the file at all}


matthias krull
  • 4,262
  • 3
  • 29
  • 54
Aristotle Pagaltzis
  • 101,052
  • 21
  • 94
  • 96
11

First of all, open :h :s to see the section "4.2 Substitute" of documentation on "Change". Here's what the command accepts:

:[range]s[ubstitute]/{pattern}/{string}/[flags] [count]

Notice the description about pattern and string

For the {pattern} see |pattern|.
{string} can be a literal string, or something
special; see |sub-replace-special|.

So now you know that the search pattern and replacement patterns follow different rules. If you follow the link to |pattern|, it takes you to the section that explains the whole regexp patterns used in Vim.

Meanwhile, |sub-replace-special| takes you to the subsection of "4.2 Substitute", which contains the patterns for substitution, among which is \r for line break/split.

(The shortcut to this part of manual is :h :s%)

syockit
  • 5,605
  • 1
  • 21
  • 32