0
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

zshでシェルスクリプト:複数行コメントアウトしたいんじゃ!

Posted at

お忙しい方のための一例

以下で一応それっぽいことできます

#!/bin/zsh

echo "1行だけコメントアウトしたいときは#を使う!"
# echo "ここはコメント!"

echo "複数行コメントアウトしたいときは : <<  の後、'任意の文字列'で囲む!" 

: << 'COMMENT'
echo "ここはコメント!"
echo "ここはコメント!"
echo "ここはコメント!"
COMMENT

echo "ここは表示されるよ!" 

でも行頭に全部#付けるのがベターっぽい。

新しいプログラミング言語学ぶときって

新しいプログラミング言語勉強するときに1番最初に何を調べますか?
チュートリアルに従うと大体"Hello,world!"の出力からはじまるのでprint文的なものを最初に目にすると思うのですが、私はそれと同時にどうしても知りたくなるものがあります。
それは、
コメントアウトの仕方です。

最近シェルスクリプトを書く機会が身近になりました。
シェルはPCがMacでデフォルトがzshだからという理由だけでzshを使っています。
「シェルスクリプト?.shってやつ?なんか最初にコメントみたいなの書いてるやつよね」
くらいの認識だったがためにちまちま勉強しつつ書いているのですが1行コメントアウトの仕方はすぐにわかった一方で、複数行コメントアウトの仕方が全然ズバッと出てこない!
というわけで、いろいろ調べていろいろ実験したので記事にしてみようと思います。

以下ダラダラ書くので、よろしければお付き合いください。

軽い気持ちでググってみた

コメントアウトの仕方ぐらいググれば出てくるやろ〜と思っていたので
zsh 複数行 コメントアウトでググってみました。
結果、出てくるものbashの記事ばっかり!
まあでも...大体同じなんじゃないん?と思ったのでググって1個目に出てきたbashのコメントアウトのやり方をzshで試してみました。

test_1_zsh.sh
#!/bin/zsh

echo "ここは表示されるよ!" 

<< COMMENT
echo "ここはコメント!"
echo "ここはコメント!"
echo "ここはコメント!"
COMMENT

echo "ここは表示されるよ!" 

以下結果

$ zsh test_1_zsh.sh
ここは表示されるよ!
echo "ここはコメント!"
echo "ここはコメント!"
echo "ここはコメント!"
ここは表示されるよ!

なんか、めっちゃ表示されたやん...

マニュアルを読んでみる

全然コメントアウトできんやんけ!!ってことで、こういうときは公式を見ます。
man zshをターミナルで実行するといろいろ情報出してくれるのですね。
zshのホームページ(https://www.zsh.org/)からたどっていって
ドキュメンテーション(https://zsh.sourceforge.io/Doc/)を参照します。

読み進めると6.7 Commentsにコメントについての記載があります。
短いので以下に全文引用

In non-interactive shells, or in interactive shells with the INTERACTIVE_COMMENTS option set, a word beginning with the third character of the histchars parameter (‘#’ by default) causes that word and all the following characters up to a newline to be ignored.

複数行のコメントアウトについて一切触れられていない、だと、、

じゃあbashはどうなん?

じゃあbashはどう書かれているんだろう?と気になって調べてみました。
man bashを実行するとマニュアル表示されて、その中にコメントについての記載がありました。

COMMENTS
In a non-interactive shell, or an interactive shell in which the
interactive_comments option to the shopt builtin is enabled (see
SHELL BUILTIN COMMANDS below), a word beginning with # causes that
word and all remaining characters on that line to be ignored. An
interactive shell without the interactive_comments option enabled
does not allow comments. The interactive_comments option is on by
default in interactive shells.

い、一緒のこと書いてあるーー!!!

シェルスクリプトには複数行コメントアウトというものが存在していない

ドキュメンテーションに何も書いていない時点でお察しなのですが、
調べていく中で、まず大前提としてシェルスクリプトには
複数行コメントアウトというものが存在していないということがわかりました。

じゃあbashの複数行コメントアウトの例として出てきていた
<<からはじまるやつはなんなんだという話になってくるのですが
ヒアドキュメントというものを使って便宜的に複数行コメントアウトを実現しているものだということです。

<< ヒアドキュメント(Here Documents)

bashはman bashの中に、zshはドキュメンテーションの7 Redirectionに記載があります。
活用方法は以下の記事に詳しいです。大変参考になります。
【参考】bashのヒアドキュメントを活用する

<<ではじめ、それに続くワード(任意)で囲まれた範囲の行に記載されているものがヒアドキュメントと呼ばれるとのことで、以下のようなイメージです。

<< WORD
ここの行から
ヒアドキュメント
WORD
# 上のWORDの行まで。WORD自身は特に読み込まれたりすることはない。

コメントアウトとして使う:bashでの挙動

さてこれがコメントアウトとして働いてくれるのかということですが、
以下のコードを実行してみましょう。
なんか記事が全体的に長くなってきた気がするのでコードは折り畳んじゃいます。

bashのサンプルコード
test_2_bash.sh
#!/bin/bash

echo "ヒアドキュメントのテスト"

num=1

echo "--------------"

echo "なにもなし"

<< here_zsh
これは文字列だよ
$num
ans_1=$((2 * 2))
echo "$ans"
here_zsh

echo "ans_1の値:$ans_1"

echo "--------------"

echo "クオーテーションあり"

<< 'here_zsh'
これは文字列だよ
$num
ans_2=$((2 * 2))
here_zsh

echo "ans_2の値:$ans_2"

echo "--------------"

echo "バックスラッシュをつける"
<< \here_zsh
これは文字列だよ 
$num 
ans_3=$((2 * 2)) 
here_zsh

echo "ans_3の値:$ans_3"

以下実行結果

$ bash test_2_bash.sh
ヒアドキュメントのテスト
--------------
なにもなし
ans_1の値:
--------------
クオーテーションあり
ans_2の値:
--------------
バックスラッシュをつける
ans_3の値:

囲んで書いた部分は特に出力されないようです。

コメントアウトとして使う:zshでの挙動

ではzshではどうでしょうか?
シバン以外はbashと全く同じものですが一応下にコードをば。

zshのサンプルコード
test_2_zsh.sh
#!/bin/zsh

echo "ヒアドキュメントのテスト"

num=1

echo "--------------"

echo "なにもなし"

<< here_zsh
これは文字列だよ
$num
ans_1=$((2 * 2))
echo "$ans"
here_zsh

echo "ans_1の値:$ans_1"

echo "--------------"

echo "クオーテーションあり"

<< 'here_zsh'
これは文字列だよ
$num
ans_2=$((2 * 2))
here_zsh

echo "ans_2の値:$ans_2"

echo "--------------"

echo "バックスラッシュをつける"
<< \here_zsh
これは文字列だよ 
$num 
ans_3=$((2 * 2)) 
here_zsh

echo "ans_3の値:$ans_3"


以下実行結果

$ zsh test_2_zsh.sh
ヒアドキュメントのテスト
--------------
なにもなし
これは文字列だよ
1
ans_1=4
echo ""
ans_1の値:
--------------
クオーテーションあり
これは文字列だよ
$num
ans_2=$((2 * 2))
ans_2の値:
--------------
バックスラッシュをつける
これは文字列だよ 
$num 
ans_3=$((2 * 2)) 
ans_3の値:

ヒアドキュメントも出力されました。
さらにドキュメントに

If any character of word is quoted with single or double quotes or a ‘\’, no interpretation is placed upon the characters of the document.Otherwise, parameter and command substitution occurs,

と記載されている通り
here_zshをクオーテーションで囲むなどしなかった場合には
計算が行われるなどしています。

計算されるだけで代入はされない、その後の処理に影響していない、
という点を見ればコメントアウトできているとも言えるのかもしれないですが、
出力されてしまうのでどうもこりゃ上手くいってない感があります。

zshで複数行コメントアウト

前置きが大変長くなってしまったのですが、bashでは<<から始まるもので複数行コメントアウトっぽいことができると確認できました。
一方で中身が出力されてしまうzshはどうすれば良いのでしょうか?

いろいろ調べていく中で出会ったstack overflowの回答にコレ!というものがありました。
【stack overflow】Block Comments in a Shell Script

: [ arg ... ] で何もしない

zshでもbashでも同じのようなのですが、行頭にコロンを置くとそれに続く記述の処理は何もせず終了コードを返すという処理になるそうです。
zshのドキュメントでは17 Shell Builtin Commandsに記載があります。
以下コロンに関する記述を全文引用

: [ arg ... ]
This command does nothing, although normal argument expansions is performed which may have effects on shell parameters. A zero exit status is returned.

何もしないけどコマンドの展開はしてしまうので: << COMMENTではなく: << 'COMMENT'もしくは: << \COMMENTで始めるのがベターという議論。これはzshだけでなくbashも同じなので、bashにおいてコメントアウトできてるっぽく見える<< COMMENTで始まるものも出力されていないだけで内部的にはコマンドの展開がされているらしい。。。おそろしや

というわけで一応のソリューションが冒頭のコメントアウト方法だということで落ち着きました。

echo "複数行コメントアウトしたいときは : <<  の後、'任意の文字列'で囲む!" 

: << 'COMMENT'
echo "ここはコメント!"
echo "ここはコメント!"
echo "ここはコメント!"
COMMENT

echo "ここは表示されるよ!" 

すっごい参考になったので追記

:の扱いについて本当にとても参考になる記事に、この記事を推敲している段階でぶち当たったのでリンクを貼ります。
何もしない組み込みコマンド ":" (コロン)の使い道

特に「何か書かなければならないところに仮置きする」というのはめっちゃなるほどとなりました。
これまで当たり障りない処理とかで誤魔化してたので、、、精進せねばですね。

stack overflowでされているその他のコメントアウト法議論

前出のstack overflowでは上記のものの他にもいくつか複数行コメントアウト法が議論されていましたので、いかにいくつか挙げてみます。

:のより簡素な使い方

:以下をシングルクオーテーションで囲む方法
コメントアウトを解除するときなんか消しそびれそうで怖いなと思いました。

#!/bin/zsh

: '
echo "このやり方でも"
echo "コメントアウトみたいにできる"
'

if文を使ういくつかの方法

条件が偽であればなんでもいいので、if文使って書くのは複数派閥がある様子。

#!/bin/zsh

# 1と0はイコールじゃないもんね
if [ 1 -eq 0 ]; then
    echo "このやり方でも"
    echo "コメントアウトみたいにできる"
fi

# もはや条件何も書かない
if [ ]; then
    echo "このやり方でも"
    echo "コメントアウトみたいにできる"
fi

# これが一番みてわかりやすい!と言ってる人も
if false; then
    echo "このやり方でも"
    echo "コメントアウトみたいにできる"
fi

エイリアスに書いちゃう

これまでに挙げた方法をエイリアスに書いちゃう派の人もいるようです。

#!/bin/zsh

# コロン
alias BEGINCOMMENT_COLON=": << 'COMMENT'"
alias ENDCOMMENT_COLON="COMMENT"

BEGINCOMMENT_COLON
    echo "このやり方でも"
    echo "コメントアウトみたいにできる"
ENDCOMMENT_COLON

# if文
alias BEGINCOMMENT_IF="if [ ]; then"
alias ENDCOMMENT_IF="fi"

BEGINCOMMENT_IF
    echo "このやり方でも"
    echo "コメントアウトみたいにできる"
ENDCOMMENT_IF

ブロック選択モードにして行頭全部に#つけりゃええやん

以下stack overflowより引用

In all honesty, why so much overengineering...
I consider it really a bad practice to write active code for generating passive code.
My solution: most editors have block select mode. Just use it to add # to all lines you want to comment out. What's the big deal...
Notepad example:
To create: Alt - mousedrag down, press #.
To delete: Alt-mousedrag down, shift-right arrow, delete.

ごもっともでございます。

VSCodeならalt+shift押しつつ範囲選択ですね。

まとめ

  • シェルスクリプトには複数行コメントアウトは正式には存在していない
  • zshにおいてはコロンとヒアドキュメントを使ってそれっぽいことすることが可能
  • bashにおいてはヒアドキュメントを使うだけでそれっぽいことすることが可能
  • ヒアドキュメントの開始行はコメントアウトしたい内容がコマンド展開されないように'COMMTNT'とするか\COMMENTとしておくのがベター
  • 他にもいくつか複数行コメントアウトっぽくする方法あるのでお好みで
  • でもちゃんとしたコメントアウトじゃないからコメントアウトしたい全部の行頭に#付けるのが一番よい

思いのほか長くなってしまった。
お付き合いいただきありがとうございました!

0
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?