What is the use case for Ruby's %q / %Q quoting methods?

RubyStringQuotes

Ruby Problem Overview


I've been reading through Thomas' Programming Ruby 1.9 and came upon the alternative delimited single and double-quoting methods (%q / %Q). I've known of them from other Ruby language references as well.

%q/I'm acting like a single-quoted string/

%Q|"I'm acting like a double-quoted string" --Anonymous|

I have not been working with Ruby for long, but I have never encountered this quoting method in production code.

Other than the obvious ability to avoid internally escaping quotes with backslashes, what are the common use cases for this method of quoting over regular single or double quotes? Are they typically used in single or multiline strings? If used in multiline strings, are they ever favored over HEREDOC strings? Is there a particular Ruby idiom where they're commonly found?

Ruby Solutions


Solution 1 - Ruby

They're extraordinarily useful for escaping HTML with JavaScript in it where you've already "run out" of quoting methods:

link = %q[<a href="javascript:method('call')">link</a>]

I've also found them to be very useful when working with multi-line SQL statements:

execute(%Q[
  INSERT INTO table_a (column_a)
    SELECT value
      FROM table_b
      WHERE key='value'
])

The advantage there is you don't need to pay attention to the type of quoting used within your query. It will work with either single, double, or both. They're also a lot less fuss than the HEREDOC style method.

Ruby provides other convenience methods like this such as %r which can construct regular expressions. That avoids slash-itis when trying to write one that handles stuff like http:// that would otherwise have to be escaped.

Solution 2 - Ruby

Overview

Apart from "avoid internally escaping quotes" and the examples previously provided by @tadman there are other use-cases as well:

  • auto-generating code in the same language as the generator itself (e.g., Ruby generating Ruby)
  • providing cleanly-formatted code that does not confuse the syntax-highlighting feature of your text editor
  • allow storage of codeblocks that may have to pass through multiple storage layers (such as a database interacting with a website, or a snippets management system interacting with a text editor, which interacts with a subshell, and so forth)
Details

This approach is a general-purpose and robust idiom that works well with any kind of tool that does automated code generation, including but not limited to tools that write boilerplate code in other languages, or tools that manage code snippets for an IDE or text editor.

Besides the examples already provided by @tadman, there is the general case of generating code where the code being generated is the same or substantially similar syntax as the code of the generating program.

In these cases, the solution does a lot more than help with avoiding the use of backslashes to escape quotes. Without a solution like this, there are cases where the generated code can get extremely difficult to maintain.

To see examples of this, feel free to take a look at the following references.

References

[ See e.g.,

]

Solution 3 - Ruby

I know this is an old thread, but I have used them in production when calling a system command and wanting to interpolate some values, like so:

 system(%Q(ffmpeg -y -i "#{input_filepath}" -qscale:a 2 "#{output_filepath}"))

Super handy.

Solution 4 - Ruby

Its perhaps worth noting that %q is used by bundler and jeweler by default when generating gemspecs, for the summary/description section of the gemspec. This prevents someone from using quotes in the summary or description and breaking the gemspec.

Solution 5 - Ruby

They are useful when your string contains single- or double- quotes. This doesn't happen all that often, but the technique is very useful when it does.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionMichael BerkowskiView Question on Stackoverflow
Solution 1 - RubytadmanView Answer on Stackoverflow
Solution 2 - RubydreftymacView Answer on Stackoverflow
Solution 3 - RubyRyan ReboView Answer on Stackoverflow
Solution 4 - RubyDan GarlandView Answer on Stackoverflow
Solution 5 - RubyPeterView Answer on Stackoverflow