खोज…


वाक्य - विन्यास

  • स्थूल नाम (पूर्व) ... अंत
  • बोली ... अंत
  • : (...)
  • $ x
  • Meta.quot (एक्स)
  • QuoteNode (एक्स)
  • esc (एक्स)

टिप्पणियों

जूलिया के मेटाप्रोग्रामिंग फीचर्स लिस्प जैसी भाषाओं से काफी प्रेरित हैं, और कुछ लिस्प बैकग्राउंड वाले लोगों से परिचित होंगे। Metaprogramming बहुत शक्तिशाली है। जब सही तरीके से उपयोग किया जाता है, तो यह अधिक संक्षिप्त और पठनीय कोड हो सकता है।

quote ... end quasiquote सिंटैक्स है। मूल्यांकन किए जाने के भीतर अभिव्यक्तियों के बजाय, वे बस पार्स हैं। quote ... end का मूल्य quote ... end अभिव्यक्ति परिणामस्वरूप सार सिंटैक्स ट्री (एएसटी) है।

The :(...) सिंटैक्स quote ... end समान है quote ... end सिंटैक्स quote ... end है, लेकिन यह अधिक हल्का होता है। यह वाक्यविन्यास quote ... end तुलना में अधिक संक्षिप्त है quote ... end

एक quasiquote अंदर, $ ऑपरेटर खास है और AST में अपने तर्क interpolates। तर्क को एक अभिव्यक्ति होने की उम्मीद है, जिसे सीधे एएसटी में उतारा गया है।

Meta.quot(x) फ़ंक्शन इसके तर्क को उद्धृत करता है। यह अक्सर प्रक्षेप के लिए $ का उपयोग करने के साथ संयोजन में उपयोगी होता है, क्योंकि यह भाव और प्रतीकों को एएसटी में शाब्दिक रूप से उतारा जाता है।

@Show मैक्रो को लागू करना

जूलिया में, @show मैक्रो अक्सर डिबगिंग उद्देश्यों के लिए उपयोगी है। यह मूल्यांकन किए जाने वाले अभिव्यक्ति और उसके परिणाम दोनों को प्रदर्शित करता है, अंत में परिणाम का मूल्य लौटाता है:

julia> @show 1 + 1
1 + 1 = 2
2

यह @show का अपना संस्करण बनाने के लिए सीधा है:

julia> macro myshow(expression)
           quote
               value = $expression
               println($(Meta.quot(expression)), " = ", value)
               value
           end
       end

नए संस्करण का उपयोग करने के लिए, बस @myshow मैक्रो का उपयोग करें:

julia> x = @myshow 1 + 1
1 + 1 = 2
2

julia> x
2

लूप तक

हम सब के लिए इस्तेमाल कर रहे हैं while वाक्य रचना, कि अपने शरीर को निष्पादित करता है, जबकि हालत के लिए मूल्यांकन किया जाता है true । क्या होगा यदि हम एक लूप until लागू करना चाहते हैं, जो कि एक लूप को निष्पादित करता है जब तक कि स्थिति का true मूल्यांकन नहीं किया जाता true ?

जूलिया में, हम एक @until मैक्रो बनाकर कर सकते हैं, जो स्थिति पूरी होने पर अपने शरीर को निष्पादित करने के लिए बंद हो जाता है:

macro until(condition, expression)
    quote
        while !($condition)
            $expression
        end
    end |> esc
end

यहां हमने फ़ंक्शन चाइनिंग सिंटैक्स |> , जो कि पूरे quote ब्लॉक पर esc फ़ंक्शन को कॉल करने के बराबर है। esc फ़ंक्शन मैक्रो स्वच्छता को मैक्रो की सामग्री पर लागू करने से रोकता है; इसके बिना, बाहरी चर के साथ टकराव को रोकने के लिए मैक्रो में स्कैन किए गए चर का नाम बदला जाएगा। अधिक विवरण के लिए मैक्रो हाइजीन पर जूलिया प्रलेखन देखें।

आप इस पाश में एक से अधिक अभिव्यक्ति का उपयोग कर सकते हैं, बस एक begin ... end अंदर सब कुछ डालकर begin ... end ब्लॉक:

julia> i = 0;

julia> @until i == 10 begin
           println(i)
           i += 1
       end
0
1
2
3
4
5
6
7
8
9

julia> i
10

QuoteNode, Meta.quot, और Expr (: उद्धरण)

एक जूलिया फ़ंक्शन का उपयोग करके कुछ उद्धरण करने के तीन तरीके हैं:

julia> QuoteNode(:x)
:(:x)

julia> Meta.quot(:x)
:(:x)

julia> Expr(:quote, :x)
:(:x)

"उद्धरण" का क्या अर्थ है, और यह किसके लिए अच्छा है? उद्धरण हमें जूलिया द्वारा विशेष रूपों के रूप में व्याख्या किए जाने से अभिव्यक्ति की रक्षा करने की अनुमति देता है। एक सामान्य उपयोग मामला तब होता है जब हम ऐसे भाव उत्पन्न करते हैं जिनमें प्रतीकों का मूल्यांकन करने वाली चीजें होनी चाहिए। (उदाहरण के लिए, इस मैक्रो को एक ऐसे प्रतीक को वापस करने की ज़रूरत है जो प्रतीक का मूल्यांकन करता है।) यह केवल प्रतीक को वापस करने के लिए काम नहीं करता है:

julia> macro mysym(); :x; end
@mysym (macro with 1 method)

julia> @mysym
ERROR: UndefVarError: x not defined

julia> macroexpand(:(@mysym))
:x

यहाँ क्या चल रहा है? @mysym विस्तार होता है :x , जिसे एक्सप्रेशन के रूप में वेरिएबल x रूप में समझा जाता है। लेकिन अभी तक कुछ भी x नहीं सौंपा गया है, इसलिए हमें एक x not defined त्रुटि मिलती है।

इसके आसपास जाने के लिए, हमें अपने मैक्रो के परिणाम को उद्धृत करना होगा:

julia> macro mysym2(); Meta.quot(:x); end
@mysym2 (macro with 1 method)

julia> @mysym2
:x

julia> macroexpand(:(@mysym2))
:(:x)

यहां, हमने अपने प्रतीक को एक उद्धृत प्रतीक में बदलने के लिए Meta.quot फ़ंक्शन का उपयोग किया है, जो कि वह परिणाम है जो हम चाहते हैं।

Meta.quot और QuoteNode बीच क्या अंतर है, और मुझे QuoteNode उपयोग करना चाहिए? लगभग सभी मामलों में, अंतर वास्तव में मायने नहीं रखता है। QuoteNode बजाय QuoteNode का उपयोग करना कभी-कभी थोड़ा सुरक्षित Meta.quot । हालांकि, जूलिया एक्सप्रेशन और मैक्रोज़ कैसे काम करते हैं, इस अंतर की खोज करना जानकारीपूर्ण है।

Meta.quot और QuoteNode बीच का अंतर, समझाया गया

यहाँ एक नियम है:

  • यदि आपको प्रक्षेप का समर्थन करने की आवश्यकता है या करना चाहते हैं, तो Meta.quot उपयोग Meta.quot ;
  • यदि आप प्रक्षेप को अनुमति नहीं दे सकते हैं या नहीं चाहते हैं, तो QuoteNode उपयोग QuoteNode

संक्षेप में, अंतर यह है कि Meta.quot उद्धृत चीज़ के भीतर प्रक्षेप की अनुमति देता है, जबकि QuoteNode अपने तर्क को किसी भी प्रक्षेप से बचाता है। प्रक्षेप को समझने के लिए, $ अभिव्यक्ति का उल्लेख करना महत्वपूर्ण है। जूलिया में एक तरह की अभिव्यक्ति है जिसे $ अभिव्यक्ति कहा जाता है। ये भाव बचने की अनुमति देते हैं। उदाहरण के लिए, निम्नलिखित अभिव्यक्ति पर विचार करें:

julia> ex = :( x = 1; :($x + $x) )
quote 
    x = 1
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

जब मूल्यांकन किया जाता है, तो यह अभिव्यक्ति 1 का मूल्यांकन करेगा और इसे x को असाइन करेगा, फिर फॉर्म की अभिव्यक्ति का निर्माण करेगा _ + _ जहां x के मूल्य से _ को बदल दिया जाएगा। इस प्रकार, इसका परिणाम अभिव्यक्ति 1 + 1 होना चाहिए (जिसका अभी तक मूल्यांकन नहीं किया गया है, और मूल्य 2 से इतना अलग है)। वास्तव में, यह मामला है:

julia> eval(ex)
:(1 + 1)

आइए अब कहते हैं कि हम इस प्रकार के भाव बनाने के लिए एक मैक्रो लिख रहे हैं। हमारा मैक्रो एक तर्क लेगा, जो उपरोक्त ex में 1 को बदल देगा। यह तर्क निश्चित रूप से कोई भी अभिव्यक्ति हो सकती है। यहाँ कुछ ऐसा है जो हम नहीं चाहते हैं:

julia> macro makeex(arg)
           quote
               :( x = $(esc($arg)); :($x + $x) )
           end
       end
@makeex (macro with 1 method)

julia> @makeex 1
quote 
    x = $(Expr(:escape, 1))
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

julia> @makeex 1 + 1
quote 
    x = $(Expr(:escape, 2))
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

दूसरा मामला गलत है, क्योंकि हमें 1 + 1 मूल्यांकन नहीं करना चाहिए। हम Meta.quot साथ तर्क को उद्धृत करके इसे ठीक करते हैं:

julia> macro makeex2(arg)
           quote
               :( x = $$(Meta.quot(arg)); :($x + $x) )
           end
       end
@makeex2 (macro with 1 method)

julia> @makeex2 1 + 1
quote 
    x = 1 + 1
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

मैक्रो स्वच्छता किसी उद्धरण की सामग्री पर लागू नहीं होती है, इसलिए इस मामले में (और वास्तव में कानूनी नहीं) इस मामले में बचना आवश्यक नहीं है।

जैसा कि पहले उल्लेख किया गया है, Meta.quot प्रक्षेप करने की अनुमति देता है। तो चलिए कोशिश करते हैं कि:

julia> @makeex2 1 + $(sin(1))
quote 
    x = 1 + 0.8414709848078965
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

julia> let q = 0.5
           @makeex2 1 + $q
       end
quote 
    x = 1 + 0.5
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

पहले उदाहरण से, हम देखते हैं कि प्रक्षेप हमें sin(1) को रेखांकित करने की अनुमति देता है, इसके बजाय अभिव्यक्ति शाब्दिक sin(1) । दूसरा उदाहरण दिखाता है कि यह प्रक्षेप मैक्रो इनवोकेशन दायरे में किया जाता है, न कि मैक्रो के स्वयं के दायरे में। ऐसा इसलिए है क्योंकि हमारे मैक्रो ने वास्तव में किसी भी कोड का मूल्यांकन नहीं किया है; यह सब कर रहा है कोड उत्पन्न कर रहा है। कोड का मूल्यांकन (जो अभिव्यक्ति में अपना रास्ता बनाता है) तब किया जाता है जब अभिव्यक्ति मैक्रो उत्पन्न करता है वास्तव में चलाया जाता है।

अगर हमने इसके बजाय QuoteNode का उपयोग किया QuoteNode तो क्या होता? जैसा कि आप अनुमान लगा सकते हैं, क्योंकि QuoteNode को QuoteNode होने से रोकता है, इसका मतलब है कि यह काम नहीं करेगा।

julia> macro makeex3(arg)
           quote
               :( x = $$(QuoteNode(arg)); :($x + $x) )
           end
       end
@makeex3 (macro with 1 method)

julia> @makeex3 1 + $(sin(1))
quote 
    x = 1 + $(Expr(:$, :(sin(1))))
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

julia> let q = 0.5
           @makeex3 1 + $q
       end
quote 
    x = 1 + $(Expr(:$, :q))
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

julia> eval(@makeex3 $(sin(1)))
ERROR: unsupported or misplaced expression $
 in eval(::Module, ::Any) at ./boot.jl:234
 in eval(::Any) at ./boot.jl:233

इस उदाहरण में, हम सहमत हो सकते हैं कि Meta.quot अधिक लचीलापन देता है, क्योंकि यह प्रक्षेप की अनुमति देता है। तो हम कभी भी QuoteNode का उपयोग करने पर विचार क्यों कर सकते हैं? कुछ मामलों में, हम वास्तव में प्रक्षेप की इच्छा नहीं कर सकते हैं, और वास्तव में शाब्दिक $ अभिव्यक्ति चाहते हैं। यह वांछनीय कब होगा? चलिए @makeex एक सामान्यीकरण पर विचार @makeex जहाँ हम अतिरिक्त तर्क पास कर सकते हैं जो यह निर्धारित करते हैं कि + चिह्न के बाएँ और दाएँ पर क्या आता है:

julia> macro makeex4(expr, left, right)
           quote
               quote
                   $$(Meta.quot(expr))
                   :($$$(Meta.quot(left)) + $$$(Meta.quot(right)))
               end
           end
       end
@makeex4 (macro with 1 method)

julia> @makeex4 x=1 x x
quote  # REPL[110], line 4:
    x = 1 # REPL[110], line 5:
    $(Expr(:quote, :($(Expr(:$, :x)) + $(Expr(:$, :x)))))
end

julia> eval(ans)
:(1 + 1)

@makeex4 के हमारे कार्यान्वयन की एक सीमा यह है कि हम अभिव्यक्ति के बाएँ और दाएँ पक्ष के रूप में सीधे अभिव्यक्ति का उपयोग नहीं कर सकते, क्योंकि वे प्रक्षेपित होते हैं। दूसरे शब्दों में, अभिव्यक्तियों को प्रक्षेप के लिए मूल्यांकन किया जा सकता है, लेकिन हम उन्हें संरक्षित रख सकते हैं। (चूँकि यहाँ उद्धृत और मूल्यांकन के कई स्तर हैं, तो हम स्पष्ट करें: हमारा मैक्रो कोड उत्पन्न करता है जो एक अभिव्यक्ति का निर्माण करता है जब मूल्यांकन किया जाता है कि एक और अभिव्यक्ति का उत्पादन होता है!

julia> @makeex4 x=1 x/2 x
quote  # REPL[110], line 4:
    x = 1 # REPL[110], line 5:
    $(Expr(:quote, :($(Expr(:$, :(x / 2))) + $(Expr(:$, :x)))))
end

julia> eval(ans)
:(0.5 + 1)

हमें उपयोगकर्ता को यह निर्दिष्ट करने की अनुमति देना चाहिए कि प्रक्षेप कब होना है, और कब नहीं होना चाहिए। सैद्धांतिक रूप से, यह एक आसान समाधान है: हम अपने आवेदन में केवल $ संकेतों में से एक को हटा सकते हैं, और उपयोगकर्ता को अपना योगदान दे सकते हैं। इसका मतलब यह है कि हम उपयोगकर्ता द्वारा दर्ज किए गए अभिव्यक्ति के एक उद्धृत संस्करण को प्रक्षेपित करते हैं (जो हमने पहले ही उद्धृत किया है और एक बार प्रक्षेपित किया है)। यह निम्न कोड की ओर जाता है, जो कि पहले से ही उद्धृत और unquoting के कई नेस्टेड स्तरों के कारण थोड़ा भ्रमित हो सकता है। प्रत्येक भागने के लिए क्या है, इसे पढ़ने और समझने की कोशिश करें।

julia> macro makeex5(expr, left, right)
           quote
               quote
                   $$(Meta.quot(expr))
                   :($$(Meta.quot($(Meta.quot(left)))) + $$(Meta.quot($(Meta.quot(right)))))
               end
           end
       end
@makeex5 (macro with 1 method)

julia> @makeex5 x=1 1/2 1/4
quote  # REPL[121], line 4:
    x = 1 # REPL[121], line 5:
    $(Expr(:quote, :($(Expr(:$, :($(Expr(:quote, :(1 / 2)))))) + $(Expr(:$, :($(Expr(:quote, :(1 / 4)))))))))
end

julia> eval(ans)
:(1 / 2 + 1 / 4)

julia> @makeex5 y=1 $y $y
ERROR: UndefVarError: y not defined

चीजें अच्छी शुरू हुईं, लेकिन कुछ गलत हो गया है। मैक्रो का जेनरेट किया गया कोड मैक्रो इनवोकेशन स्कोप में y की कॉपी को इंटरपोल करने की कोशिश कर रहा है; लेकिन मैक्रो इनवोकेशन स्कोप में y की कोई कॉपी नहीं है। हमारी त्रुटि मैक्रो में दूसरे और तीसरे तर्क के साथ प्रक्षेप की अनुमति दे रही है। इस त्रुटि को ठीक करने के लिए, हमें QuoteNode उपयोग करना चाहिए।

julia> macro makeex6(expr, left, right)
           quote
               quote
                   $$(Meta.quot(expr))
                   :($$(Meta.quot($(QuoteNode(left)))) + $$(Meta.quot($(QuoteNode(right)))))
               end
           end
       end
@makeex6 (macro with 1 method)

julia> @makeex6 y=1 1/2 1/4
quote  # REPL[129], line 4:
    y = 1 # REPL[129], line 5:
    $(Expr(:quote, :($(Expr(:$, :($(Expr(:quote, :(1 / 2)))))) + $(Expr(:$, :($(Expr(:quote, :(1 / 4)))))))))
end

julia> eval(ans)
:(1 / 2 + 1 / 4)

julia> @makeex6 y=1 $y $y
quote  # REPL[129], line 4:
    y = 1 # REPL[129], line 5:
    $(Expr(:quote, :($(Expr(:$, :($(Expr(:quote, :($(Expr(:$, :y)))))))) + $(Expr(:$, :($(Expr(:quote, :($(Expr(:$, :y)))))))))))
end

julia> eval(ans)
:(1 + 1)

julia> @makeex6 y=1 1+$y $y
quote  # REPL[129], line 4:
    y = 1 # REPL[129], line 5:
    $(Expr(:quote, :($(Expr(:$, :($(Expr(:quote, :(1 + $(Expr(:$, :y)))))))) + $(Expr(:$, :($(Expr(:quote, :($(Expr(:$, :y)))))))))))
end

julia> @makeex6 y=1 $y/2 $y
quote  # REPL[129], line 4:
    y = 1 # REPL[129], line 5:
    $(Expr(:quote, :($(Expr(:$, :($(Expr(:quote, :($(Expr(:$, :y)) / 2)))))) + $(Expr(:$, :($(Expr(:quote, :($(Expr(:$, :y)))))))))))
end

julia> eval(ans)
:(1 / 2 + 1)

QuoteNode का उपयोग QuoteNode , हमने अपने तर्कों को प्रक्षेप से सुरक्षित किया है। चूंकि QuoteNode केवल अतिरिक्त सुरक्षा का प्रभाव होता है, इसलिए जब तक आप प्रक्षेप की इच्छा नहीं करते, तब तक QuoteNode का उपयोग करना कभी भी हानिकारक नहीं होता है। हालांकि, अंतर को समझने से यह समझना संभव हो जाता है कि Meta.quot कहां और क्यों एक बेहतर विकल्प हो सकता है।

यह लंबी कवायद एक उदाहरण के साथ है जो स्पष्ट रूप से किसी भी उचित अनुप्रयोग में दिखाने के लिए बहुत जटिल है। इसलिए, हमने पहले उल्लिखित अंगूठे के निम्नलिखित नियम को सही ठहराया है:

  • यदि आपको प्रक्षेप का समर्थन करने की आवश्यकता है या करना चाहते हैं, तो Meta.quot उपयोग Meta.quot ;
  • यदि आप प्रक्षेप को अनुमति नहीं दे सकते हैं या नहीं चाहते हैं, तो QuoteNode उपयोग QuoteNode

Expr (: बोली) के बारे में क्या?

Expr(:quote, x) Meta.quot(x) Expr(:quote, x) Meta.quot(x) बराबर है। हालांकि, उत्तरार्द्ध अधिक मुहावरेदार है और पसंद किया जाता है। कोड है कि भारी metaprogramming का उपयोग करता है के लिए, एक using Base.Meta लाइन अक्सर प्रयोग किया जाता है, जो की अनुमति देता है Meta.quot बस के रूप में भेजा जा करने के लिए quot

मार्गदर्शक

& मेटाप्रोग्रामिंग बिट्स और बोब्स

लक्ष्य:

  • उपयुक्त लक्षित कार्यात्मक / उपयोगी / गैर-अमूर्त उदाहरणों (उदाहरण @swap या @assert ) के @swap से @assert जो उपयुक्त संदर्भों में अवधारणाओं को प्रस्तुत करते हैं।

  • कोड को व्याख्या के पैराग्राफ के बजाय अवधारणाओं को दर्शाने / प्रदर्शित करने को प्राथमिकता दें

  • 'आवश्यक पढ़ने' को अन्य पृष्ठों से जोड़ने से बचें - यह कथा को बाधित करता है

  • चीजों को एक समझदार क्रम में प्रस्तुत करें जो सीखने को आसान बना देगा

संसाधन:

julialang.org
विकीबूक (@Cormullion)
5 परतें (लिआह हैंसन)
SO-Doc उद्धरण (@TotalVerb)
SO-Doc - वे प्रतीक जो कानूनी पहचानकर्ता नहीं हैं (@TotalVerb)
SO: जूलिया में एक प्रतीक क्या है (@StefanKarpinski)
प्रवचन धागा (@ PI-) Metaprogramming

अधिकांश सामग्री प्रवचन चैनल से आई है, जिसमें से अधिकांश fcard से आई है ... कृपया मुझे ठेस पहुंचाएं अगर मैं भूल गया था।

प्रतीक

julia> mySymbol = Symbol("myName")  # or 'identifier'
:myName

julia> myName = 42
42

julia> mySymbol |> eval  # 'foo |> bar' puts output of 'foo' into 'bar', so 'bar(foo)'
42

julia> :( $mySymbol = 1 ) |> eval
1

julia> myName
1

कार्यों में झंडे पारित करना:

function dothing(flag)
  if flag == :thing_one
    println("did thing one")
  elseif flag == :thing_two
    println("did thing two")
  end
end
julia> dothing(:thing_one)
did thing one

julia> dothing(:thing_two)
did thing two

एक हैशेक उदाहरण:

number_names = Dict{Symbol, Int}()
number_names[:one] = 1
number_names[:two] = 2
number_names[:six] = 6

(उन्नत) (@fcard) :foo aka :(foo) एक प्रतीक देता है यदि foo एक वैध पहचानकर्ता है, अन्यथा एक अभिव्यक्ति है।

# NOTE: Different use of ':' is:
julia> :mySymbol = Symbol('hello world')

#(You can create a symbol with any name with Symbol("<name>"), 
# which lets us create such gems as:
julia> one_plus_one = Symbol("1 + 1")
Symbol("1 + 1")

julia> eval(one_plus_one)
ERROR: UndefVarError: 1 + 1 not defined
...

julia> valid_math = :($one_plus_one = 3)
:(1 + 1 = 3)

julia> one_plus_one_plus_two = :($one_plus_one + 2)
:(1 + 1 + 2)

julia> eval(quote
           $valid_math
           @show($one_plus_one_plus_two)
       end)
1 + 1 + 2 = 5
...

मूल रूप से आप प्रतीकों को हल्के तारों के रूप में मान सकते हैं। ऐसा नहीं है कि वे क्या कर रहे हैं, लेकिन आप यह कर सकते हैं, तो क्यों नहीं। जूलिया का बेस खुद इसे करता है, print_with_color(:red, "abc") एक लाल रंग का abc प्रिंट करता है।

Expr (एएसटी)

(लगभग) जूलिया में सब कुछ एक अभिव्यक्ति है, यानी Expr का एक उदाहरण, जो एएसटी आयोजित करेगा।

# when you type ...
julia> 1+1
2

# Julia is doing: eval(parse("1+1"))
# i.e. First it parses the string "1+1" into an `Expr` object ...
julia> ast = parse("1+1")
:(1 + 1)

# ... which it then evaluates:
julia> eval(ast)
2

# An Expr instance holds an AST (Abstract Syntax Tree).  Let's look at it:
julia> dump(ast)
Expr
  head: Symbol call
  args: Array{Any}((3,))
    1: Symbol +
    2: Int64 1
    3: Int64 1
  typ: Any
  
# TRY: fieldnames(typeof(ast))
 
julia>      :(a + b*c + 1)  ==
       parse("a + b*c + 1") ==
       Expr(:call, :+, :a, Expr(:call, :*, :b, :c), 1)
true

नेस्टिंग Expr रों:

julia> dump( :(1+2/3) )
Expr
  head: Symbol call
  args: Array{Any}((3,))
    1: Symbol +
    2: Int64 1
    3: Expr
      head: Symbol call
      args: Array{Any}((3,))
        1: Symbol /
        2: Int64 2
        3: Int64 3
      typ: Any
  typ: Any
  
# Tidier rep'n using s-expr
julia> Meta.show_sexpr( :(1+2/3) ) 
(:call, :+, 1, (:call, :/, 2, 3))

बहु Expr s का उपयोग करके quote

julia> blk = quote
           x=10
           x+1
       end
quote  # REPL[121], line 2:
    x = 10 # REPL[121], line 3:
    x + 1
end

julia> blk == :( begin  x=10; x+1  end )
true

# Note: contains debug info:
julia> Meta.show_sexpr(blk)
(:block,
  (:line, 2, Symbol("REPL[121]")),
  (:(=), :x, 10),
  (:line, 3, Symbol("REPL[121]")),
  (:call, :+, :x, 1)
)

# ... unlike:
julia> noDbg = :( x=10; x+1 ) 
quote 
    x = 10
    x + 1
end

... तो quote कार्यात्मक रूप से समान है लेकिन अतिरिक्त डिबग जानकारी प्रदान करता है।

(*) टिप: उपयोग let रखने के लिए x ब्लॉक के भीतर

quote एक quote

Expr(:quote, x) का उपयोग उद्धरणों के भीतर उद्धरणों को दर्शाने के लिए किया जाता है।

Expr(:quote, :(x + y)) == :(:(x + y))

Expr(:quote, Expr(:$, :x)) == :(:($x))

QuoteNode(x) Expr(:quote, x) QuoteNode(x) के समान है लेकिन यह प्रक्षेप को रोकता है।

eval(Expr(:quote, Expr(:$, 1))) == 1

eval(QuoteNode(Expr(:$, 1))) == Expr(:$, 1)

( जूलिया मेटाप्रोग्रामिंग में विभिन्न उद्धृत तंत्रों का खंडन करें

क्या $ और : (…) किसी तरह एक दूसरे के उलटफेर करते हैं?

:(foo) अर्थ है "मूल्य को मत देखो, अभिव्यक्ति को देखो" $foo अर्थ है "अभिव्यक्ति को उसके मूल्य में बदलें"

:($(foo)) == foo$(:(foo)) एक त्रुटि है। $(...) एक ऑपरेशन नहीं है और अपने आप से कुछ भी नहीं करता है, यह एक "यह प्रक्षेप है!" साइन इन करें कि उद्धृत वाक्यविन्यास उपयोग करता है। यानी यह केवल एक उद्धरण के भीतर मौजूद है।

क्या $ foo , eval( foo ) ?

नहीं! $foo का संकलन-समय मान eval(foo) इसका मतलब है कि रनटाइम पर

वैश्विक स्कोप प्रक्षेपित होगा स्थानीय में eval होगा

eval(:<expr>) को वैसा ही लौटना चाहिए जैसा कि <expr> (यह मानते हुए कि <expr> वर्तमान वैश्विक स्थान में एक वैध अभिव्यक्ति है)

eval(:(1 + 2)) == 1 + 2

eval(:(let x=1; x + 1 end)) == let x=1; x + 1 end

macro एस

तैयार? :)

# let's try to make this!
julia> x = 5; @show x;
x = 5

आइए बनाते हैं अपना खुद का @show मैक्रो:

macro log(x)
  :(
    println( "Expression: ", $(string(x)), " has value: ", $x )
  )
end

u = 42
f = x -> x^2
@log(u)       # Expression: u has value: 42
@log(42)      # Expression: 42 has value: 42
@log(f(42))   # Expression: f(42) has value: 1764
@log(:u)      # Expression: :u has value: u

एक Expr को कम expand लिए expand

5 लेयर्स (लिआह हैनसन) <- बताती है कि जूलिया कैसे एक स्ट्रिंग के रूप में सोर्स कोड लेती है, इसे एक Expr टीटीआरटी (एएसटी) में बदल देती है, सभी मैक्रोज़ (अभी भी एएसटी) को बाहर निकालती है, कम करती है (एएसटी को कम करती है), फिर एलएलवीएम में परिवर्तित हो जाती है। (और आगे - फिलहाल हमें चिंता करने की जरूरत नहीं है कि इससे आगे क्या है!)

क्यू: code_lowered कार्यों पर काम करता है। क्या Expr को कम करना संभव है? A: हाँ!

# function -> lowered-AST
julia> code_lowered(*,(String,String))
1-element Array{LambdaInfo,1}:
 LambdaInfo template for *(s1::AbstractString, ss::AbstractString...) at strings/basic.jl:84

# Expr(i.e. AST) -> lowered-AST
julia> expand(:(x ? y : z))
:(begin
        unless x goto 3
        return y
        3:
        return z
    end)

julia> expand(:(y .= x.(i)))
:((Base.broadcast!)(x,y,i))

# 'Execute' AST or lowered-AST
julia> eval(ast)

यदि आप केवल मैक्रोज़ का विस्तार करना चाहते हैं, तो आप macroexpand उपयोग कर सकते हैं:

# AST -> (still nonlowered-)AST but with macros expanded:
julia> macroexpand(:(@show x))
quote
    (Base.println)("x = ",(Base.repr)(begin  # show.jl, line 229:
                #28#value = x
            end))
    #28#value
end

... जो एक गैर-कम एएसटी देता है, लेकिन सभी मैक्रोज़ के विस्तार के साथ।

esc()

esc(x) एक Expr देता है जो कहता है कि "इस पर स्वच्छता लागू न करें", यह Expr(:escape, x) । स्वच्छता वह है जो एक मैक्रो को आत्म-निहित रखता है, और यदि आप उन्हें "लीक" करना चाहते हैं तो आप चीजों को esc करते हैं। जैसे

उदाहरण: swap मैक्रो को चित्रण esc()

macro swap(p, q)
  quote
    tmp = $(esc(p))
    $(esc(p)) = $(esc(q))
    $(esc(q)) = tmp
  end
end

x,y = 1,2
@swap(x,y)
println(x,y)  # 2 1

$ हमें ' quote से बाहर निकलने' की अनुमति देता है। तो बस $p और $q क्यों नहीं? अर्थात

    # FAIL!
    tmp = $p
    $p = $q
    $q = tmp

क्योंकि वह करने के लिए पहले विचार करेंगे macro के लिए गुंजाइश p , और यह एक स्थानीय मिलेगा p यानी पैरामीटर p (हाँ, यदि आप बाद में पहुँच p बिना esc आईएनजी, मैक्रो समझता है p एक स्थानीय चर के रूप में पैरामीटर)।

तो $p = ... केवल स्थानीय p को असाइन करना है। कॉलिंग के संदर्भ में जो भी चर पारित किया गया था उसे प्रभावित नहीं कर रहा है।

ठीक है तो कैसे के बारे में:

    # Almost!
    tmp = $p          # <-- you might think we don't 
    $(esc(p)) = $q    #       need to esc() the RHS
    $(esc(q)) = tmp

तो esc(p) कॉलिंग संदर्भ में 'लीक' p है। "वह चीज़ जो मैक्रो में पारित की गई थी जिसे हम p रूप में प्राप्त करते हैं "

julia> macro swap(p, q)                  
         quote                           
           tmp = $p                      
           $(esc(p)) = $q                
           $(esc(q)) = tmp               
         end                             
       end                               
@swap (macro with 1 method)              

julia> x, y = 1, 2                       
(1,2)                                    

julia> @swap(x, y);                      

julia> @show(x, y);                      
x = 2                                    
y = 1                                    

julia> macroexpand(:(@swap(x, y)))       
quote  # REPL[34], line 3:               
    #10#tmp = x # REPL[34], line 4:      
    x = y # REPL[34], line 5:            
    y = #10#tmp                          
end                                      

जैसा कि आप देख सकते हैं कि tmp को हाइजीन ट्रीटमेंट #10#tmp , जबकि x और y नहीं। जूलिया tmp लिए एक विशिष्ट पहचानकर्ता बना रही है, ऐसा कुछ जिसे आप मैन्युअल रूप से gensym साथ कर सकते हैं, अर्थात:

julia> gensym(:tmp)
Symbol("##tmp#270")

लेकिन: एक गोत्र है:

julia> module Swap
       export @swap

       macro swap(p, q)
         quote
           tmp = $p
           $(esc(p)) = $q
           $(esc(q)) = tmp
         end
       end
       end
Swap

julia> using Swap

julia> x,y = 1,2
(1,2)

julia> @swap(x,y)
ERROR: UndefVarError: x not defined

एक और बात जूलिया की मैक्रो हाइजीन है, अगर मैक्रो दूसरे मॉड्यूल से है, तो यह किसी भी वैरिएबल को बनाता है (जो कि मैक्रो की रिटर्निंग एक्सप्रेशन के अंदर नहीं दिया गया है, जैसे कि इस मामले में tmp तरह) मौजूदा मॉड्यूल के ग्लोबल्स, इसलिए $p Swap.$p , इसी तरह $q -> Swap.$q

सामान्य तौर पर, यदि आपको एक चर की आवश्यकता है जो कि मैक्रो के दायरे से बाहर है, तो आपको इसे esc(p) करना चाहिए, इसलिए आपको esc(p) और esc(q) परवाह किए बिना अगर वे एक अभिव्यक्ति के एलएचएस या आरएचएस पर हैं, या यहां तक कि खुद से भी।

लोगों ने पहले से ही कुछ समय में gensym sa का उल्लेख किया है और जल्द ही आप पूरी अभिव्यक्ति को दूर करने के अंधेरे पक्ष से gensym कुछ gensym s के साथ यहाँ और वहाँ peppered है, लेकिन ... यह समझने की कोशिश करें कि स्वच्छता कैसे काम करने से पहले काम करती है। इससे ज्यादा स्मार्ट! यह एक विशेष रूप से जटिल एल्गोरिथ्म नहीं है इसलिए इसे बहुत लंबा नहीं होना चाहिए, लेकिन इसे जल्दी मत करो! उस शक्ति का उपयोग तब तक न करें जब तक कि आप इसके सभी प्रभावों को नहीं समझ लेते ... (@fcard)

उदाहरण: मैक्रो until

(@ इस्माइल-वीसी)

"until loop"
macro until(condition, block)
    quote
        while ! $condition
            $block
        end
    end |> esc
end

julia> i=1;  @until(  i==5,  begin; print(i); i+=1; end  )
1234

(@fcard) |> हालांकि विवादास्पद है। मुझे आश्चर्य है कि एक भीड़ अभी तक बहस करने नहीं आई है। (शायद हर कोई बस थक गया है)। अधिकांश होने की सिफारिश है यदि सभी मैक्रो सिर्फ एक फ़ंक्शन के लिए कॉल न हों, तो:

macro until(condition, block)
    esc(until(condition, block))
end

function until(condition, block)
    quote
        while !$condition
            $block
        end
    end
end

... एक सुरक्षित विकल्प है।

## @ fcard की सरल मैक्रो चुनौती

टास्क: ऑपरेंड्स को स्वैप करें, इसलिए swaps(1/2) 2.00 यानी 2/1 देता है

macro swaps(e)
    e.args[2:3] = e.args[3:-1:2]   
    e
end
@swaps(1/2)
2.00

यहाँ @fcard से अधिक मैक्रो चुनौतियां


प्रक्षेप और स्थूल assert

http://docs.julialang.org/en/release-0.5/manual/metaprogramming/#building-an-advanced-macro

macro assert(ex)
    return :( $ex ? nothing : throw(AssertionError($(string(ex)))) )
end

क्यू: क्यों पिछले $ ? उत्तर: यह जूलिया को उस string(ex) को eval लिए मजबूर करता है, क्योंकि निष्पादन इस मैक्रो के आह्वान से गुजरता है। यदि आप बस उस कोड को चलाते हैं तो यह किसी भी मूल्यांकन को बाध्य नहीं करेगा। लेकिन जैसे ही आप कर assert(foo) जूलिया इस मैक्रो की जगह लागू करेगा अपनी 'एएसटी टोकन / Expr' के साथ जो कुछ भी यह देता है, और $ कार्रवाई में शुरू होगा।

ब्लॉक के लिए {} का उपयोग करने के लिए एक मजेदार हैक

(@fcard) मुझे नहीं लगता कि {} को ब्लॉक के रूप में उपयोग किए जाने से कुछ भी तकनीकी रखना है, वास्तव में कोई इसे काम करने के लिए अवशिष्ट {} सिंटैक्स पर भी सजा सकता है:

julia> macro c(block)
         @assert block.head == :cell1d
         esc(quote
           $(block.args...)
         end)
       end
@c (macro with 1 method)

julia> @c {
         print(1)
         print(2)
         1+2
       }
123

* (अगर {} सिंटैक्स का पुनर्भुगतान किया जाता है तो / अभी भी काम करने की संभावना नहीं है)


तो पहले जूलिया मैक्रो टोकन देखता है, इसलिए यह मिलान end तक टोकन पढ़ेगा / पार्स करेगा, और क्या बनायेगा? के साथ एक Expr .head=:macro या कुछ और? क्या यह "a+1" को एक स्ट्रिंग के रूप में संग्रहीत करता है या क्या इसे अलग करता है :+(:a, 1) ? कैसे देखें?

?

(@fcard) इस मामले में लेक्सिकल स्कोप के कारण, @M स्कोप में अपरिभाषित है, इसलिए यह ग्लोबल वैरिएबल का उपयोग करता है ... मैं वास्तव में अपने गूंगे उदाहरण में फ़्लिपलिन की अभिव्यक्ति से बचना भूल गया था, लेकिन "केवल भीतर काम करता है" एक ही मॉड्यूल "इसका हिस्सा अभी भी लागू होता है।

julia> module M
       macro m()
         :(a+1)
       end
       end
M

julia> a = 1
1

julia> M.@m
ERROR: UndefVarError: a not defined

इसका कारण यह है कि, यदि मैक्रो का उपयोग किसी अन्य मॉड्यूल में किया गया है, जो कि इसे परिभाषित किया गया था, के अलावा किसी भी चर को कोड-से-विस्तारित के भीतर परिभाषित नहीं किया गया है, मैक्रो के मॉड्यूल के ग्लोबल्स के रूप में माना जाता है।

julia> macroexpand(:(M.@m))
:(M.a + 1)

उन्नत

### @ इस्माइल-कुलपति

@eval begin
    "do-until loop"
    macro $(:do)(block, until::Symbol, condition)
        until ≠ :until && 
            error("@do expected `until` got `$until`")
        quote
            let
                $block
                @until $condition begin
                    $block
                end
            end
        end |> esc
    end
end
julia> i = 0            
0                       

julia> @do begin        
           @show i      
           i += 1       
       end until i == 5 
i = 0                   
i = 1                   
i = 2                   
i = 3                   
i = 4

स्कॉट का मैक्रो:

"""
Internal function to return captured line number information from AST

##Parameters
- a:     Expression in the julia type Expr

##Return

- Line number in the file where the calling macro was invoked
"""
_lin(a::Expr) = a.args[2].args[1].args[1]

"""
Internal function to return captured file name information from AST

##Parameters
- a:     Expression in the julia type Expr

##Return
- The name of the file where the macro was invoked
"""
_fil(a::Expr) = string(a.args[2].args[1].args[2])

"""
Internal function to determine if a symbol is a status code or variable
"""
function _is_status(sym::Symbol)
    sym in (:OK, :WARNING, :ERROR) && return true
    str = string(sym)
    length(str) > 4 && (str[1:4] == "ERR_" || str[1:5] == "WARN_" || str[1:5] == "INFO_")
end

"""
Internal function to return captured error code from AST

##Parameters
- a:     Expression in the julia type Expr

##Return
- Error code from the captured info in the AST from the calling macro
"""
_err(a::Expr) =
    (sym = a.args[2].args[2] ; _is_status(sym) ? Expr(:., :Status, QuoteNode(sym)) : sym)

"""
Internal function to produce a call to the log function based on the macro arguments and the AST from the ()->ERRCODE anonymous function definition used to capture error code, file name and line number where the macro is used

##Parameters
- level:     Loglevel which has to be logged with macro
- a:         Expression in the julia type Expr
- msgs:      Optional message

##Return
- Statuscode
"""
function _log(level, a, msgs)
    if isempty(msgs)
        :( log($level, $(esc(:Symbol))($(_fil(a))), $(_lin(a)), $(_err(a)) )
    else
        :( log($level, $(esc(:Symbol))($(_fil(a))), $(_lin(a)), $(_err(a)), message=$(esc(msgs[1]))) )
    end
end

macro warn(a, msgs...)  ; _log(Warning, a, msgs) ; end

कबाड़ / असंसाधित ...

एक मैक्रो देखें / डम्प करें

(@ pi-) मान लीजिए कि मैं सिर्फ macro m(); a+1; end एक ताजा REPL में macro m(); a+1; end । बिना a परिभाषित के। मैं इसे 'कैसे' देख सकता हूं? जैसे, क्या किसी मैक्रो को 'डंप' करने का कोई तरीका है? वास्तव में इसे क्रियान्वित किए बिना

(@fcard) मैक्रोज़ के सभी कोड वास्तव में कार्यों में लगाए जाते हैं, इसलिए आप केवल उनके निचले या टाइप-इनफर्ड कोड को देख सकते हैं।

julia> macro m()  a+1  end
@m (macro with 1 method)

julia> @code_typed @m
LambdaInfo for @m()
:(begin 
        return Main.a + 1
    end)

julia> @code_lowered @m  
CodeInfo(:(begin
        nothing
        return Main.a + 1
    end))
# ^ or: code_lowered(eval(Symbol("@m")))[1] # ouf!

मैक्रो फ़ंक्शन को प्राप्त करने के अन्य तरीके:

julia> macro getmacro(call) call.args[1] end
@getmacro (macro with 1 method)

julia> getmacro(name) = getfield(current_module(), name.args[1])
getmacro (generic function with 1 method)

julia> @getmacro @m
@m (macro with 1 method)

julia> getmacro(:@m)
@m (macro with 1 method)
julia> eval(Symbol("@M"))
@M (macro with 1 method)

julia> dump( eval(Symbol("@M")) )
@M (function of type #@M)

julia> code_typed( eval(Symbol("@M")) )
1-element Array{Any,1}:
 LambdaInfo for @M()

julia> code_typed( eval(Symbol("@M")) )[1]
LambdaInfo for @M()
:(begin 
        return $(Expr(:copyast, :($(QuoteNode(:(a + 1))))))
    end::Expr)

julia> @code_typed @M
LambdaInfo for @M()
:(begin 
        return $(Expr(:copyast, :($(QuoteNode(:(a + 1))))))
    end::Expr)

^ लगता है कि मैं इसके बजाय code_typed उपयोग कर सकता हूं

eval(Symbol("@M")) कैसे समझें?

(@fcard) वर्तमान में, प्रत्येक मैक्रो में इसके साथ एक फ़ंक्शन जुड़ा हुआ है। यदि आपके पास M नामक एक मैक्रो है, तो मैक्रो के फ़ंक्शन को @M कहा जाता है। आम तौर पर आप उदाहरण के लिए eval(:print) साथ एक फ़ंक्शन का मान प्राप्त कर सकते हैं, लेकिन मैक्रो के फ़ंक्शन के साथ आपको Symbol("@M") , बस :@M एक Expr(:macrocall, Symbol("@M")) और मूल्यांकन कि एक मैक्रो-विस्तार का कारण बनता है।

code_typed डिस्प्ले code_typed क्यों नहीं करता है?

(@ PI-)

julia> code_typed( x -> x^2 )[1]
LambdaInfo for (::##5#6)(::Any)
:(begin 
        return x ^ 2
    end)

^ यहाँ मैं एक देखता हूं ::Any पैराम, लेकिन यह टोकन x साथ जुड़ा हुआ नहीं लगता है।

 julia> code_typed( print )[1]
LambdaInfo for print(::IO, ::Char)
:(begin 
        (Base.write)(io,c)
        return Base.nothing
    end::Void)

^ यहां भी; वहाँ io कनेक्ट करने के लिए कुछ भी नहीं है ::IO तो निश्चित रूप से यह उस विशेष print विधि के एएसटी प्रतिनिधित्व का एक पूरा डंप नहीं हो सकता है ...?

(@fcard) print(::IO, ::Char) केवल आपको बताता है कि यह किस पद्धति का है, यह एएसटी का हिस्सा नहीं है। यह अब मास्टर में भी मौजूद नहीं है:

julia> code_typed(print)[1]
CodeInfo(:(begin
        (Base.write)(io,c)
        return Base.nothing
    end))=>Void

(@ p-) मुझे समझ नहीं आ रहा है कि आपका क्या मतलब है। ऐसा लगता है कि उस विधि के शरीर के लिए एएसटी को डंप करना, नहीं? मैंने सोचा था कि code_typed फ़ंक्शन के लिए एएसटी देता है। लेकिन यह पहला कदम याद आ रहा है, यानी परम के लिए टोकन स्थापित करना।

(@fcard) code_typed का मतलब केवल शरीर की एएसटी दिखाना है, लेकिन अब यह LambdaInfo (0.5) या CodeInfo (0.6) के रूप में विधि का पूरा एएसटी देता है, लेकिन बहुत सारी जानकारी छोड़ दी जाती है जब जवाब के लिए मुद्रित। सभी विवरण प्राप्त करने के लिए आपको LambdaInfo क्षेत्र का निरीक्षण करना होगा। dump आपके उत्तर को भरता जा रहा है, इसलिए आप कोशिश कर सकते हैं:

macro method_info(call)
  quote
    method = @code_typed $(esc(call))
    print_info_fields(method)
  end
end

function print_info_fields(method)
  for field in fieldnames(typeof(method))
    if isdefined(method, field) && !(field in [Symbol(""), :code])
      println("  $field = ", getfield(method, field))
    end
  end
  display(method)
end

print_info_fields(x::Pair) = print_info_fields(x[1])

जो किसी विधि के AST के नामित क्षेत्रों के सभी मान देता है:

julia> @method_info print(STDOUT, 'a')
  rettype = Void
  sparam_syms = svec()
  sparam_vals = svec()
  specTypes = Tuple{Base.#print,Base.TTY,Char}
  slottypes = Any[Base.#print,Base.TTY,Char]
  ssavaluetypes = Any[]
  slotnames = Any[Symbol("#self#"),:io,:c]
  slotflags = UInt8[0x00,0x00,0x00]
  def = print(io::IO, c::Char) at char.jl:45
  nargs = 3
  isva = false
  inferred = true
  pure = false
  inlineable = true
  inInference = false
  inCompile = false
  jlcall_api = 0
  fptr = Ptr{Void} @0x00007f7a7e96ce10
LambdaInfo for print(::Base.TTY, ::Char)
:(begin
        $(Expr(:invoke, LambdaInfo for write(::Base.TTY, ::Char), :(Base.write), :(io), :(c)))
        return Base.nothing
    end::Void)

Lil ' def = print(io::IO, c::Char) ? तुम वहाँ जाओ! (यह भी slotnames = [..., :io, :c] भाग) इसके अलावा, हाँ, आउटपुट में अंतर इसलिए है क्योंकि मैं मास्टर पर परिणाम दिखा रहा था।

???

(@ इस्माइल-वीसी) आप इस तरह से मतलब है? प्रतीक के साथ सामान्य प्रेषण

आप इसे इस तरह से कर सकते हैं:

julia> function dispatchtest{alg}(::Type{Val{alg}})
           println("This is the generic dispatch. The algorithm is $alg")
       end
dispatchtest (generic function with 1 method)

julia> dispatchtest(alg::Symbol) = dispatchtest(Val{alg})
dispatchtest (generic function with 2 methods)

julia> function dispatchtest(::Type{Val{:Euler}})
           println("This is for the Euler algorithm!")
       end
dispatchtest (generic function with 3 methods)

julia> dispatchtest(:Foo)
This is the generic dispatch. The algorithm is Foo

julia> dispatchtest(:Euler)

यह Euler एल्गोरिथम के लिए है! मुझे आश्चर्य है कि जेनेरिक प्रतीक प्रेषण के बारे में @fcard क्या सोचता है! --- ^: परी:

मॉड्यूल गोथा

@def m begin
  a+2
end

@m # replaces the macro at compile-time with the expression a+2

अधिक सटीक रूप से, केवल मॉड्यूल के टॉपवेल के भीतर काम करता है जिसमें मैक्रो को परिभाषित किया गया था।

julia> module M
       macro m1()
         a+1
       end
       end
M

julia> macro m2()
         a+1
       end
@m2 (macro with 1 method)

julia> a = 1
1

julia> M.@m1
ERROR: UndefVarError: a not defined

julia> @m2
2

julia> let a = 20
         @m2
       end
2

esc ऐसा होने से रोकता है, लेकिन हमेशा इसका उपयोग करने में चूक करना भाषा के डिजाइन के खिलाफ जाता है। इसके लिए एक अच्छा बचाव यह है कि मैक्रों के भीतर नामों का उपयोग करने और उन्हें शुरू करने से एक को बनाए रखा जाए, जिससे उन्हें एक मानव पाठक को ट्रैक करना मुश्किल हो जाता है।

अजगर 'तानाशाह' / JSON सिंटैक्स की तरह `डिक्ट 'शाब्दिक।

परिचय

जूलिया शब्दकोशों के लिए निम्नलिखित वाक्यविन्यास का उपयोग करता है:

Dict({k₁ => v₁, k₂ => v₂, …, kₙ₋₁ => vₙ₋₁, kₙ => vₙ)

जबकि पायथन और JSON ऐसा दिखता है:

{k₁: v₁, k₂: v₂, …, kₙ₋₁: vₙ₋₁, kₙ: vₙ}

उदाहरण के प्रयोजनों के लिए हम जूलिया में इस वाक्यविन्यास का उपयोग कर सकते हैं और इसमें नए शब्दार्थ जोड़ सकते हैं (जूलिया में Dict सिंटैक्स एक मुहावरेदार तरीका है, जिसे अनुशंसित किया गया है)।

पहले देखते हैं कि यह किस प्रकार की अभिव्यक्ति है:

julia> parse("{1:2 , 3: 4}") |> Meta.show_sexpr
(:cell1d, (:(:), 1, 2), (:(:), 3, 4))

इसका मतलब है कि हमें इसे लेने की आवश्यकता है :cell1d अभिव्यक्ति और इसे बदलने या एक नई अभिव्यक्ति लौटाने के लिए जो इस तरह दिखना चाहिए:

julia> parse("Dict(1 => 2 , 3 => 4)") |> Meta.show_sexpr
(:call, :Dict, (:(=>), 1, 2), (:(=>), 3, 4))

मैक्रो की परिभाषा

निम्नलिखित मैक्रो, जबकि सरल, ऐसी कोड पीढ़ी और परिवर्तन को प्रदर्शित करने की अनुमति देता है:

macro dict(expr)
    # Check the expression has the correct form:
    if expr.head ≠ :cell1d || any(sub_expr.head ≠ :(:) for sub_expr ∈ expr.args)
        error("syntax: expected `{k₁: v₁, k₂: v₂, …, kₙ₋₁: vₙ₋₁, kₙ: vₙ}`")
    end

    # Create empty `:Dict` expression which will be returned:
    block = Expr(:call, :Dict)    # :(Dict())

    # Append `(key => value)` pairs to the block:
    for pair in expr.args
        k, v = pair.args
        push!(block.args, :($k => $v))
    end    # :(Dict(k₁ => v₁, k₂ => v₂, …, kₙ₋₁ => vₙ₋₁, kₙ => vₙ))

    # Block is escaped so it can reach variables from it's calling scope:
    return esc(block)
end

आइए परिणामी मैक्रो विस्तार की जाँच करें:

julia> :(@dict {"a": :b, 'c': 1, :d: 2.0}) |> macroexpand
:(Dict("a" => :b,'c' => 1,:d => 2.0))

प्रयोग

julia> @dict {"a": :b, 'c': 1, :d: 2.0}
Dict{Any,Any} with 3 entries:          
  "a" => :b                            
  :d  => 2.0                           
  'c' => 1                             

julia> @dict {                      
           "string": :b,            
           'c'     : 1,             
           :symbol : π,             
           Function: print,         
           (1:10)  : range(1, 10)   
       }                            
Dict{Any,Any} with 5 entries:       
  1:10     => 1:10                  
  Function => print                 
  "string" => :b                    
  :symbol  => π = 3.1415926535897...
  'c'      => 1         

अंतिम उदाहरण इसके बराबर है:

Dict(                      
    "string" => :b,            
    'c'      => 1,             
    :symbol  => π,             
    Function => print,         
    (1:10)   => range(1, 10)   
)

Misusage

julia> @dict {"one": 1, "two": 2, "three": 3, "four": 4, "five" => 5}
syntax: expected `{k₁: v₁, k₂: v₂, …, kₙ₋₁: vₙ₋₁, kₙ: vₙ}`

julia> @dict ["one": 1, "two": 2, "three": 3, "four": 4, "five" => 5]
syntax: expected `{k₁: v₁, k₂: v₂, …, kₙ₋₁: vₙ₋₁, kₙ: vₙ}`

ध्यान दें कि जूलिया के बृहदान्त्र के लिए अन्य उपयोग हैं : जैसे कि आपको कोष्ठक के साथ श्रेणी शाब्दिक अभिव्यक्तियों को लपेटना होगा या उदाहरण के लिए range फ़ंक्शन का उपयोग करना होगा।



Modified text is an extract of the original Stack Overflow Documentation
के तहत लाइसेंस प्राप्त है CC BY-SA 3.0
से संबद्ध नहीं है Stack Overflow