"Glob"ها الگوهای سادهای هستند که میتوانند برای مطابقت نام فایلها و رشتهها به کار بروند. به طور معمول آنها خیلی قدرتمند نمیباشند. اگر قدرت بیشتری لازم دارید، چند گزینه در دسترس است.
اگر میخواهید بر تمام فایلهای که با جانشین A یا جانشین B مطابقت دارند عمل کنید، فقط هر دو را در یک سطر قرار بدهید:
rm -- *.bak *.old
اگر میخواهید یک OR منطقی فقط در قسمتی از جانشین به کار ببرید(بزرگتر از یک کاراکتر منفرد -- برای مورد یک کاراکتری، کلاسهای کاراکتر داخل کروشه کفایت میکند)، در Bash، میتوانید از بسط ابرو استفاده کنید:
rm -- *.{bak,old}
اگر موردی بازهم بیشتر عمومی و قدرتمند نیاز دارید، در KornShell یا BASH میتوانید از جانشینهای توسعهیافته استفاده کنید. در Bash، لازم است گزینه extglob تنظیم باشد. به این ترتیب میتوند کنترل شود:
shopt extglob
و به این ترتیب تنظیم میشود:
shopt -s extglob
برای دستگرمی، تمام فایلهایی که با foo شروع میشوند و به .d ختم نمیشوند را به دایرکتوری foo_thursday.d منتقل میکنیم:
mv foo!(*.d) foo_thursday.d
یک مثال پیچیدهتر -- حذف تمام فایلهایی که نام آنها شامل Pink_Floyd و هچنین فاقد The_Final_Cut باشد:
rm !(!(*Pink_Floyd*)|*The_Final_Cut*)
ضمناً: این نوع از الگوها میتوانند با KornShell نیز به کار بروند. در آنجا نیازی به فعال کردن نیست، بلکه الگوهای پیشفرض هستند.
پرسش و پاسخ 16 (آخرین ویرایش 2009-12-30 17:14:33 توسط MatthiasPopp)
غالباً یک فرمان تعدادی فایل را به عنوان شناسه میپذیرد، به عنوان مثال:
zcat -- *.gz
در برخی سیستمها، به جای zcat باید از gzcat استفاده کنید. اگر هیچ یک در دسترس نیستند، یا اگر علاقمند به انجام بازیهای حدسی نمیباشید، فقط از gzip -dc به جای آن استفاده کنید.
کاربرد -- از ایجاد نتایج غیر منتظره ناشی از نام فایلهایی که با علامت خط تیره آغاز میشوند، ممانعت میکند.
اگر یک حلقه آشکار میخواهید یا اگر فرمان شما نمیتواند شناسههای چندتایی را در یک فراخوانی قبول کند، حلقه for میتواند به کار برود:
# Bourne for file in ./*.gz do echo "$file" # "$file" انجام کاری با done
برای انجام آن به طور بازگشتی، از find استفاده کنید:
# Bourne find . -name '*.gz' -type f -exec do-something {} \;
اگر به دلایلی لازم است فایلها داخل پوسته پردازش بشوند، پس نتایج find را در حلقهای بخوانید:
# Bash while IFS= read -r file; do echo "Now processing $file" # "$file" انجام کار مورد علاقه با done < <(find . -name '*.gz' -print)
این مثال از جایگزینی پردازش استفاده میکند(همچنین پرسش و پاسخ 24 را ببینید) هرچند لوله نیز در بسیاری از وضعیتها میتواند مناسب باشد. به هرحال، نام فایلهایی که شامل سطرجدید میباشند را به طور صحیح مدیریت نمیکند. برای مدیریت نامفایلهای دلبخواهی، پرسش و پاسخ 20 را ببینید.
پرسش و پاسخ 15 (آخرین ویرایش 2010-07-30 14:32:06 توسط GreyCat)
تغییر مسیر خروجی استاندارد یک فرمان منفرد به همین آسانی است:
date > file
برای تغییر مسیر خروجی استاندارد خطا:
date 2> file
برای تغییر مسیر هردو:
date > file 2>&1
یا، روش تفننیتر:
# اما غیرقابل حمل date > file 2>&1 معادل با ,Bash فقط در date &> file
تغییر مسیر کل یک حلقه:
for i in $list; do echo "Now processing $i" # more stuff here... done > file 2>&1
به هرحال، اگر باید خروجی فرمانهای بسیاری تغییر مسیر داده شود، این روش میتواند کسل کننده گردد. اگر تمام خروجی یک اسکریپت باید به یک فایل(به عنوان مثال فایل log بروند)، فرمان exec میتواند به کار برود:
# "log.txt" تغییر مسیر هردو خروجی استاندارد وخطا به فایل exec > log.txt 2>&1 # میروند "log.txt" تمام خروجی از جمله خطای استاندارد اکنون به فایل
در غیر آن صورت، گروهبندی فرمانها، کمک میکند:
{ date # some other commands echo done } > messages.log 2>&1
در این مثال، خروجی تمام فرمانهای درون ابروها به فایل messages.log تغییر مسیر داده میشود.
پرسش و پاسخ 14 (آخرین ویرایش 2010-07-01 13:04:42 توسط GreyCat)
در شل، عملگری(صریح) برای الحاق نمودن رشتهها (لفظی یا بدون ارجاع به متغیر) وجود ندارد، فقط میتوانید آنها را در مجاورت هم بنویسید:
var=$var1$var2
اگر طرف راست شامل کاراکترهای فضای سفید باشد، لازم است که نقلقولی بشود:
var="$var1 - $var2"
اگر شما رشتهای الحاق میکنید که به قسمتی از نام متغیر شباهتی ندارد، تنها تمام آنها را در کنار یکدیگر قرار بدهید:
var=$var1/.-
در غیر این صورت، ابروها یا نقلقولها میتوانند برای رفع ابهام طرف راست به کار بروند:
var=${var1}xyzzy # به عنوان نام متغیر تفسیر خواهد شد var1xyzzy بدون ابروها var="$var1"xyzzy # ترکیب دستوری جایگزین
جایگزینی فرمان به خوبی میتواند به کار برود. سطر زیر یک فایل ثبت وقایع logname متشکل ازتاریخ جاری ایجاد میکند، که منجر به نامهایی به عنوان مثال مانند log.2004-07-26 میشود:
logname="log.$(date +%Y-%m-%d)"
تفاوتی میان استفاده مجدد نام یک متغیر یا تخصیص مجدد مقدار(رشتهای که نگاه میدارد) دلخواه به آن، وجود ندارد:
string="$string more data here"
Bash (از) نگارش 3.1 عملگر جدید += را دارد که ممکن است گاه گاهی ببینید:
string+=" more data here" # !بینهایت غیرقابل حمل
به طور معمول بهترین کار استفاده از ترکیب دستوری قابل حمل است.
پرسش و پاسخ 13 (آخرین ویرایش 2009-12-30 08:27:15 توسط MatthiasPopp)
میتوانید از گزینه -c پوسته برای اجرای شل جهت اجرای تکه کوتاهی از اسکریپت استفاده کنید:
sh -c 'echo "Hi! This is a short script."'
این کار بدون وسیلهای برای عبور دادن دادهها به آن، روش قشنگ معمولاً بلااستفادهای است. بهترین راه برای عبور دادن خُردهای از دادهها به پوسته شما، ارائه آنها به عنوان پارامترهای مکانی است:
sh -c 'echo "Hi! This short script was run with the arguments: $@"' -- "foo" "bar"
به -- قبل از پارامترهای مکانی واقعی، توجه کنید. اولین شناسهای که به پردازش شل میدهید(که شناسهای برای گزینه -c نیست) در متغیر $0 قرار داده میشود. پارامترهای مکانی از $1 شروع میشوند، بنابراین ما یک placeholder کوچک در $0 قرار میدهیم. این هر چیزی که شما مایل باشید میتواند باشد، در مثال، ما -- عمومی را به کار بردهایم.
این شیوه غالباً در اسکریپتنویسی پوسته، موقعی به کار میرود که میخواهید یک برنامه خطفرمانی غیر پوستهای برخی کُدهای bash را اجرا کند، مثلاً با find(1):
find /foo -name '*.bar' -exec bash -c 'mv "$1" "${1%.bar}.jpg"' -- {} \;
در اینجا از find درخواست میکنیم فرمان bash را برای هر فایل *.bar که پیدا میکند، به عنوان اولین پارامتر مکانی پردازش bash اجرا نماید. پردازش bash پس از انجام بسط پارامتر روی پارامتر مکانی اول، دستور mv را به منظور تغییر نام فایلهای با پسوند bar به فایلهای با پسوند jpg اجرا میکند.
به طور جایگزین، اگر برنامه غیرپوستهای شما امکان تنظیم متغیرهای محیط را برای شما فراهم میکند، میتوانید این کار را انجام بدهید و سپس با استفاده از متغیرهای معمولی همان نام آنها را بخوانید.
به طور مشابه، فرض کنید برنامهای(مثلاً مدیریت فایل) به شما اجازه میدهد یک فرمان خارجی تعریف کنید که یک شناسه به آن ضمیمه خواهد شد، اما شما آن شناسه را جایی در وسط لازم دارید. در آن حالت:
#!/bin/sh sh -c 'command foo "$1" bar' -- "$@"
پرسش و پاسخ 12 (آخرین ویرایش 2011-04-12 20:01:42 توسط Lhunath)