एक और समस्या के समाधान के साथ सभी का फिर से स्वागत है! आज, हम त्रिकोणीय संख्याओं और उनके विभाजकों के बारे में बात करेंगे: विशेष रूप से विशाल संख्याओं के बारे में! जबकि हम प्रकृति में त्रिकोणीय संख्याओं के मेरे शानदार कलात्मक प्रतिनिधित्व की प्रशंसा कर सकते हैं, आइए हमारे सामान्य अस्वीकरण रखें: यह समस्या अद्भुत वेबसाइट द्वारा प्रदान की गई है , जिसे आप सब्सक्राइब कर सकते हैं ! हल करने के लिए 800 से अधिक गणित और कोडिंग समस्याएं हैं और उनके बारे में चर्चाओं का एक विशाल संग्रह है। यह सचमुच अपना दिमाग घुमाने और कुछ नया सीखने के लिए एकदम सही उपकरण है! इसे जांचें और अपनी चुनौतियों को भी हल करने का प्रयास करें! ProjectEuler.net यहाँ मैं कोई विशेषज्ञ प्रोग्रामर नहीं हूं: बस एक व्यक्ति हूं जिसे इस तरह की चीज़ों के बारे में लिखना अच्छा लगता है और अपने शॉट्स साझा करना पसंद करता है। मुझे यकीन है कि यह इष्टतम समाधान नहीं होगा, इसलिए यदि आपके पास कोई बेहतर समाधान है या इसे अनुकूलित करने के बारे में कोई विचार है तो यदि आप साझा करना चाहते हैं तो आपका हार्दिक स्वागत है! बहुत हो गई बात, आइए देखें आज की समस्या क्या पेश करती है: त्रिभुज संख्याओं का क्रम प्राकृतिक संख्याओं को जोड़कर उत्पन्न किया जाता है। तो 7वीं त्रिभुज संख्या 1+2+3+4+5+6+7=28 होगी। पहले दस पद होंगे: 1,3,6,10,15,21,28,36,45,55,… आइए पहले सात त्रिभुज संख्याओं के गुणनखंडों को सूचीबद्ध करें: हम देख सकते हैं कि 28 पाँच से अधिक भाजक वाली पहली त्रिभुज संख्या है। पाँच सौ से अधिक भाजक वाली पहली त्रिभुज संख्या का मान क्या है? समस्या बिल्कुल सीधी है: हमें यह समझने के लिए कहा गया है कि पहली क्या है जिसमें 500 से अधिक हैं। सबसे पहले चीज़ें, आइए बेहतर ढंग से देखें कि त्रिकोणीय संख्या क्या है और विभाजक क्या है। त्रिकोणीय संख्या भाजक त्रिकोणीय संख्याएँ त्रिकोणीय संख्याएँ संख्याओं का एक विशेष उपसमूह होती हैं जो एक निश्चित तक की सभी प्राकृतिक संख्याओं के योग से बनती हैं। उन्हें त्रिकोणीय कहा जाता है क्योंकि । यहां कुछ उदाहरण दिए गए हैं: आप उन्हें हमेशा त्रिकोण के रूप में व्यवस्थित कर सकते हैं ऊपर की छवि में, यह तीसरी, चौथी और पांचवीं त्रिकोणीय संख्या है। इसलिए, उदाहरण के लिए, तीसरी संख्या 1+2+3 = 6 के रूप में बनती है, चौथी संख्या 1+2+3+4 = 10 के रूप में बनती है, और इसी तरह। आम तौर पर कहें तो, nᵗʰ त्रिकोणीय संख्या, Tₙ, के बराबर है निस्संदेह, यह गणित की सबसे प्रसिद्ध श्रृंखलाओं में से एक है, जिसे ने भी प्रस्तुत किया था, जिन्होंने कहा था कि लगातार पूर्णांकों का योग बराबर होता है: गॉस इसलिए मूल रूप से, उदाहरण के लिए, यदि हम तीसरी त्रिकोणीय संख्या की गणना करना चाहते हैं, तो हम बस 3*4/2 = 6 की गणना करते हैं। जब हम अपना समाधान लिखना शुरू करेंगे तो यह बहुत मददगार होगा! विभाजक किसी संख्या के (या ) की परिभाषा देना वास्तव में सरल है: यह उदाहरण के लिए, 3, 18 का भाजक है, क्योंकि हम 3 और 6 को गुणा करके पुनः 18 प्राप्त कर सकते हैं। n विभाजक कारक एक संख्या है जिसे फिर से प्राप्त करने के लिए किसी अन्य पूर्णांक से गुणा किया जा सकता है। j n k हालाँकि, 5, 18 का भाजक नहीं है, क्योंकि ऐसी कोई संख्या नहीं है जिसे 5 से गुणा करने पर 18 प्राप्त हो। k परिभाषा के अनुसार, हमें एक महत्वपूर्ण गुण भी प्राप्त होता है: यदि , का भाजक है क्योंकि n प्राप्त करने के लिए इसे से गुणा किया जा सकता है तो भी का भाजक है क्योंकि प्राप्त करने के लिए इसे से गुणा किया जा सकता है। j n k , k n n j (वास्तव में, हमेशा 1 हो सकता है और हो सकता है)। इस तरह हम यह सुनिश्चित कर सकते हैं कि संख्या में हमेशा कम से कम दो भाजक और होते हैं n j k j k n इसके अलावा, इसे दूसरे तरीके से कहें तो, तो, उदाहरण के लिए, 18/3 = 6, और शेषफल 0 है। एक संख्या संख्या का भाजक है यदि का शेषफल 0 के बराबर है। j, n n/j हम के साथ इस अवधारणा को बेहतर ढंग से औपचारिक रूप दे सकते हैं कि , का भाजक है यदि दूसरे शब्दों में, हम यह दूसरी संपत्ति प्राप्त करते हैं: मॉड्यूलर अंकगणित j n n % j = 0 है। तीसरी और आखिरी संपत्ति जिसमें हम रुचि रखते हैं वह यह है कि इसे समझना बहुत आसान है. पहली संपत्ति से, हम जानते हैं कि कारक किसी तरह 1,…,n की सीमा में एक साथ "जुड़े" हैं। किसी संख्या का बजाय के। n n/2 से बड़ा कोई भाजक नहीं हो सकता है, n ऐसा इसलिए है क्योंकि फिर से, यदि तो इसका कारण यह है इसलिए, आइए 18 को फिर से एक उदाहरण के रूप में लें, इसके विभाजक खोजें, और इस गुण को प्रतिबिंबित करने के लिए उन्हें रंग दें। j\n, कि j * k = n. k\n भी। इसलिए, उदाहरण के लिए, यदि और इसके विपरीत यदि तो इसका मतलब है कि 1 के अलावा, हम जिस सबसे छोटे उपयोग कर सकते हैं, वह 2 है, जो हमें सबसे बड़ा देता है। संभव, (हमारे मामले में 9) यह सही है: j = 3, k = 6, j = 6, k = 3, j का k n/2 । सम संख्याओं के लिए, इस स्थिति में सबसे बड़ा गुणनखंड बिल्कुल n के आधे के बराबर होगा; विषम संख्याओं के लिए: यदि हम 2 से विभाजित नहीं कर सकते (क्योंकि विषम होने पर हमें एक तर्कसंगत संख्या मिलेगी); इसका मतलब है कि हम केवल जो हमें हमेशा n को j > 2 का उपयोग कर सकते हैं, k < n/2 परिणाम देगा। अंततः, उदाहरण के लिए, जब एक संभावित कारक एक अन्य कारक उत्पन्न करेगा। हम इस मामले के बारे में बाद में कोड भाग में अधिक चर्चा करेंगे। केवल एक ही मामला है जिसमें और बराबर हो सकते हैं: जब एक वर्ग संख्या है। j k n n = 36, j = 6 k = 6 बेशक, ये अवधारणाएँ बहुत तुच्छ हैं, लेकिन वे हमारे समाधान में बहुत महत्वपूर्ण होंगी! कोड कोड में लिखा जाएगा, क्योंकि यह वास्तव में एक तेज़ भाषा है जो अभी भी पठनीयता का एक बड़ा स्तर बनाए रखती है। हम समाधान को दो भागों में विभाजित करेंगे: पहले, हम गो किसी संख्या के विभाजक खोजने के लिए एक फ़ंक्शन बनाएंगे, और फिर हम इसे त्रिकोणीय संख्याओं पर लागू करेंगे। आइए पहले फ़ंक्शन देखें: https://gist.github.com/NicolaM94/8fb97503da0c4d4431fcd5a858d13cdb?embedable=true हम अपना फ़ंक्शन बनाते हैं जो एक पूर्णांक स्वीकार करता है और पूर्णांकों की एक सरणी है, जिसमें हमारे विभाजक शामिल होंगे। उसके बाद, हम तुच्छ कारकों को हैं, अर्थात् 1 और । n out out n फिर हम 2 से तक लूप करना शुरू करते हैं (दूसरी संपत्ति से; यह भी ध्यान दें कि हमें में कोई दिलचस्पी नहीं है क्योंकि यदि सम है, तो द्वारा भाजक में जोड़ा जाएगा पुनरावृत्ति: यही कारण है कि हम पर रुकते हैं न कि )। j n/2 n/2 n j = 2 k = n/2 j = 2 j<n/2 j≤ n/2 इस तरह, हम केवल के पहले भाग में विभाजकों की जांच कर सकते हैं, जिससे प्रक्रिया काफी तेज हो जाएगी। n प्रत्येक लूप में, हम 3 चीज़ें जाँचते हैं: तीसरा if कथन वास्तव में जाँच करता है कि क्या , दूसरे शब्दों में, यदि 0 ≡ n (mod j)। यदि ऐसा है, तो हमें एक कारक मिल गया है, और हम सूची में जोड़ते हैं। हम गणना करके और फिर अगले पर जाकर इसके समकक्ष भी सूची में जोड़ सकते हैं; n%j==0 j n/j j k को दूसरा यदि कथन जाँचता है कि क्या एक वर्ग है, और इसलिए का मूल है। यदि ऐसा है, तो डुप्लिकेट से बचने के लिए, में केवल एक विभाजक जोड़ा जाएगा, और फिर एल्गोरिदम आगे बढ़ता है। n j n out j मान लीजिए : यदि यह छोटा ब्लॉक गायब होगा, तो तीसरा if स्टेटमेंट ट्रिगर हो जाएगा, और प्राप्त होंगे, इस प्रकार, एक डुप्लिकेट बनाया जाएगा। n = 36 out j = 6 k = n/j = 36/6 = 6 पहला if कथन सबसे जटिल है: इसका उद्देश्य यह जांचना है कि क्या हम दोहराए जाने वाले जोड़े बनाना शुरू कर रहे हैं। यदि ऐसा है, तो हम तुरंत लूप को तोड़ सकते हैं, क्योंकि हमारा कारक पहले से ही मौजूद होगा। jk out विशेष रूप से इस तीसरे बिंदु के बारे में, जाँच करने के लिए दो मामले हैं: क्या या out[len(out)-1] == j out[len(out)-2] == j पहला मामला क्लासिक है: उदाहरण के लिए संख्या T₇ = 28 लें: जब , यह डाले गए अंतिम मान के बराबर होगा। इसलिए, हम लूप को तोड़ सकते हैं। j = 7 दूसरा मामला तभी घटित होता है जब हमें एक वर्ग मिलता है। उदाहरण के लिए, 36 को फिर से लें: n जब , यह के वर्गमूल से पहले जोड़े गए अंतिम मान के बराबर होगा, जो केवल एक बार दिखाई देता है। इसलिए, इस बिंदु पर, हम लूप को तोड़ सकते हैं। j = 9 n अंततः, हम आसानी से अपने विभाजक प्राप्त करने के लिए सकते हैं। return out परिणाम लागू करना अब चूँकि हमारे पास अपना कार्य है, हम इसे प्रत्येक त्रिकोणीय संख्या पर लागू कर सकते हैं। हम एक काउंटर बनाने जा रहे हैं जो हमारे त्रिकोणीय संख्याओं के लिए जनरेटर के रूप में काम करेगा। हम गॉस समीकरण के साथ संबंधित त्रिकोणीय संख्या की गणना करते हैं और जांचते हैं कि इसमें कितने विभाजक हैं। c tn यदि यह 500 से अधिक है, तो हम परिणाम के रूप में वह लौटा देते हैं। tn https://gist.github.com/NicolaM94/b01fb72874cb903982b887846fcc5c0c?embedable=true#file-main-go लेकिन...वहाँ एक पकड़ है! गणित की पहेलियों और समस्याओं को प्रस्तुत करने के अलावा, ProjectEuler.net वास्तव में एक प्यारा प्रोजेक्ट है: उनका मुख्य ध्यान गणित के बारे में सीखना, सोचना और तर्क करना शुरू करने के लिए एक उपकरण प्रदान करना है। और मुझे यह पसंद है: वे इतने प्रतिबद्ध हैं कि उनकी समस्याओं के लिए परिणाम और मार्गदर्शिकाएँ प्रकाशित करना वास्तव में निषिद्ध है (यह पहली 100 समस्याओं में से एक है, इसलिए मैं समझता हूँ कि इसकी अनुमति है)। इसे देखते हुए, इस कारण से, मैं आपको वास्तविक समाधान नहीं दे रहा हूं: इसे स्वयं आज़माएं और मेरे से बेहतर एल्गोरिदम के साथ आने का प्रयास करें (ऐसे बहुत सारे हैं)। क्षमा करें दोस्तों! 😅 मुझे नहीं लगता कि केवल कॉपी-पेस्ट करने और उपलब्धि हासिल करने का समाधान देना उचित होगा। समय की जटिलता मुझे विश्वास है कि कई बेहतर समाधान हैं क्योंकि मुझे लगता है कि यह एक बहुत ही भयानक शॉट है! फ़ंक्शन रैखिक समय में चलता है: चूंकि यह उदाहरण के आकार पर निर्भर करता है, और यह अधिकतम लूप पर भी चलता है; हम इसे O(n) मान सकते हैं। FindDivisor n n/2 हालाँकि, हमें पहले प्रत्येक त्रिकोणीय संख्या की गणना करनी चाहिए, जिसमें O(tn) समय भी लगता है, जहाँ tn परिणाम है (अंतिम जिसकी हम वास्तव में गणना करते हैं)। इसे देखते हुए, लगभग पूरे एल्गोरिदम की समय जटिलता O(tn*n) होनी चाहिए। चूँकि तर्क या हमारा फ़ंक्शन बन जाता है, और हम इसके अंदर अधिकतम लूप चलाते हैं, हम समय जटिलता को O(tn*tn/2) = O(tn²/2) = O(tn²) के रूप में फिर से लिख सकते हैं। तो एक , दुर्भाग्य से! tn n/2 द्विघात समय समाधान हालाँकि मुझे आश्चर्य हुआ कि भले ही एल्गोरिथ्म उस तरह की जटिलता के बारे में है, फिर भी यह बहुत तेज़ चलता है। मेरी मशीन पर परिणाम देने में (AMD Ryzen 5, औसत ck. 2700 MHz) 322.564227 एमएस लगा। 1000 विभाजकों से अधिक पहली त्रिकोणीय संख्या को खोजने का प्रयास करने में 3.827144472 सेकंड लगे, इसलिए यह स्पष्ट रूप से दिखाई दे रहा है कि समय की खपत तेजी से बढ़ रही है। निष्कर्ष आख़िरकार हमने इसे बना लिया! मुझे आशा है कि आप लोगों को लेख पसंद आया होगा, और मुझे आशा है कि आप इससे कुछ दिलचस्प सीख सकते हैं: यदि हां, तो कृपया एक या दो तालियाँ छोड़ें और लेख को किसी ऐसे व्यक्ति के साथ साझा करें जिसकी इस विषय में रुचि हो! शानदार काम के लिए प्रोजेक्टयूलर को फिर से एक बड़ा धन्यवाद: मैं वास्तव में आपको सुझाव देना चाहता हूं कि आप जाएं और देखें कि वे क्या पेशकश कर सकते हैं; मुझे यकीन है कि आपको यह बेहद दिलचस्प लगेगा! और हमेशा की तरह, पढ़ने के लिए धन्यवाद। निकॉला