2017-09-12 10 views
5

क्या कोई व्यक्ति नाम-पैरामीटर => T और Function0 पैरामीटर () => T स्केल कंपाइलर द्वारा एक दूसरे में परिवर्तित हो सकता है, इस पर कोई निश्चित उत्तर दे सकता है? मुझे पता है कि वे समान नहीं हैं, लेकिन अंतर बहुत सूक्ष्म है क्योंकि इन्हें कई परिदृश्यों में एक दूसरे के रूप में इस्तेमाल किया जा सकता है।स्कैला: फंक्शन 0 बनाम नाम पैरामीटर

उदाहरण: अगर मैं

def someFunction: Int = 2 
def f(x: => Int): Unit = println(x) 

को परिभाषित तो मैं सफलतापूर्वक कॉल कर सकते हैं

f(2) 
f(someFunction) 

() => Int=> Int के लिए एक स्वीकार्य प्रतिस्थापन कैसी है?

अधिक आम तौर पर () => T=> T पैरामीटर के लिए एक सार्वभौमिक रूप से स्वीकार्य प्रतिस्थापन है?

इसके अलावा, मुझे सही करें अगर मैं निम्नलिखित तर्क पर गलत हूँ: क्योंकि पहले एक मान प्रकार (T) है => T() => T के लिए एक स्वीकार्य प्रतिस्थापन नहीं है, अन्य एक समारोह प्रकार है। यही है, अगर मेरे पास def f(x:() => Int) है, तो मैं कभी भी Int, या आलसी Int पास नहीं कर पाऊंगा (वैसे भी समझ में नहीं आता है क्योंकि कोई आलसी प्रकार नहीं हैं)।

+5

आपके उदाहरण निश्चित रूप से * समकक्ष नहीं हैं। पहला व्यक्ति 'कोशिश करें [() ⇒ Int] ', दूसरा' प्रयास करें [int]'। प्रकार '() ⇒ Int' के आपके फ़ंक्शन तर्क को' ⇒ टी 'में 'टी'' (⇒ Int 'के साथ परिवर्तित किया जाता है। –

+0

यहां कुछ अच्छी जानकारी है: https://tpolecat.github.io/2014/06/26/call-by-name.html – nevets1219

+0

वे एक-दूसरे को प्रतिस्थापित नहीं करते हैं और वे स्कैला कंपाइलर द्वारा एक-दूसरे में परिवर्तित नहीं होते हैं। वे बस अलग-अलग चीजें हैं और आमतौर पर विभिन्न उद्देश्यों की पूर्ति करते हैं। –

उत्तर

3

ठीक है, यहां पूर्ण टूटना है।

def value: Int = ??? 
def method(): Int = ??? 

def f1(f:() => Int) = ??? 
def f2(f: => Int) = ??? 

f1(value) // fails 
f1(method) // works 
f2(value) // works 
f2(method) // works with a warning "empty-paren method accessed as parameterless" 
  1. f1 (मूल्य)

यह एक विफल रहता है क्योंकि f1 एक यूनिट => इंट समारोह उम्मीद कर रही है, लेकिन एक इंट मूल्य दिया जाता है।

  1. f1 (विधि)

यह एक काम करता है क्योंकि f1 एक समारोह की उम्मीद कर रहा है, और एक विधि दी गई है। यहां अंतर है: विधि स्कैला में कोई मान नहीं है; यह अपने आप पर मौजूद नहीं हो सकता है और संदर्भ में एक विशेषता है जिसे परिभाषित किया गया है (वर्ग, विशेषता, वस्तु आदि)। कार्य एक मूल्य है; इसे एक संग्रह में रखा जा सकता है, जिसे किसी अन्य फ़ंक्शन में तर्क के रूप में लिया जाता है, फ़ंक्शन इत्यादि से लौटाया जाता है। जब कंपाइलर फ़ंक्शन की अपेक्षा कर रहा है और उसे विधि दी जाती है, तो यह ईटा विस्तार करता है। एक समारोह को देखते हुए उदा। f: (a: Int, b: Int) => Int, ईटा विस्तार हस्ताक्षर को संरक्षित करते समय चारों ओर एक और परत के निर्माण की प्रक्रिया है, इसलिए यह (a: Int, b: Int) => f(a, b) बन जाता है। यह तकनीक उपयोगी है क्योंकि हम तरीकों को कार्य में बदल सकते हैं। कुछ विधि def f(a: Int): Int = ??? को देखते हुए, हम int => int out of: (a: Int) => f(a) के फ़ंक्शन को बनाने के लिए ईटा-विस्तार कर सकते हैं। यदि आप रुचि रखते हैं तो मैंने कुछ समय पहले इस बारे में blog post लिखा था।

  1. f2 (मूल्य)

वर्क्स आश्चर्य के बिना, लेकिन तथ्य यह है कि भेजे गए मान को हर बार यह प्रयोग किया जाता है पहुँचा जा सकता है पर ध्यान देना समारोह शरीर

  1. f2 (विधि)

काम करता है, लेकिन एक चेतावनी है कि हम एक विधि लागू कर रहे हैं कि खाली कोष्ठक के साथ का उपयोग करके परिभाषित किया गया है के साथ कोई कोष्ठक। अच्छा अभ्यास ब्रांड्स के बिना विधियों का उपयोग करना है (उदा। f) जब वे केवल एक मान का प्रतिनिधित्व करते हैं, लेकिन प्रत्येक बार जब इसे एक्सेस किया जाता है, तो पुन: गणना की जाती है, उदा। numberOfUpvotes, और खाली कोष्ठक के साथ विधियों का उपयोग करने के लिए (उदा। f()) जब किसी प्रकार का साइड-इफेक्ट किया जाता है और इसलिए विधि idempotent नहीं है, उदा। createSnapshot() (फिर से, यह पूरी तरह से कार्यात्मक कोड में मौजूद नहीं होना चाहिए)।

सलाह का शब्द: अपने दिमाग को किसी चीज के बदले में न रखें। प्रतिस्थापन का उपयोग न करें। अगर किसी को किसी फ़ंक्शन की आवश्यकता होती है, तो उसे एक फ़ंक्शन प्रदान करें। यदि इसे एक मूल्य की आवश्यकता है, तो एक मूल्य प्रदान करें। यदि किसी विधि को माता-पिता के बिना परिभाषित किया गया है, तो इसे बिना माता-पिता के आह्वान करें। यदि उसके माता-पिता हैं, तो इसे माता-पिता के साथ बुलाओ।

यदि आपको विधि से कार्य करने की आवश्यकता है और कंपाइलर फ़ंक्शन की अपेक्षा कर रहा है, तो ईटा-विस्तार स्वचालित रूप से हो जाएगा। यदि यह किसी फ़ंक्शन की अपेक्षा नहीं कर रहा है, तो आपको इसे मैन्युअल रूप से करने की आवश्यकता है।

def f(): Int = ??? 
val a = f    // no function context; a is a string 
val b:() => Int = f // b is a function Unit => Int 
val c = f2 _   // c is a function Unit => Int 

अंतिम मामला आंशिक रूप से लागू फ़ंक्शन है। मुझे लगता है कि मैं अब बहुत व्यापक जा रहा हूं इसलिए मैं यहां रुक जाऊंगा। मुझे आशा है कि इससे मदद मिलेगी।

संबंधित मुद्दे