बिल्कुल नहीं ... मूल सवाल का जवाब है:
ImageNamed का उपयोग करते हुए UIImage बनाते समय NSBundle विधि pathForResource का उपयोग करके किसी छवि के पथ की खोज करना कितना आवश्यक है?
ज्यादा .. है कैसे सही Zoul से स्वीकार किए जाते हैं जवाब और रंगा से अन्य से एक हैं। निष्पक्ष होने के लिए: यदि आप एप्लिकेशन बंडल निर्देशिका संरचना, या (दुर्लभ) मामले के बारे में बात कर रहे हैं, तो वे सही हैं, जहां छवि एक्सकोड में "नीले" फ़ोल्डर में है (बाद में उस पर अधिक), लेकिन अधिकांश के लिए नहीं सामान्य मामलों
वैसे भी, एक सच्चे उत्तर पर।
सामान्य रूप से मुझे जवाब खोजने का प्रयास करते समय मुझे यह प्रश्न मिला। मुझे इस प्रश्न के दस्तावेज़ीकरण या अन्य उत्तरों को संतोषजनक नहीं मिला, इसलिए मैंने परीक्षण करने का फैसला किया।
मेरा परीक्षण विवरण नीचे दिया गया है, लेकिन मुझे यहां परिणामों को सारांशित करने दें। संक्षेप में, जब imageNamed का उपयोग कर: अपनी छवियों को लोड करने के लिए है, यह निर्भर करता है जहाँ आप उन्हें डाल:
अगर आपकी छवियों अपनी परियोजना की जड़ में हैं, भले ही एक विशुद्ध रूप से तार्किक Xcode समूह है, तो में आयोजित नहीं , आप को पथ के बारे में सोचने की आवश्यकता नहीं है: बस छवि का नाम।
अगर आपकी छवियों एक समूह है कि आपकी फ़ाइल प्रणाली में एक निर्देशिका से जुड़ा हुआ है में कर रहे हैं, के माध्यम से "जोड़ा फ़ोल्डरों के लिए समूह बना" तो आप अभी भी नाम के बारे में चिंता करने की जरूरत नहीं है।
यदि आपकी छवियां "नीले" समूह में हैं, जो आपके फ़ाइल सिस्टम में "अतिरिक्त फ़ोल्डरों के लिए फ़ोल्डर संदर्भ बनाकर" निर्देशिका के माध्यम से एक निर्देशिका से जुड़ी हुई है, तो आप इसे छवि के साथ लोड कर सकते हैं नामित: एक सापेक्ष पथ निर्दिष्ट करके उपरोक्त स्वीकृत उत्तर द्वारा सुझाया गया (संयोग से?)।
अगर आप imageNamed :, imageWithContentsOfFile करने के लिए मुख्य विकल्प का उपयोग :, आप वास्तव में जो आप जानना चाहते हैं कि Xcode नाविक संरचना अपने बंडल में रास्तों में तब्दील हो की जरूरत का मतलब है बंडल पथ सहित फ़ाइल, का पूर्ण पथ की आवश्यकता क्यों है निर्देशिका संरचना।
इन दो तरीकों के बीच अन्य महत्वपूर्ण मतभेद:
- imageNamed की आवश्यकता नहीं है कि आप, जबकि imageWithContentsOfFile करता filetype विस्तार, तो केवल "आइकन" नहीं "icon.png" निर्दिष्ट पूर्ण फ़ाइल नाम की आवश्यकता होती है
- यह पहला बिंदु दूसरी सुविधा के साथ मदद करता है: imageNamed होगा स्वचालित रूप से लोड एक छवि की रेटिना संस्करण अगर वहाँ एक है, @ 2 जोड़कर एक्स आपके फ़ाइल नाम पर। तो यदि आप रेटिना डिस्प्ले पर "आइकन" मांगते हैं तो यह "[email protected]" लोड करने का प्रयास करेगा। imageWithContentsOfFile नहीं
- imageNamed छवि कैश करता है: और उसमें उसके चारों ओर विवाद का एक बहुत निहित है: तुम इतनी खोज या बड़े पैमाने पर वेब आप की सिफारिश आप इसे से बचना चाहिए क्योंकि यह नहीं करता है पदों की एक बहुत कुछ मिल जाएगा, तो टी अपने कैश को ठीक से साफ़ करें। हालांकि, यह साल पहले तय किया गया था, इसलिए आप को इसके कैश को साफ़ करने में विफल होने की चिंता करने की आवश्यकता नहीं है। आप अभी भी इस तथ्य के बारे में चिंता करने की ज़रूरत है कि यह कैश है, हालांकि। यदि आपकी छवियां बड़ी हैं और अक्सर लोड नहीं होती हैं, तो आप फ़ाइल से उन्हें लोड करके और उन्हें कैशिंग नहीं करके मेमोरी को सुरक्षित रखेंगे। यह लीक के साथ कुछ भी नहीं करना है: भले ही आपके पास रिसाव न हो, फिर भी आप डिवाइस पर सीमित स्मृति रखते हैं, और आप को अनावश्यक रूप से कैश नहीं करना चाहते हैं। यह क्लासिक कैशिंग ट्रेडऑफ है: आपकी स्थिति में क्या अधिक महत्वपूर्ण है? मेमोरी प्रदर्शन या सीपीयू प्रदर्शन (समय)।
तो मेरे परीक्षणों के साथ।
मैंने जो किया वह 3 सरल आइकन फ़ाइलों के साथ एक साधारण UITableView ऐप बना रहा था, जो विभिन्न तरीकों का उपयोग करके तालिका की पंक्तियों में दिखाया गया था। आइकन एक्सकोड प्रोजेक्ट संरचना में उनके स्थान में भिन्न हैं। एक्सकोड पर जोर दें। मूल प्रश्न के उत्तर को समझने की कुंजी यह है कि आईओएस ऐप में तीन पूरी तरह से अलग परियोजना निर्देशिका संरचनाएं हैं: एक्सकोड नेविगेटर में आप जो देखते हैं, वही प्रोजेक्ट के लिए फाइल सिस्टम पर एक जिसे आप खोजक में देखते हैं (एक्सकोड नेविगेटर में किसी भी आइटम पर राइट क्लिक करें और "फाइंडर में दिखाएं" चुनें) और, जिसे आप शायद ही कभी देखते हैं, तैनात ऐप की "बंडल" निर्देशिका संरचना। आप फ़ाइंडर में यह अंतिम भी देख सकते हैं - ~ / लाइब्रेरी/एप्लिकेशन सपोर्ट/आईफोन सिम्युलेटर में अपना ऐप ढूंढकर, और .app निर्देशिका में ड्रिलिंग करके। मैं आपको एक मिनट में एक तस्वीर दिखाऊंगा।
तो मेरे एप्लिकेशन में, मैं सभी तीन आइकन png छवि फ़ाइलों Xcode में अलग अलग तरीकों से घसीटा:
icon1.png (एक घड़ी), मैं Xcode के मार्ग पर एक फ़ाइल के रूप में में घसीटा प्रोजेक्ट, तो मैं बाद में ने एक्सकोड में एक नया समूह बनाया और इसे में खींच लिया। इस समूह को फ़ाइल सिस्टम में किसी भी निर्देशिका द्वारा प्रदर्शित नहीं किया गया है: यह एक शुद्ध एक्सकोड समूह है। इसलिए यह नाम है: "JustGroup"
icon2.png (एक आँख), मैं मूल रूप से मेरी फाइल सिस्टम "RealDir" कहा जाता निर्देशिका में डाल दिया है, और मैं Xcode में इस पूरी निर्देशिका घसीटा, और जब पूछा, मैंने चुना "किसी भी जोड़े गए फ़ोल्डर के लिए समूह बनाएं" विकल्प। इसका मतलब है कि एक्सकोड में रियलडियर समूह वास्तविक निर्देशिका से जुड़ा हुआ है जिसे फाइल सिस्टम में रीयलडियर कहा जाता है (मेरी प्रोजेक्ट निर्देशिका में) और वह icon2.png वहां है।
icon3.png (एक लक्ष्य), मेरे पास एक अलग निर्देशिका में भी था, जिसे मैंने को एक्सकोड में भी खींच लिया था। केवल इस बार मैंने दूसरे रेडियो विकल्प को चुना "किसी भी जोड़े गए फ़ोल्डरों के लिए फ़ोल्डर संदर्भ बनाएं"। यह एक्सकोड में एक तथाकथित "नीला" समूह बनाता है। बाद में यह सब कुछ क्या है। मैं इस समूह (और निर्देशिका) "FolderReference" कहा जाता है
यहाँ विकल्प है कि Xcode आप देता है के एक शॉट है:
:
और यहाँ मेरी परियोजना संरचना Xcode पर कैसा होता है
अब, मेरे ऐप में, मैंने प्रत्येक आइकन को लोड करने के लिए दो विधियों का उपयोग किया: UIImage imageNamed: और UIImage imageWithContentsOfFile। मैंने प्रत्येक तालिका के शीर्षक के साथ अपनी तालिका में पंक्तियों का एक गुच्छा बनाया है जिसमें आइकन का नाम है: जस्ट ग्रुप, रीयलडियर या फ़ोल्डर रेफरेंस, साथ ही उपयोग की गई विधि का नाम: छवि नामित बनाम से फ़ाइल (जो मैं हूं imageWithContentsOfFile के संक्षेप के रूप में उपयोग करना)
सेल का विस्तार लेबल (शीर्षक के नीचे फेंटर टेक्स्ट) विधि को मैंने फ़ाइल या पथ नाम दिखाया है।
स्पष्ट होने के लिए, "सेफाइल" के मामले में, मैं आपके द्वारा देखे गए "रिश्तेदार" नाम पर बंडल पथ जोड़ रहा हूं। तो "fromFile" के लिए, मैं वास्तव में इस कोड का उपयोग कर रहा हूँ:
NSString *bundlePath = [[NSBundle mainBundle] bundlePath];
NSString *imagePath = [NSString stringWithFormat:@"%@/%@", bundlePath, filePath];
UIImage *image = [UIImage imageWithContentsOfFile:imagePath];
जहां "FilePath" पथ आप तालिका सेल विस्तार लेबल में देखते हैं। छवि के लिए नामित :, दूसरी तरफ, सेल विवरण में फ़ाइलपाथ शब्दशः पारित किया जाता है।
और पंक्ति छवि स्वाभाविक रूप से लोड की गई छवि है। इसलिए तालिका में पंक्तियों के लिए जिनके पास कोई छवि नहीं है, छवि लोडिंग विफल रही।
यहां, संक्षेप में, परिणाम हैं। यदि आप इस पोस्ट के कुछ भी नहीं पढ़ते हैं, तो कम से कम इस छवि पर एक नज़र आपको बताएगी कि आपको बस इतना ही पता होना चाहिए।
यहाँ आसानी से पच अंक में बुनियादी व्याख्या दी गई है:
के रूप में यह imageNamed आधिकारिक दस्तावेज में कहा गया है,: विधि लोड आवेदन बंडल से छवियों। इसका मतलब है कि आपको बंडल स्थान निर्दिष्ट करने की आवश्यकता नहीं है, केवल फ़ाइल का नाम। और फिर भी, फ़ाइल का मूल नाम। दस्तावेज यहां थोड़ा पतला है, इसे वास्तव में यह स्पष्ट करना चाहिए कि यह फ़ाइल बंडल रूट निर्देशिका में दिए गए फ़ाइल पथ सापेक्ष से छवि लोड करता है।
(यहां किकर है, इस पर ध्यान दें) जो बंडल निर्देशिका के बारे में नियम है, आपके नियोजित ऐप के बंडल में मूल निर्देशिका को संदर्भित करता है। यदि आप अन्वेषण करते हैं, तो इसका अर्थ है ".app" निर्देशिका में। यही कारण है कि नहीं एक ही Xcode नाविक में Xcode प्रोजेक्ट, की रूट निर्देशिका के रूप में है और न ही वह पाने वाले में Xcode परियोजना के रूट निर्देशिका के रूप में ही है
इस वजह से, जब के लिए अपने app की तैनाती डिवाइस (या सिम्युलेटर) सभी प्रोजेक्ट निर्देशिकाओं को समूहों द्वारा जोड़ा गया फ़ोल्डर्स "के लिए प्रतिनिधित्व किया गया है flattened हैं। यही है, निर्देशिका को नजरअंदाज कर दिया गया है, और इसकी सभी सामग्री बंडल की मूल निर्देशिका में अनजाने में डंप हो गई है। (मैं "अनजाने में" कहता हूं क्योंकि यदि अलग-अलग फ़ोल्डरों में एक ही नाम वाली फाइलें हैं, तो वे यहां टकराएंगे और आपको समस्याओं का समाधान करने में कोई मदद नहीं मिलेगी।) यह मेरे उदाहरण में रियलडियर का मामला है: में तैनात ऐप, RealDir अब मौजूद नहीं है, और icon2.png सामान्य जनसंख्या (डरावनी) के साथ मिश्रण करने के लिए छोड़ दिया गया है। यह लगभग बिना कहने के चला जाता है कि "जस्ट ग्रुप", पूरी तरह तार्किक एक्सकोड समूह को भी अनदेखा किया जाता है - यह कभी भी असली निर्देशिका नहीं था, वैसे भी एक्सकोड उपयोगकर्ता के लिए एक दृश्य सहायता - और icon1.png भी बंडल रूट में है।
यही कारण है कि imageNamed: icon2 लोड करने में सक्षम था।
और यह भी कारण है कि imageWithContentsOfFile "RealDir/image2.png" में यह पता लगाने के लिए सक्षम नहीं था: क्योंकि वहाँ तैनात एप्लिकेशन में कोई RealDir निर्देशिका है।
"नीले फ़ोल्डर", दूसरे हाथ पर, वह है, निर्देशिका द्वारा "जोड़ा फ़ोल्डरों के लिए फ़ोल्डर संदर्भ", एप्लिकेशन बंडल निर्देशिका संरचना में बनाए रखा वास्तव में प्रतिनिधित्व किया। यह स्पष्ट रूप से नीले फ़ोल्डर्स का बिंदु है: वे आपको आपके तैनात ऐप में निर्देशिका संरचना बनाने का एक तरीका देते हैं। मुझे इसके लिए मूल राइस डी'एट्रे का यकीन नहीं है, लेकिन एक अच्छा उपयोग केस है जहां आपके पास एक ही नाम के साथ संसाधन फ़ाइलों के वैकल्पिक संस्करणों वाली कई निर्देशिकाएं हैं और आप चाहते हैं कि आपका ऐप रनटाइम पर उनके बीच स्विच करने में सक्षम हो निर्देशिका बदलकर। वैसे भी, मेरे फ़ोल्डर संदर्भ में icon3.png, तैनात ऐप में मेरी फ़ोल्डर रिफरेंस निर्देशिका में बना रहा।
- यही कारण है कि imageNamed है: यह "icon3" के साथ नहीं मिल सकता है, लेकिन के साथ "FolderReference/icon3" यह मिल सकता है
- imageWithContentsOfFile यह भी FolderReference का उपयोग कर खोजने के लिए सक्षम था, लेकिन केवल जब संलग्न हो, ऊपर दिए गए कोड का उपयोग कर पूर्ण बंडल पथ को याद रखें। (यहां मुख्य अंतर: छवि नामित इस मामले में एक सापेक्ष पथ के साथ काम करता है, imageWithContentsOfFile हमेशा एक पूर्ण पथ के साथ काम करता है)।
स्पष्ट करने के लिए, यहाँ मेरी फ़ोल्डर संरचनाएँ हैं: अंत में,
और शायद:
तुम मेरे Xcode प्रोजेक्ट नाविक संरचना ऊपर देखा, यहां फाइल सिस्टम निर्देशिका इसे नीचे है सबसे महत्वपूर्ण बात यह है कि तैनात बंडल फ़ाइल सिस्टम निर्देशिका संरचना:
नोट: मुझे टी नहीं मिला मेरे मैक पर इस स्थान पर: आपको अपना एक ही स्थान पर मिल जाएगा - आपको यह पता लगाने के लिए थोड़ा सा खोजना पड़ सकता है कि कौन सी बदसूरत-GUID- नामित उपनिर्देशिका में आपका ऐप शामिल है।
~/Library/Application Support/iPhone Simulator/5.1/Applications/4EB386B2-CD7E-4590-9757-18DDDEE6AF4F/ImageLoadingTest.app
मुझे उम्मीद है कि इससे मदद मिलती है। परीक्षण, अन्वेषण और अंत में, का वर्णन करने से यह निश्चित रूप से मेरी मदद करता है।
[यहां] (http://stackoverflow.com/questions/924740/dispelling-the-uiimage-imagenamed-fud) आपके कोड में नामित छवि का उपयोग करने के बारे में एक SO पोस्ट है जिसे आप पढ़ना भी चाह सकते हैं। सीधे आपके प्रश्न से संबंधित नहीं है, लेकिन मुझे यह बहुत उपयोगी पाया। – zero0cool