シェルスクリプト総合 その28

■ このスレッドは過去ログ倉庫に格納されています
2018/08/02(木) 05:22:16.82
シェルスクリプトの総合スレです。
□お約束
・特記なき場合はBourne Shell(/bin/sh)もしくはPOSIX準拠の互換シェルがデフォルトです。
 bash/zsh/ksh/ash/dash/yash/poshなどの専用機能に依存する場合は明示しましょう。
 Linuxユーザは/bin/shの正体がbashまたはdashなので特に注意。
 FreeBSDユーザは/bin/shの正体がashなので注意。
・POSIXについてのリンクは https://en.wikipedia.org/wiki/POSIX にまとめられています
 最新の仕様はこちらへ http://pubs.opengroup.org/onlinepubs/9699919799/
 (左上の「Shell & Utilities」 から参照することができます。)
・v7 shに一番近くて、現役(?)のshは、OpenSolaris由来のheirloom sh。
  http://src.illumos.org/source/xref/illumos-gate/usr/src/cmd/sh/
  http://heirloom.sourceforge.net/sh.html
・csh/tcshのシェルスクリプトは推奨されません。
 (理由は「csh-whynot」でググれ)
・UNIXにはシェルスクリプトに便利な小さなコマンドがいろいろあります。
 manや参考リンクを見ましょう。
 aproposないしはman -kでそれらしい単語による簡単な検索もできます。
・シェルで使えるワイルドカード等は正規表現ではありません。
 正規表現の話題はスレ違い(正規表現スレへ)
・シェルスクリプトのことをシェルってゆうな
□初心者へのアドバイス:
・適した道具を判断するのも頭の重要な使い方。シェルスクリプトよりも
 awkまたはperlの方が適した処理にはそちらを使いましょう。
・知らないコマンドが出てきたらmanを引きましょう。
・思い通りに動かないときは、まずは sh -x でトレースしましょう。
□回答者への注意事項:
・シェルスクリプトでの処理方法を質問しているのに、よくわからずに
 「そういうのはperl使いましょう」と回答するのはやめましょう。
 安易にperlに逃げずにシェルスクリプトで処理するのが頭のいいやり方。
 質問に対して問題が間違ってるといちゃもんをつけるのもやめましょう
シェルスクリプト総合 その27
https://mevius.5ch.net/test/read.cgi/unix/1525337663/
2019/08/05(月) 12:11:44.60
一応これでうまくいきました

cat > index.txt << "EOF"
ほにゃらら
ほんじゃらら
かきく
EOF

seq 0 5 | awk -v 'ORS= あいう\nかきく\n' '{print $1}' >> index.txt
sed -i -e '$d' index.txt
echo さしす >> index.txt
2019/08/05(月) 12:13:50.09
>>132
自分のやり方はなだったらしそうなのでそのキーワード調べてみます
2019/08/05(月) 12:28:22.64
えー、難しい...
>>129はこんなもんしか考えてなかったけど

awk 'BEGIN{print"ほにゃらら\nほんじゃらら";for(i=0;i<6;i++){printf"かきく\n%d あいう\n",i};print"さしすせそ"}'
136名無しさん@お腹いっぱい。
垢版 |
2019/08/05(月) 12:28:39.74
>>128
cat <<END >index.txt
ほにゃらら
ほんじゃらら
$(printf 'かきく\n%s あいう\n' $(seq 0 5))
さしすせそ
END
2019/08/05(月) 12:29:43.26
あ、バックスラッシュが円記号になってしまった...
2019/08/05(月) 12:30:25.69
cat > index.txt << "EOF"
ほにゃらら
ほんじゃらら
EOF

printf "かきく\n%s\n" {0..5}あいう >> index.txt

これだと大分短くて出来たけど、数字と「あいう」の間にスペースが入れられない
2019/08/05(月) 12:32:03.53
被ってしまった
みなさんありがとうございます
質問ぬしの自分は>>138です
135と136のやり方調べてみます
2019/08/05(月) 12:43:19.31
>>136
catで書き出すファイルの中でprintfをやって、そのrintfの中でseqで連番作るんでね
これだとファイルへの書き込みは1回ですみますね

>>135
これも> index.txtで終えれば1回ですみますね
だけどawk難しいです
がんばって135のやり方理解できるようにします
141名無しさん@お腹いっぱい。
垢版 |
2019/08/05(月) 13:06:08.43
>>140
> これだとファイルへの書き込みは1回ですみますね

どんな方法で作るにせよ { } で括れば一回にまとめられるよ

{
printf '%s\n' 'ほにゃらら'
printf '%s\n' 'ほんじゃらら'
printf 'かきく\n%s あいう\n' $(seq 0 5)
printf '%s\n' 'さしすせそ'
} >index.txt
2019/08/05(月) 15:17:00.15
>>141
こんな方法もあるんですね
ありがとうございます

curlの標準出力の6行目を変数に入れて他の文字列と組み合わせて標準出力に出力したいです。
URL="hoge"
ROKU="curl -sS $URL | sed -n '6p'"
echo "あいうえお"$ROKU"さしすせそ"

しかし結果はこうなりました
あいうえおcurl -sS hoge | sed -n '6p'さしすせそ
2019/08/05(月) 15:18:57.84
>>142
続き
>>141
さんで教わった{}を使えばよさそうな気がするですがエラーになりました
2019/08/05(月) 16:33:57.17
printf "あいうえお"
$ROKU #chomp のように\nを消すパイプ必要だと思う
printf "さしすせそ"
2019/08/05(月) 17:18:21.98
>>144
echoじゃなくてprintfなら改行されないからあいうえおに続けて書けますね
だけど、このままだとROKUのところで改行されてしまうからさしすせそは2行目に来ちゃいますね
2019/08/05(月) 18:01:04.99
>>128
> と書かれたテキストファイルを作りたいです

こうすれば良いのでは?

cat<<HERE > index.txt
ほにゃらら
ほんじゃらら
かきく
0 あいう
かきく
1 あいう
かきく
2 あいう
かきく
3 あいう
かきく
4 あいう
かきく
5 あいう
さしすせそ
HERE
147名無しさん@お腹いっぱい。
垢版 |
2019/08/05(月) 19:41:32.36
>>142
コマンドの標準出力を変数に設定したりコマンドの引数にしたりするには
「コマンド置換」 $(…) を使います

あと、「6行目」以外の部分も取り出したくなったときに curl を何度も実行したくはないので
curl と sed は分けておきたいところ

するとこんな感じ

PAGE=$(curl -sS "$URL")
ROKU=$(printf '%s\n' "$PAGE" | sed -n '6p')
NANA=$(printf '%s\n' "$PAGE" | sed -n '7p')
printf 'あいうえお%sさしすせそ\n' "$ROKU"
printf 'かきくけこ%sたちつてと\n' "$NANA"
2019/08/06(火) 10:37:34.03
>>146
複数行書く方法ありがとうございます

>>147
うまくいきました
ありがとうございます
2019/08/29(木) 02:47:55.71
$ bash -c 'unset a b;echo 10|read a;let b=a+10;echo $b'
10

この動作ってPOSIX的にはどうなの?
2019/08/29(木) 05:08:51.78
読みづれーなw

bash -c '
unset a b
echo 10 | read a
# echo $a = 空
let b=a+10
echo $b
'

letはPOSIXにない
readはサブシェルなんだからaは空に決まってる
ごく普通の正しい動作
2019/08/29(木) 05:09:53.11
正しいコード

bash -c '
unset a b
echo 10 | {
read a
let b=a+10
echo $b
}
'
2019/08/29(木) 23:22:24.50
letまで考えて無かったすまん
まあ説明の為なのでそこは目をつぶってくれたまい

opensuzeなんだけども

$ ksh -c 'unset a b;echo 10|read a;let b=a+10;echo $b'
20
$

この動作ってPOSIX的にはどうなの?
2019/08/30(金) 06:34:55.40
バグ
2019/09/10(火) 19:48:12.68
curlのやり方教えてください
ブラウザのフォームにjsonを書いてポストボタンを押すとokが出るけどターミナルからcurlでやろうとすると、ボディが空だとなり送信出来ません
書き方間違えていますか?

https://i.imgur.com/Nvm7CMj.jpg
2019/09/11(水) 14:49:50.58
ブラウザの開発ツールからcURLでコピーした方が早そうだぞ
2019/09/11(水) 15:08:13.17
>>155
どうもありがとうございます
-H 'Content-Type: application/json'
を付けたら動きました
2019/09/12(木) 17:05:57.53
アプリの起動判定をしたいんですが
ps -alxw | grep アプリ名
2行より多ければアプリが起動してるんだと思いますけど確実な方法ありますか?
2019/09/12(木) 17:48:19.11
killall -0 アプリ名
とか?
2019/09/12(木) 19:39:47.90
>>158
どうもありがとうございます
そのコマンドで何も変えらなかったら起動中という事であってますか?
2019/09/13(金) 07:33:53.20
起動してないとわかっても、その直後に起動することも有るから
ロックを掛けないと確実にはならないけどな
2019/09/14(土) 01:15:30.85
pgrepでええやん、と思ったけど環境依存か
2019/09/14(土) 02:03:14.71
>>160
どの環境でも使えるロックの機構ってなにかある?
2019/09/14(土) 03:45:32.29
>>162
set -C もしくは mkdir
164名無しさん@お腹いっぱい。
垢版 |
2020/01/20(月) 04:38:26.60
怒らないで教えて欲しいんだけど何でお前らPowerShell使わないの?
今やPowerShellの方が勉強会でドヤれるしちょまどなんかもPowerShellだってよ
2020/01/20(月) 07:42:20.05
>>164
なんでそんなに無意味な宣伝するの?
2020/01/20(月) 13:22:02.65
>>164
Dockerのalpine、debianなど、ほぼすべてのイメージでことごとく動かないから。
shで十分なのに、PowerShellなんて使いませんよw
2020/01/20(月) 15:09:31.95
>>164
linux のコマンド群が使えないから
2020/01/21(火) 10:19:02.74
ソースコードをダウンロードしてビルドしようとしたら…

ERROR: You must either be root or be able to use sudo

ビルドするのに root になれって? この時点で話にならないよ。
2020/01/21(火) 11:46:33.77
>>168
それってパワーシェルの話?
2020/01/21(火) 21:58:33.32
そのメッセージでググっても、インストール時のメッセージしか見つからないね
https://www.google.com/search?q=%22ERROR%3A+You+must+either+be+root+or+be+able+to+use+sudo%22

ビルドしようとしたら出たって言うから、
別の話か、嘘つきでは?w
2020/01/21(火) 22:20:06.48
>>168
エラーメッセージには「sudo を使え」と書いてあるが
2020/01/22(水) 02:27:49.96
とりあえず適当なことを言ってPowerShellを
貶めたかったんだろ?バレて逆効果になってるがw
2020/02/17(月) 09:08:53.72
grep 等の外部コマンド?を使わずに任意の文字列の最初に見つかる3桁の数字を得る方法があれば教えて下さい
(例えば、abc_4_de_99_fgh_ijklm_no_567_2_123_pqrst_uvwx_yz だと567)
2020/02/17(月) 12:51:47.30
bashなら=~と、$BASH_REMATCH使えばできるんじゃね
2020/02/17(月) 12:52:43.82
>>173
おまえは俺かw

似たようなことをやったよ。grepで(笑)

外部コマンドを使わないなら、こうなるから面倒くさいんだよな。
できるけど面倒くさい。だから遅いけどgrepにした。

v=abc_4_de_99_fgh_ijklm_no_567_2_123_pqrst_uvwx_yz
v=${v#"${v%%[0-9][0-9][0-9]*}"}
v=${v%"${v#[0-9][0-9][0-9]}"}
2020/02/17(月) 13:01:06.73
シェルスクリプトだと(bash依存は知らん)
マッチした部分を含めて削除はできるけど
マッチした部分を残したその他を削除が簡単にできないんだよね。

だから「マッチした部分を含めて削除」したら残りが
「マッチしなかった部分」になるのを利用して、
改めて全体から「マッチしなかった部分」を取り除くというのを前後でやればできる。
2020/02/19(水) 01:25:53.44
ありがとうございます
面倒でも前後から不要部分を取り除くしかないのですね
2020/02/19(水) 03:44:34.80
「マッチした部分を含めて削除」ってどうやるの?
2020/02/19(水) 07:39:09.82
>>178
${v%%[0-9][0-9][0-9]*} ってかいてあるやん

abc_4_de_99_fgh_ijklm_no_567_2_123_pqrst_uvwx_yz ・・・(1)

↓ ${v%%[0-9][0-9][0-9]*}

abc_4_de_99_fgh_ijklm_no_ ・・・(2)

↓ (1)ー(2)

567_2_123_pqrst_uvwx_yz ・・・(A)

↓ ${v%"${v#[0-9][0-9][0-9]}"}

_2_123_pqrst_uvwx_yz ・・・(B)

↓ (A)ー(B)

567
2020/02/19(水) 12:18:21.18
>>179
丁寧にありがとう!
2020/02/21(金) 17:55:59.15
今更だが>>179

× ${v%"${v#[0-9][0-9][0-9]}"}
○ ${v#[0-9][0-9][0-9]}"

だな
182名無しさん@お腹いっぱい。
垢版 |
2020/02/22(土) 16:29:53.08
それらをPowerShellで書くとどうなるか?
一度でも考えてみたことはありますか?
2020/02/22(土) 16:45:28.02
PowerShellで書くとLinuxで動かすのが大変になります。
2020/02/26(水) 03:34:54.75
5年くらいするとセキュリティパッチが出なくなって使用禁止になるから
また作り直しになるんだろうどうせ。
2020/03/13(金) 23:32:50.67
例えば中身が
3,soba,inarizusi,hishimoti,ushioziru ...
のcsvファイルなら最初の数字が3なので以下の文字列の3番目をつないで soBa inArizusi hiShimoti usHioziru ... でbash ... というように
csvファイルの最初の値で以降の処理が変わる場合はどうすればいいでしょうか?
一度ファイルを最後まで空読みすれば簡単なのですが、読み込みは1度で終わらせたいです。
2020/03/14(土) 16:01:32.08
宿題は自分でやりなよ
187名無しさん@お腹いっぱい。
垢版 |
2020/03/14(土) 17:54:51.47
>>185
なんでそんな分かりづらい説明するんだ?
188名無しさん@お腹いっぱい。
垢版 |
2020/03/14(土) 17:55:37.96
全く意味がわからん。
誰か翻訳してくれ
2020/03/14(土) 18:20:44.65
>>185
引数で単語と数字nを受け取ってn文字目の単語を返す関数を書けばいいだけ
2020/03/14(土) 18:34:21.42
>>189
日本語ではなくてbashでお願いします!!
2020/03/14(土) 18:41:35.26
>>188
結果だけ見たら

入力テキストをカンマで区切って
最初の数値を取得 ⇒ nとする
残りの文字列のn番目の文字をアッパーケースに変換して空白区切りで連結する

かな
俺ならテキストが小さいならBashで一気に読んで処理すればいいと思うしでかいならPythonとかで処理する
192名無しさん@お腹いっぱい。
垢版 |
2020/03/14(土) 20:00:13.75
> 俺ならテキストが小さいならBashで一気に読んで処理すればいいと思うしでかいならPythonとかで処理する
これはなんで? 小さいとかの境目はどれくらい?
2020/03/14(土) 20:15:35.44
そんなこと聞いてくるようではセンスがないよ
センスがない奴にいくら説明しても無駄
2020/03/14(土) 20:18:24.03
pythonだと単語のn文字目を取り出すなんて朝飯前だしな
関数書く手間が一つ省ける
2020/03/14(土) 20:24:34.26
疑問になった点は「テキストが小さいなら」です。
その他の話は聞いてません。
2020/03/14(土) 20:26:37.39
> pythonだと単語のn文字目を取り出すなんて朝飯前だしな

単語の3文字目(オフセット2から1文字)を取り出す
echo "${str:2:1}"

Pythonだとどう朝飯前なんですか?
2020/03/14(土) 20:48:13.52
早く答えのスクリプトを書いてください!!
遅いですよ!!
2020/03/14(土) 21:05:30.89
>>195
個人的には今どきなら1MB以下なら小さい1GB以上ならでかい
その間ならケースバイケース
PCの性能やメモリ容量とかと相談
2020/03/14(土) 21:19:09.39
>>185
> 一度ファイルを最後まで空読みすれば
これはどういうスクリプトなの?
2020/03/14(土) 22:00:38.59
>>198
1MBってことは、1行256バイトとして4096行以下なら小さいってことですね。
10MBで約4万行、100MBで約40万行、1GBで400万行
昔のExcelの制限が1シート65000行だったことを考えると
エクセルで扱えるデータなら問題なさそうですね
2020/03/14(土) 22:38:14.38
アスペかよw
アスペならちゃんと改行コードも数えるんだぞ?
2020/03/15(日) 13:40:07.33
>>201
お前なんのためにレスしたの?
>>200が言ってることは正しいし、お前の主張は何も含まれていない。
ただ、レスしただけ
2020/03/20(金) 18:13:05.78
185です。説明不足ですみません。
やりたいことは1行目を読み取り、2行目以降はそのままwhileループで最後まで読みたいのです。
ただそのためだけにフラグを使うのもなんだかなぁと。
2020/03/20(金) 19:59:54.90
わざわざ空読みする理由が分かんねえ
今出来てる状態のスクリプト貼って何がうまく行かないのか説明してみ
2020/03/21(土) 00:02:26.75
#!/bin/sh

お願いします
2020/03/21(土) 00:18:20.18
>>205
そこまでは上手く書けてる
その調子で頑張れ
2020/03/21(土) 13:58:51.87
>>190
今日一番ワロタ
キミもbashで質問しなよ
2020/03/21(土) 14:07:34.22
bash は重複しない4文字である。つまり2bitのデータと考えられる
日本語を1文字16bitのデータと考えると、
1文字をbashの4文字×8組=32個のデータで表現できる
つまりbashで質問すると32倍のデータが必要になる
1レスに2048バイト、1024文字を書き込めるから32文字の質問であれば
bashで質問することは可能
2020/03/21(土) 15:26:35.50
向いてないんだろ
こういうの
2020/03/22(日) 10:34:31.24
>>208
> 1文字をbashの4文字×8組=32個のデータで表現できる

この行から間違えてるだろ
2020/03/22(日) 17:21:49.94
>>199
こいうんじゃね?

というのを期待してた?残念w
2020/03/23(月) 20:28:21.83
>>211
while IFS=, read は思いつくけど
最後まで空読みってどんな処理なのか気になっただけ
2020/03/24(火) 21:32:10.12
>>196
>>> str = "foobarbaz"
>>> print(str[3])
b

python3は重いかわりに色々便利なのが定義されてる
2020/04/11(土) 10:24:56.50
タブで字下げされてるテキストファイルの中にある
_TAG("S.T.A.R.T") が含まれる行と
_TAG("E.N.D") が含まれる行との間にある行数を数えるにはどうすればいいでしょうか
2020/04/11(土) 11:20:47.30
...... | sed -n '/_TAG("S.T.A.R.T")/,/_TAG("E.N.D")/p' | wc -l
こんな感じ
2020/04/12(日) 07:36:15.54
さんくす
sedやwcなどの外部コマンド?を使わない方法はありますか?
2020/04/12(日) 07:43:15.09
宿題は自分でやろう
2020/04/12(日) 13:13:39.28
>>216
そういうことするなら
perl5に行きなよ
普通はそれも最初から入ってて、文字処理に強い言語だから
---
そんな処理だと、その処理の前に
タグがネストしてたり、正しく対になっているのか調べれた方が安全
おみあげに類似したコードあげる
https://ideone.com/XCz8Ou
対が壊れてたら -1 を
壊れてないならネストの最大の深さを返す
2020/04/12(日) 17:00:27.78
>>218
ダクネスが居たように見えた
2020/04/13(月) 19:58:42.58
>>214
問題の意図が明確
その _TAG("S.T.A.R.T") とかは字下げされているのかどうなのか
_TAG("S.T.A.R.T") が含まれてると言うが、
一行 は _TAG("S.T.A.R.T") だけではないのか?
echo ' _TAG("S.T.A.R.T")' なんかも考慮する必要あるのか?

俺だったら一行まるまる、_TAG("S.T.A.R.T") だけにするがな
その方がパースしやすいので
2020/04/13(月) 19:59:20.04
あと _TAG("S.T.A.R.T") は複数あるのか?という質問も追加
ネストされている場合は考慮するのか?とか
2020/04/13(月) 20:01:47.27
まあいずれにしろテキストファイルは read で読み込めるんだから
こんな感じになるだろうが

while IFS= read -r line; do
 case $line in
  '_TAG("S.T.A.R.T")') なんとか ;;
  '_TAG("E.N.D")') かんとか ;;
 esac
 if 条件; then
  count=$((count + 1))
 fi
done < テキストファイル
2020/04/13(月) 20:03:02.16
>>218みたいにそういう場合だったら Perl にとかいうやつがいるけど
シェルスクリプトで同等のコードを書いても大差ないんだけどなw
2020/04/13(月) 21:07:49.84
sedやwc使うなとかいう制約も謎だわ
言語やツールが何であれ結局裏で似たようなライブラリの関数呼び出すのに
2020/04/13(月) 21:38:58.14
理由がわからないなら普通に聞けばいいだけなのに。
自分が知らないだけなのに、なんでそんなことありえない、
あってたまるかみたいな書き込みをするんだろうねw
2020/04/13(月) 22:33:21.03
理由がわからないんじゃなくて、そんなことを言い出す人間が理解できないって話だよ
こだわりがあるならこんなところで人に聞くんじゃないよって話で
2020/04/13(月) 22:41:09.86
自分とは違う方針の質問は受け付けない
そんなレスはこてんぱんにしてやる。ですか?
2020/04/13(月) 23:40:18.71
自分の期待とは違う方針の回答は受け付けない
そんなスレはこてんぱんにしてやる。じゃないの?
2020/04/14(火) 04:36:43.78
>>227
スレの流れ見たけど受け付けられてるじゃん
宿題を済ませたいのか知らないが
人に聞くくらいなら針路変更したらどう?
知り合いにコンピューターの勉強してた人居るけど
大学で課題をすべて友人からパクってたそうだ
今その人バイトで食いつないでる
ITの仕事できたら低くても年収400万は行ったろうにね
2020/04/14(火) 07:50:06.07
>>229
知り合いの人は大学をやめて会社を起こして金持ちになったよ。
お前が何をいいたいのか知らんが
2020/04/14(火) 07:52:02.07
>>228
さんくすとお礼を言った上に、条件をつけ忘れてたから
補足しただけでしょ

たったそれだけで、はぁ?理解できない。謎だわ。なんでそんな事するの?バカなの?
みたいに言うのはなんで?
232名無しさん@お腹いっぱい。
垢版 |
2020/04/14(火) 08:10:41.32
>>230
知り合いは宝くじ当たったよ

中卒、高卒、大卒の平均年収見てみ
■ このスレッドは過去ログ倉庫に格納されています
5ちゃんねるの広告が気に入らない場合は、こちらをクリックしてください。