के साथ समस्या मैं अंतिम पंक्ति को छोड़कर फ़ाइल की सभी पंक्तियों को हटाने की कोशिश कर रहा था लेकिन निम्न आदेश काम नहीं करता था, हालांकि file.txt खाली नहीं है।बैश आउटपुट रीडायरेक्शन
$cat file.txt |tail -1 > file.txt
$cat file.txt
ऐसा क्यों है?
के साथ समस्या मैं अंतिम पंक्ति को छोड़कर फ़ाइल की सभी पंक्तियों को हटाने की कोशिश कर रहा था लेकिन निम्न आदेश काम नहीं करता था, हालांकि file.txt खाली नहीं है।बैश आउटपुट रीडायरेक्शन
$cat file.txt |tail -1 > file.txt
$cat file.txt
ऐसा क्यों है?
एक पाइपलाइन के माध्यम से एक फ़ाइल से रीडायरेक्ट करना एक ही फाइल में वापस असुरक्षित है; को tail
से पहले पाइपलाइन के अंतिम चरण को सेट करते समय खोल से अधिलेखित किया जाता है, तो आप पहले चरण को पढ़ना शुरू कर देते हैं, तो आप खाली आउटपुट के साथ समाप्त होते हैं।
करो बजाय निम्नलिखित:
tail -1 file.txt >file.txt.new && mv file.txt.new file.txt
... अच्छी तरह से, वास्तव में, ऐसा नहीं करते हैं कि उत्पादन कोड में; खासकर यदि आप एक सुरक्षा के प्रति संवेदनशील वातावरण में कर रहे हैं और रूट के रूप में चल रहा है, निम्नलिखित अधिक उपयुक्त है:
tempfile="$(mktemp file.txt.XXXXXX)"
chown --reference=file.txt -- "$tempfile"
chmod --reference=file.txt -- "$tempfile"
tail -1 file.txt >"$tempfile" && mv -- "$tempfile" file.txt
एक और दृष्टिकोण (अस्थायी फ़ाइलों से परहेज है, जब तक <<<
परोक्ष अपने मंच पर उन्हें बनाता है) पीछा कर रहा है:
lastline="$(tail -1 file.txt)"; cat >file.txt <<<"$lastline"
(उपर्युक्त कार्यान्वयन बैश-विशिष्ट है, लेकिन ऐसे मामलों में काम करता है जहां गूंज नहीं है - जैसे कि अंतिम पंक्ति में "--version" है, उदाहरण के लिए)।
अंत में, एक moreutils से स्पंज का उपयोग कर सकते हैं:
tail -1 file.txt | sponge file.txt
ऐसा लगता है कि आप इसे उसी फ़ाइल नाम पर वापस लिख रहे हैं। यदि आप निम्न कार्य करते हैं तो यह काम करता है:
$cat file.txt | tail -1 > anotherfile.txt
के लिए कोई ज़रूरत नहीं है खो जाएगा होगा अटकलें "लगता है" , और रीडायरेक्शन (इस प्रकार, ट्रंकेटिंग मोड में 'anotherfile.txt' का खुलासा) * * निष्पादन से पहले * होता है (इस मामले में,' पूंछ 'के; चाहे यह' बिल्ली 'के निष्पादन से पहले होता है, अनिश्चित है, लेकिन एक कार्यक्रम के बाद से चूंकि 'बिल्ली' को शुरू करने के लिए समय की आवश्यकता होती है, इसलिए यह बहुत अधिक संभावना है कि 'catfile.txt' का कटाव पहले से ही हो जाएगा 'बिल्ली' लोड होने से पहले और इनपुट के लिए अपना तर्क खोलने के लिए तैयार हो जाएगा)। –
लुईस Baumstark कहते हैं, यह है कि आप उसी फ़ाइल नाम के लिए लिख रहे पसंद नहीं है।
ऐसा इसलिए है क्योंकि खोल "file.txt" खुलता है और "cat file.txt" चलाने से पहले इसे पुनर्निर्देशन करने के लिए इसे छोटा करता है। तो, आप
tail -1 file.txt > file2.txt; mv file2.txt file.txt
tail -1 > file.txt
करने के लिए है आपकी फ़ाइल के ऊपर लिख देगा, क्योंकि पुनर्लेखन का से पहले क्या होगा आपके पाइप लाइन में आदेशों के किसी भी क्रियान्वित कर रहे हैं एक खाली फ़ाइल को पढ़ने के लिए बिल्ली के कारण।
'बिल्ली' निष्पादित होने से पहले, बैश ने अपनी सामग्री को साफ़ करने के लिए पहले से ही 'file.txt' खोला है।
सामान्य रूप से, उन फ़ाइलों को न लिखें जिन्हें आप उसी कथन में पढ़ रहे हैं। इसे उपरोक्त के रूप में एक अलग फ़ाइल में लिखकर चारों ओर काम किया जा सकता है:
$cat file.txt | tail -1 >anotherfile.txt $mv anotherfile.txt file.txtया स्पंज जैसे moreutils:
$cat file.txt | tail -1 | sponge file.txtयह काम करता है क्योंकि स्पंज तब तक प्रतीक्षा करता है जब तक इसकी इनपुट स्ट्रीम अपनी आउटपुट फ़ाइल खोलने से पहले समाप्त हो जाती है।
क्या 'cat' आपको 'tail -1 file.txt' पर कोई मान देता है? यह निश्चित रूप से चीजें ** दूर ** कम कुशल बनाता है यदि आपका 'file.txt' बड़ा है - यदि 'पूंछ' में प्रत्यक्ष खोज योग्य फ़ाइल डिस्क्रिप्टर है, तो यह सीधे फ़ाइल के अंतिम केबी पर कूद सकता है, केवल उसे पढ़ सकता है, और पिछली पंक्ति को 1kb से बड़ा होने पर बैक अप लेने की अंतिम पंक्ति को खोजने का प्रयास करें; यदि यह सब 'बिल्ली' से एक पाइप है, तो इसे शुरुआत से फ़ाइल को पढ़ना होगा - इससे कोई फर्क नहीं पड़ता कि यह कितना बड़ा है - अंत तक पहुंचने के लिए। –
... 'पूंछ -1
जब आप पार्टी की योजना बनाई अपने कमांड स्ट्रिंग सबमिट करते हैं, यह निम्नलिखित है:
जब तक 'बिल्ली' पढ़ना शुरू होता है, तब तक 'file.txt' को 'पूंछ' द्वारा छोटा कर दिया गया है।
यह यूनिक्स और शैल पर्यावरण के डिजाइन का हिस्सा है, और मूल बोर्न शैल के लिए सभी तरह से वापस चला जाता है। 'एक विशेषता है, एक बग नहीं।
क्या (2) और (3) के बीच ऑर्डरिंग निर्दिष्ट है? मेरी धारणा यह है कि वे एक साथ हो रहे हैं, जो प्रभावी रूप से एक दौड़ होता है - हालांकि '/ usr/bin/tail' से पहले लिखने के लिए' file.txt' खोला जाता है, जिससे छंटनी जीतने की अत्यधिक संभावना होती है दौड़ (चूंकि '/ usr/bin/cat' को एक निष्पादन की आवश्यकता होती है, जिसमें सभी लिंकर/लोडर प्रदर्शन प्रभाव का तात्पर्य है)। –
tmp = $ (tail -1 file.txt); echo $ tmp> file.txt;
अस्थायी रूप से अस्थायी फ़ाइलों से बचाता है, लेकिन सभी मानक मुद्दों से बचने के लिए इसे उद्धृत किया जाना चाहिए। – wnoise
आप एसईडी सभी लाइनों को हटाने के लिए एक फ़ाइल से पिछले उपयोग कर सकते हैं लेकिन:
sed -i '$!d' file
इस बारे में एकमात्र दुर्भाग्यपूर्ण बात यह है कि 'sed -i' POSIX-compliant नहीं है लेकिन एक जीएनयू एक्सटेंशन है। जगह में संपादित करने के लिए 'ed' या' ex' का उपयोग करना उन चेतावनियों से बच जाएगा। –
यह एक लिनक्स खोल में अच्छी तरह से काम करता है:
replace_with_filter() {
local filename="$1"; shift
local dd_output byte_count filter_status dd_status
dd_output=$("[email protected]" <"$filename" | dd conv=notrunc of="$filename" 2>&1; echo "${PIPESTATUS[@]}")
{ read; read; read -r byte_count _; read filter_status dd_status; } <<<"$dd_output"
((filter_status > 0)) && return "$filter_status"
((dd_status > 0)) && return "$dd_status"
dd bs=1 seek="$byte_count" if=/dev/null of="$filename"
}
replace_with_filter file.txt tail -1
dd
के "notrunc" विकल्प, वापस फ़िल्टर्ड सामग्री लिखने के लिए, जगह में, जबकि dd
फिर से (एक बाइट गिनती के साथ की जरूरत है प्रयोग किया जाता है) वास्तव में फ़ाइल को कम करने के लिए। यदि नया फ़ाइल आकार पुराने फ़ाइल आकार के बराबर या बराबर है, तो दूसरा dd
आमंत्रण आवश्यक नहीं है।
फ़ाइल कॉपी विधि पर इसका लाभ निम्न हैं: 1) कोई अतिरिक्त डिस्क स्थान आवश्यक नहीं है, 2) बड़ी फ़ाइलों पर तेज़ प्रदर्शन, और 3) शुद्ध खोल (डीडी के अलावा)।
मुझे यह पसंद है - यह एक अभिनव विचार है। मैं इस समय केवल इसे ऊपर नहीं उठा रहा हूं क्योंकि '$ FILTER' का उपयोग http://mywiki.wooledge.org/BashFAQ/050 से गुजरता है (स्ट्रिंग-स्प्लिटिंग पर भरोसा करने के लिए सही ढंग से कमांड बनाने के लिए अत्यंत त्रुटि-प्रवण है)। –
मुझे आशा है कि आप मेरे भारी हाथों के संपादन को ध्यान में रखें; यह समाधान अब काफी मजबूत होना चाहिए। –
मैं इसे 'replace_with_filter' नहीं कहूंगा। यह केवल फ़िल्टर के लिए काम करेगा कि प्रत्येक अगले खंड के साथ डेटा मूल खंड से अधिक नहीं होता है। अन्यथा फ़िल्टर द्वारा पढ़ी गई जानकारी को इसके आउटपुट द्वारा ओवरराइट किया जा सकता है। ध्यान दें कि पिछले 'dd' के बजाय आप 'truncate -s $ byte_count" $ filename "' का उपयोग कर सकते हैं। – ony
बस इस मामले के लिए
cat < file.txt | (rm file.txt; tail -1 > file.txt)का उपयोग करना संभव है जो "बिल्ली" कनेक्शन को "(...)" में सबहेल के साथ "file" कनेक्शन से पहले "file.txt" खोल देगा। "आरएम file.txt" डिस्क से संदर्भ को हटा देगा इससे पहले कि सबहेल इसे "पूंछ" के लिए लिखने के लिए खोल देगा, लेकिन सामग्री अभी भी खुली डिस्क्रिप्टर के माध्यम से उपलब्ध होगी जो "बिल्ली" को तब तक पारित कर दी जाएगी जब तक कि यह stdin बंद न हो जाए। वहाँ पाइपलाइन घटकों 'स्टार्टअप के बीच आदेश देने की कोई गारंटी नहीं है: तो आप बेहतर सुनिश्चित करें कि इस आदेश को खत्म हो जाएगा या "file.txt" की सामग्री को
बेनामी डाउनवॉटर को दंडित किया जाना चाहिए। यह उत्तर काम करता है और इसमें तर्क है (यानी सामग्री तक पहुंच रखने के लिए इसे पढ़ने के लिए खोलने के बाद फ़ाइल हटाएं)। – ony
मुझे (अनुमानित) प्रस्ताव के साथ असहमत होने में परेशानी है कि एक समाधान जिसमें विफलता के मामलों में डेटा हानि होती है, उससे बचा जाना चाहिए, भले ही उस चेतावनी को स्पष्ट रूप से लेबल किया गया हो; एक अनावश्यक सबहेल केक पर टुकड़ा कर रहा है। और "दंडित"? वास्तव में? –
@ चार्ल्सडफी, "शर्मिंदा दंडित" अज्ञात के लिए है। क्योंकि किसी भी तर्क के बिना डाउनवोट होने से परेशान होता है। आपको ऐसा नहीं लगता? .. शुद्धता के लिए, क्या आप सुनिश्चित हैं कि उच्च उत्तर और विश्वसनीयता - सही उत्तर के लिए आवश्यक है? कभी-कभी आप कुछ और संग्रह करने के लिए कुछ बलिदान कर सकते हैं। इस प्रकार विभिन्न विचारों से आप अलग-अलग समाधानों का इलाज कर सकते हैं। यदि आप मेरे विचार को मेरा उत्तर देते हैं तो आप पाएंगे कि इस उत्तर के लिए पूर्व शर्त पूरी नहीं की जा सकती है, इसलिए यह गलत तर्क है कि यह गलत है। – ony
echo "$(tail -1 file.txt)" > file.txt
क्या होता है यदि अंतिम पंक्ति में केवल '-n' है? क्या होगा यदि इसमें बैकस्लैश अक्षर शामिल हैं और आपका 'इको' एक्सएसआई पॉज़िक्स एक्सटेंशन के अनुरूप है (जो किसी भी भागने के लिए कॉल करता है तो उन अक्षरों को '-ई' जैसी किसी भी चीज़ के बिना भी सम्मानित किया जा सकता है)? प्रतिध्वनि पर भरोसा करने के बजाय 'printf '% s \ n'" $ (tail -1 file.txt) "का उपयोग करना सुरक्षित होगा –
ध्यान दें कि पूंछ एक फ़ाइल नाम को तर्क के रूप में स्वीकार करता है: "tail -1 file.txt> file.txt.new && mv file.txt.new file.txt" –
@ मार्सेल लेवी - बिल्कुल सही, और इसमें चलाने की क्षमता है उस तरह से अधिक कुशलता से; अपडेट किया गया। –
प्रिय @ चार्ल्सडफी, मैं यह ध्यान रखना चाहता हूं कि आपके "सुरक्षित" संस्करण की आवश्यकता है कि उपयोगकर्ता ऐसा कर रहा है जो रूट होगा। यह कैसे सुरक्षित है, मैं कह सकता हूं कि "file.txt" तक पहुंच पर इसकी अनुमतियां और मालिक लागू करने के लिए पर्याप्त नहीं है। उस पर विचार करें "फाइल।txt "की अनुमति 0666 है और इसे" ~/.ssh "फ़ोल्डर के अंदर रखा गया है जिसमें 0700 की अनुमति है। इस मामले में मेरा समाधान अधिक सुरक्षित होगा क्योंकि यह उस फ़ाइल को दुनिया में प्रकट नहीं करेगा। क्या मुझे इसके कारण आपके जवाब को कम करना चाहिए ?;) – ony