जैसा कि @samiam ने कहा है कि सूची आपके माध्यम से एक अर्ध-यादृच्छिक क्रम में वापस आ गई है readdir()
। मैं सिर्फ निम्नलिखित जोड़ूंगा।
सूची वापस आ गई है जिसे मैं निर्देशिका ऑर्डर कहूंगा। पुराने फाइल सिस्टम पर, ऑर्डर अक्सर निर्माण ऑर्डर होता है जो डायरेक्टरी टेबल में फाइल एंट्रीज को जोड़ा जाता है। निश्चित रूप से इसके लिए एक चेतावनी है, जब एक निर्देशिका प्रविष्टि को हटा दिया जाता है, तो इस प्रविष्टि को पुनर्नवीनीकरण किया जाता है, इसलिए बाद में संग्रहित किसी भी फ़ाइल को पिछली प्रविष्टि को बदल दिया जाएगा, इसलिए आदेश अब केवल सृजन समय पर आधारित नहीं होगा।
आधुनिक फाइल सिस्टम पर, जहां निर्देशिका डेटा संरचनाएं खोज ट्री या हैश टेबल पर आधारित होती हैं, ऑर्डर व्यावहारिक रूप से अप्रत्याशित है।
उदाहरण
जब आप अपना टच कमांड चलाते हैं, तो बनाई गई फाइलों को देखकर पता चलता है कि निम्नलिखित इनोड्स असाइन किए गए थे।
$ touch dir/{{1..8},{a..p}}
$ stat --printf="%n -- %i\n" dir/*
dir/1 -- 10883235
dir/2 -- 10883236
dir/3 -- 10883242
dir/4 -- 10883243
dir/5 -- 10883244
dir/6 -- 10883245
dir/7 -- 10883246
dir/8 -- 10883247
dir/a -- 10883248
dir/b -- 10883249
dir/c -- 10883250
dir/d -- 10883251
dir/e -- 10883252
dir/f -- 10883253
dir/g -- 10883254
dir/h -- 10883255
dir/i -- 10883256
dir/j -- 10883299
dir/k -- 10883302
dir/l -- 10883303
dir/m -- 10883311
dir/n -- 10883424
dir/o -- 10883426
dir/p -- 10883427
इसलिए हम देख सकते हैं कि स्पर्श द्वारा उपयोग किए गए ब्रेस विस्तार वर्णमाला को वर्णानुक्रम में बनाते हैं और इसलिए जब उन्हें एचडीडी को लिखा जाता है तो उन्हें अनुक्रमिक इनोड संख्या सौंपी जाती है। (हालांकि यह निर्देशिका में आदेश को प्रभावित नहीं करता है।)
अपने tar
आदेश को कई बार चलाने से यह प्रतीत होता है कि सूची में एक आदेश है, क्योंकि इसे कई बार चलाने से हर बार एक ही सूची मिलती है। यहाँ मैंने इसे 100 बार चलाया है और फिर रनों की तुलना की है और वे सभी समान हैं।
$ for i in {1..100};do tar cJvf file.tar.xz dir/ > run${i};done
$ for i in {1..100};do cmp run1 run${i};done
$
यदि हम रणनीतिक रूप से कहते हैं dir/e
और फिर एक नई फ़ाइल जोड़ते हैं, तो dir/ee
हम देख सकते हैं कि इस नई फ़ाइल ने dir/e
निर्देशिका प्रविष्टियों की तालिका में पहले स्थान पर कब्जा कर लिया है।
$ rm dir/e
$ touch dir/ee
अब आउटपुट को for
ऊपर दिए गए लूप में से केवल 1 से रखें।
$ mv run1 r1A
अब अगर हम for
लूप को फिर से चलाते हैं जो tar
कमांड को 100 बार फिर से चलाएगा, और पिछले एक के साथ इस दूसरे रन की तुलना करें:
$ sdiff r1A run1
dir/ dir/
...
dir/c dir/c
dir/f dir/f
dir/e | dir/ee
dir/o dir/o
dir/2 dir/2
...
हमने देखा कि निर्देशिका तालिका में जगह dir/ee
ले ली है dir/e
।
stat --printf='%i\t-- %n\n' * | sort -n | sed 's/.*\t-- //'