Computer >> कंप्यूटर >  >> प्रोग्रामिंग >> Ruby

आरबीएस:एक्शन में एक नई रूबी 3 टाइपिंग भाषा

रूबी का लंबे समय से प्रतीक्षित संस्करण 3.0.0 आखिरकार जारी कर दिया गया है। कई बेहतरीन सुधारों के साथ, जैसे पिछले संस्करण की तुलना में 3 गुना तेज प्रदर्शन बूस्ट, समांतर-समानांतर प्रयोगात्मक सुविधाएं इत्यादि, रूबी टीम ने रूबी में गतिशील टाइपिंग के लिए एक नई वाक्यविन्यास भाषा भी पेश की:आरबीएस।

शर्बत जैसे स्थैतिक प्रकार की जाँच के लिए समुदाय द्वारा विकसित उपकरणों की सफलता के आधार पर, टीम वर्षों से इस पर चर्चा कर रही थी।

शर्बत स्ट्राइप द्वारा समर्थित एक शक्तिशाली प्रकार का चेकर है। यह आरबीआई फाइलों को एनोटेट और/या परिभाषित दोनों करके आपके कोड की जांच करता है। आरबीआई फाइलें, बदले में, स्थिर और गतिशील घटकों के बीच इंटरफेस के रूप में काम करती हैं, जो उनका "विवरण" प्रदान करती हैं (स्थिरांक, पूर्वज, मेटाप्रोग्रामिंग कोड, और बहुत कुछ)।

तो, अगर शर्बत ज्यादातर स्थैतिक जाँच से संबंधित है और आरबीएस को गतिशील टाइपिंग को संबोधित करने के लिए बनाया गया था, तो उनके बीच क्या अंतर है? वे दोनों एक साथ कैसे रहेंगे? मुझे एक के बजाय दूसरे का उपयोग कब करना चाहिए?

आरबीएस की प्रमुख भूमिका के बारे में ये काफी सामान्य प्रश्न हैं। इसलिए हमने इस टुकड़े को लिखने का फैसला किया। स्पष्ट करने के लिए, व्यवहार में, आपको इसे अपनी क्षमता के आधार पर अपनाने पर विचार क्यों करना चाहिए। चलो सही में गोता लगाएँ!

बुनियादी बातों से शुरू करना

आइए स्थिर टाइपिंग _और . के बीच अंतर की स्पष्ट समझ के साथ शुरुआत करें गतिशील टाइपिंग_. हालांकि यह बुनियादी है, लेकिन आरबीएस की भूमिका को समझने के लिए इसे समझना एक महत्वपूर्ण अवधारणा है।

आइए एक संदर्भ के रूप में एक सांख्यिकीय रूप से टाइप की गई भाषा से एक कोड स्निपेट लें:

➜
String str = "";
str = 2.4;

यह कोई खबर नहीं है कि ऐसी भाषा अपनी वस्तुओं और चर के प्रकारों की परवाह करती है। कहा जा रहा है, ऊपर वाले जैसा कोड एक त्रुटि देगा।

रूबी, कई अन्य भाषाओं जैसे कि जावास्क्रिप्ट, पायथन और ऑब्जेक्टिव-सी की तरह, इस बात पर अधिक ध्यान नहीं देती है कि आप अपनी वस्तुओं के लिए किस प्रकार को लक्षित कर रहे हैं।

रूबी में वही कोड सफलतापूर्वक चलेगा, जैसा कि नीचे देखा गया है:

➜  irb
str = ""
str = 2.4
puts str # prints 2.4

यह संभव है क्योंकि रूबी का दुभाषिया जानता है कि कैसे गतिशील रूप से एक प्रकार से दूसरे प्रकार में स्विच करें।

हालाँकि, दुभाषिया क्या अनुमति देता है इसकी एक सीमा है। उदाहरण के लिए, निम्नलिखित कोड परिवर्तन लें:

➜  irb
val = "6.0"
result = val + 2.0
puts result

यह, बदले में, निम्न त्रुटि उत्पन्न करेगा:

त्रुटि:फ्लोट का स्ट्रिंग में कोई अंतर्निहित रूपांतरण नहीं

उदाहरण के लिए, जावास्क्रिप्ट के साथ समान कोड चलाना ठीक चलेगा।

कहानी का नैतिक:रूबी वास्तव में गतिशील रूप से अनुमान लगाता है; लेकिन, अन्य प्रमुख गतिशील भाषाओं के विपरीत, यह सब कुछ स्वीकार नहीं करेगा। इसके प्रति जागरूक रहें!

और यहीं पर टाइप चेकर्स (चाहे स्थिर या गतिशील) उपयोगी हो जाते हैं।

आरबीएस बनाम शर्बत

ठीक है, मुझे गतिशील बनाम स्थिर चीज़ के बारे में आपकी बात मिल गई है। लेकिन, शर्बत का क्या? क्या यह पदावनत हो जाएगा?

बिल्कुल भी नहीं। आरबीएस और शर्बत के बीच प्राथमिक (और शायद सबसे महत्वपूर्ण) अंतर यह है कि पूर्व सिर्फ एक भाषा है, जबकि बाद वाला एक पूर्ण प्रकार का चेकर है।

रूबी टीम आरबीएस के मुख्य लक्ष्य को संरचना का वर्णन करने के रूप में बताती है आपके कोड का। यह टाइप चेकिंग नहीं करेगा, बल्कि उस संरचना को परिभाषित करेगा जो चेकर्स टाइप करता है (जैसे सॉर्बेट या कोई अन्य) चेक टाइप करने के लिए उपयोग कर सकता है। कोड संरचना एक नए फ़ाइल एक्सटेंशन में संग्रहीत है — .rbs

इसे जांचने के लिए, निम्नलिखित रूबी वर्ग को एक उदाहरण के रूप में लेते हैं:

class Super
    def initialize(val)
      @val = val
    end
 
 
    def val?
      @val
    end
end
 
class Test < Super
  def initialize(val, flag)
    super(val)
    @flag = flag
  end
 
  def flag?
    @flag
  end
end

यह रूबी में एक साधारण विरासत का प्रतिनिधित्व करता है। यहां ध्यान देने वाली दिलचस्प बात यह है कि हम flag को छोड़कर, कक्षा में उपयोग की जाने वाली प्रत्येक विशेषता के प्रकारों का अनुमान नहीं लगा सकते हैं। ।

चूंकि flag डिफ़ॉल्ट मान के साथ इनिशियलाइज़ किया जाता है, डेवलपर और टाइप चेकर दोनों ही प्रकार का अनुमान लगा सकते हैं ताकि आगे के दुरुपयोग को रोका जा सके।

निम्नलिखित आरबीएस प्रारूप में उपरोक्त वर्ग का उचित प्रतिनिधित्व होगा:

class Super
  attr_reader val : untyped
 
  def initialize : (val: untyped) -> void
end
 
class Test < Super
  attr_reader flag : bool
 
  def initialize : (val: untyped, ?flag: bool) -> void
  def flag? : () -> bool
end

इसे पचाने के लिए कुछ समय निकालें। यह एक घोषणा भाषा है, इसलिए RBS फ़ाइल में केवल हस्ताक्षर दिखाई दे सकते हैं। आसान है, है ना?

चाहे वह किसी सीएलआई उपकरण (उस पर बाद में अधिक) द्वारा स्वत:उत्पन्न हुआ हो या आपके द्वारा, किसी प्रकार को untyped के रूप में एनोटेट करना अधिक सुरक्षित है जब इसका अनुमान नहीं लगाया जा सकता है।

यदि आप val . के प्रकार के बारे में सुनिश्चित हैं , उदाहरण के लिए, आपकी RBS मैपिंग को निम्न पर स्विच किया जा सकता है:

class Super
  attr_reader val : Integer
 
  def initialize : (val: Integer) -> void
end

यह भी ध्यान रखना महत्वपूर्ण है कि रूबी और शर्बत दोनों टीमें आरबीएस के निर्माण और सुधार की दिशा में काम कर रही थीं (और अभी भी हैं)। सालों से टाइप चेकिंग के साथ सॉर्बेट टीम के अनुभव ने रूबी टीम को इस प्रोजेक्ट के साथ बहुत सी चीजों को ठीक करने में मदद की।

आरबीएस और आरबीआई फाइलों के बीच इंटरऑपरेबिलिटी अभी भी विकास के अधीन है। लक्ष्य यह है कि सॉर्बेट और किसी भी अन्य चेकर टूल का पालन करने के लिए आधिकारिक और केंद्रीकृत आधार हो।

आरबीएस सीएलआई टूल

आरबीएस विकसित करते समय रूबी टीम के पास एक महत्वपूर्ण विचार था कि एक सीएलआई उपकरण शिप किया जाए जो डेवलपर्स को इसे आज़माने और इसका उपयोग करने का तरीका सीखने में मदद कर सके। इसे rbs . कहा जाता है और डिफ़ॉल्ट रूप से रूबी 3 के साथ आता है। यदि आपने अभी भी अपने रूबी संस्करण को अपग्रेड नहीं किया है, तो आप इसके रत्न को सीधे अपने प्रोजेक्ट में भी जोड़ सकते हैं:

➜  gem install rbs

कमांड rbs help उपलब्ध कमांड के साथ कमांड का उपयोग दिखाएगा।

उपलब्ध आदेशों की सूची

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

➜  rbs ancestors ::String
::String
::Comparable
::Object
::Kernel
::BasicObject

कमांड methods किसी दिए गए वर्ग की सभी विधि संरचनाएँ प्रदर्शित करता है:

➜  rbs methods ::String
! (public)
!= (public)
!~ (public)
...
Array (private)
Complex (private)
Float (private)
...
autoload? (private)
b (public)
between? (public)
...

एक विशिष्ट विधि संरचना देखना चाहते हैं? method के लिए जाएं :

➜  rbs method ::String split
::String#split
  defined_in: ::String
  implementation: ::String
  accessibility: public
  types:
      (?::Regexp | ::string pattern, ?::int limit) -> ::Array[::String]
    | (?::Regexp | ::string pattern, ?::int limit) { (::String) -> void } -> self

आज आरबीएस से शुरुआत करने वालों के लिए कमांड prototype पहले से मौजूद कक्षाओं के लिए मचान प्रकारों के साथ बहुत मदद कर सकता है। कमांड आरबीएस फाइलों के प्रोटोटाइप तैयार करता है।

आइए पिछला Test < Super take लें विरासत उदाहरण और कोड को appsignal.rb . नामक फ़ाइल में सहेजें . फिर, निम्न कमांड चलाएँ:

➜  rbs prototype rb appsignal.rb

चूंकि कमांड rb . के लिए अनुमति देता है , आरबीआई , और रनटाइम जनरेटर, आपको prototype के ठीक बाद विशिष्ट प्रकार की फ़ाइल प्रदान करने की आवश्यकता है जिसे आप मचा रहे हैं कमांड, उसके बाद फ़ाइल पथनाम।

निष्पादन का परिणाम निम्नलिखित है:

class Super
  def initialize: (untyped val) -> untyped
 
  def val?: () -> untyped
end
 
class Test < Super
  def initialize: (untyped val, ?flag: bool flag) -> untyped
 
  def flag?: () -> untyped
end

हमारे पहले आरबीएस संस्करण के समान ही। जैसा कि पहले उल्लेख किया गया है, टूल untyped as के रूप में चिह्नित करता है किसी भी प्रकार का अनुमान नहीं लगाया जा सकता है।

यह विधि रिटर्न के लिए भी मायने रखता है। flag के वापसी प्रकार पर ध्यान दें परिभाषा। एक डेवलपर के रूप में, आप शायद सुनिश्चित हैं कि विधि हमेशा एक बूलियन लौटाती है, लेकिन रूबी की गतिशील प्रकृति के कारण, टूल 100% यह कहने में असमर्थ है कि ऐसा है।

और वह तब होता है जब एक और रूबी 3 बच्चा बचाव के लिए आता है:टाइपप्रोफ।

टाइपप्रोफ टूल

टाइपप्रोफ रूबी के लिए एक प्रकार का विश्लेषण उपकरण है जिसे कुछ सिंटैक्स ट्री व्याख्या के शीर्ष पर बनाया गया था।

अभी भी प्रयोगात्मक होने के बावजूद, जब यह समझने की बात आती है कि आपका कोड क्या करने की कोशिश कर रहा है तो यह बहुत शक्तिशाली साबित हुआ है।

यदि आपके पास अभी तक रूबी 3 नहीं है, तो बस अपने प्रोजेक्ट में रत्न जोड़ें:

➜  gem install typeprof

अब, चलिए वही appsignal.rb चलाते हैं इसके खिलाफ फाइल करें:

➜  typeprof appsignal.rb

यह आउटपुट है:

# Classes
class Super
  @val: untyped
  def initialize: (untyped) -> untyped
  def val?: -> untyped
end
 
class Test < Super
  @val: untyped
  @flag: true
 
  def initialize: (untyped, ?flag: true) -> true
  def flag?: -> true
end

ध्यान दें कि कैसे flag अब मैप किया गया है। यह केवल इसलिए संभव है क्योंकि, आरबीएस प्रोटोटाइप के विपरीत, टाइपप्रोफ विधि के शरीर को यह समझने की कोशिश कर रहा है कि उस विशिष्ट चर पर कौन सी क्रियाएं की जा रही हैं। चूंकि यह इस चर में किसी भी प्रत्यक्ष परिवर्तन की पहचान नहीं कर सका, इसलिए TypeProf ने बूलियन के रूप में विधि वापसी को सुरक्षित रूप से मैप किया।

उदाहरण के लिए, इस बात पर विचार करें कि टाइपप्रोफ के पास अन्य वर्गों तक पहुंच होगी जो Test को तत्काल और उपयोग करते हैं कक्षा। इसके साथ, यह आपके कोड में और भी गहराई तक जा सकता है और इसकी भविष्यवाणियों को सुदृढ़ कर सकता है। मान लें कि निम्न कोड स्निपेट appsignal.rb . के अंत में जोड़ा गया है फ़ाइल:

testSub = Test.new("My value", "My value" == "")
testSup = Super.new("My value")

और यह कि आपने initialize . को बदल दिया है निम्नलिखित के लिए विधि हस्ताक्षर:

def initialize(val, flag)

जब आप कमांड को फिर से चलाते हैं, तो यह आउटपुट होना चाहिए:

# Classes
class Super
  @val: String
 
  def initialize: (String) -> String
  def val?: -> String
end
 
class Test < Super
  @val: String
  @flag: bool
 
  def initialize: (String val, bool flag) -> bool
  def flag?: -> bool
end

बहुत बढ़िया!

टाइपप्रोफ विरासत में मिली विशेषताओं से बहुत अच्छी तरह से निपट नहीं सकता है। इसलिए हम एक नया Super इंस्टेंट कर रहे हैं वस्तु। अन्यथा, वह val नहीं मिलेगा एक String है ।

TypeProf का प्रमुख लाभ इसकी सुरक्षा है। जब भी यह निश्चित रूप से कुछ पता लगाने में असमर्थ होता है, तो untyped वापस कर दिया जाएगा।

आंशिक RBS विशिष्टता

आधिकारिक डॉक्स से एक महत्वपूर्ण चेतावनी में कहा गया है कि, हालांकि टाइपप्रोफ बहुत शक्तिशाली है, आपको इसकी सीमाओं के बारे में पता होना चाहिए कि यह आरबीएस कोड के संदर्भ में क्या उत्पन्न कर सकता है और क्या नहीं।

उदाहरण के लिए, रूबी डेवलपर्स के बीच एक सामान्य अभ्यास विधि ओवरलोडिंग है जिसमें आप एक विधि के विभिन्न व्यवहारों को उसके तर्कों के आधार पर लागू करते हैं।

विचार करें कि एक नई विधि spell Super . में जोड़ा जाता है क्लास, जो एक Integer returns देता है या एक String पैरामीटर प्रकार के आधार पर:

def spell(val)
  if val.is_a?(String)
    ""
  else
    0
  end
end

RBS आपको संघ प्रकार (एक मान जो कई संभावित प्रकारों का प्रतिनिधित्व करता है) के माध्यम से ओवरलोडिंग से निपटने की अनुमति देकर इस अभ्यास को अपनाता है:

def spell: (String) -> String | (Integer) -> Integer

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

इसके लिए आपको कमांड के अंत में RBS फ़ाइल पथ जोड़ना होगा:

typeprof appsignal.rb appsignal.rbs

नीचे नया आउटपुट देखें:

class Super
  ...
  def spell: (untyped val) -> (Integer | String)
end

साथ ही, हम रनटाइम के दौरान Kernel#p . के माध्यम से वास्तविक प्रकारों को भी सत्यापित कर सकते हैं यह जांचने के लिए कि appsignal.rb के अंत में अगली दो पंक्तियों को जोड़कर ओवरलोडिंग काम कर रही है या नहीं फ़ाइल:

p testSup.spell(42)
p testSup.spell("str")

यह आउटपुट होना चाहिए:

# Revealed types
#  appsignal.rb:11 #=> Integer
#  appsignal.rb:12 #=> String
 
...

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

बतख टाइपिंग

आपने इसके बारे में पहले सुना है। यदि एक रूबी वस्तु वह सब कुछ करती है जो एक बतख करता है, तो वह एक बतख है।

जैसा कि हमने देखा है, रूबी इस बात की परवाह नहीं करती है कि आपकी वस्तुओं का क्या मतलब है। प्रकार गतिशील रूप से और साथ ही वस्तु संदर्भ बदल सकते हैं।

हालांकि मददगार, बतख टाइपिंग मुश्किल हो सकती है। आइए एक उदाहरण देखें।

मान लीजिए कि, अब से, val विशेषता जिसे आपने Super . के लिए घोषित किया है क्लास, जो एक String है , हमेशा एक पूर्णांक में परिवर्तनीय होना चाहिए।

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

interface _IntegerConvertible
   def to_int: () -> Integer
end

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

class Super
  attr_reader val : _IntegerConvertible
 
  def initialize : (val: _IntegerConvertible) -> void
end

इस इंटरफ़ेस को लागू करने वाले ठोस वर्ग या मॉड्यूल को यह सुनिश्चित करना होगा कि उचित सत्यापन किया गया है।

मेटाप्रोग्रामिंग

शायद रूबी की सबसे गतिशील विशेषताओं में से एक कोड बनाने की क्षमता है जो रनटाइम के दौरान स्वयं कोड बनाता है। वह मेटाप्रोग्रामिंग है।

चीजों की अनिश्चित प्रकृति के कारण, आरबीएस सीएलआई उपकरण मेटाप्रोग्रामिंग कोड से आरबीएस उत्पन्न करने में सक्षम नहीं है।

आइए निम्नलिखित स्निपेट को एक उदाहरण के रूप में लें:

class Test
    define_method :multiply do |*args|
        args.inject(1, :*)
    end
end
 
p Test.new.multiply(2, 3, 5)

यह वर्ग multiply नामक एक विधि को परिभाषित करता है रनटाइम पर और तर्कों को इंजेक्ट करने का निर्देश देता है और प्रत्येक को पिछले परिणाम से गुणा करता है।

एक बार जब आप RBS prototypeचलाते हैं कमांड, यह आउटपुट होना चाहिए:

class Test
end

आपके मेटाप्रोग्रामिंग कोड की जटिलता के आधार पर, TypeProf अभी भी इससे कुछ निकालने की पूरी कोशिश करेगा। लेकिन इसकी हमेशा गारंटी नहीं होती है।

याद रखें, आप हमेशा अपनी खुद की टाइप मैपिंग को आरबीएस फाइल में जोड़ सकते हैं और टाइपप्रोफ पहले से उनका पालन करेगा। यह मेटाप्रोग्रामिंग के लिए भी मान्य है।

नवीनतम रिपॉजिटरी परिवर्तनों के साथ अपडेट रहना भी महत्वपूर्ण है क्योंकि टीम लगातार नई सुविधाएँ जारी कर रही है, जिसमें मेटाप्रोग्रामिंग पर अपडेट भी शामिल हो सकते हैं।

कहा जा रहा है, यदि आपके कोडबेस में कुछ प्रकार के मेटाप्रोग्रामिंग शामिल हैं, तो इन उपकरणों से सावधान रहें। आँख बंद करके उनका उपयोग न करें!

रैपिंग अप

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

इसलिए, उस पर अधिक जानकारी के लिए आधिकारिक दस्तावेज़ों को देखना सुनिश्चित करें।

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

आप क्या कहते हैं? क्या आपने इसे आजमाया है? आरबीएस पर आपके क्या विचार हैं?

पी.एस. यदि आप रूबी मैजिक की पोस्ट प्रेस से बाहर होते ही पढ़ना चाहते हैं, तो हमारे रूबी मैजिक न्यूजलेटर की सदस्यता लें और एक भी पोस्ट मिस न करें!


  1. आरबीएस को समझना, रूबीज न्यू टाइप एनोटेशन सिस्टम

    रूबी के लिए आरबीएस एक नए प्रकार की सिंटैक्स प्रारूप भाषा का नाम है। RBS आपको .rbs नामक एक नए एक्सटेंशन के साथ फाइलों में अपने रूबी कोड में टाइप एनोटेशन जोड़ने देता है . वे इस तरह दिखते हैं: class MyClass def my_method : (my_param: String) -> String end RBS के साथ टाइप एनोटेशन प्रदान करने से आ

  1. रूबी में 9 नई सुविधाएँ 2.6

    रूबी का एक नया संस्करण नई सुविधाओं और प्रदर्शन में सुधार के साथ आ रहा है। क्या आप परिवर्तनों के साथ बने रहना चाहेंगे? आइए एक नज़र डालते हैं! अंतहीन रेंज रूबी 2.5 और पुराने संस्करण पहले से ही अंतहीन श्रेणी के एक रूप का समर्थन करते हैं (Float::INFINITY के साथ) ), लेकिन रूबी 2.6 इसे अगले स्तर पर ले

  1. कुंजीपटल की गलत अक्षर टाइपिंग को कैसे ठीक करें

    क्या आपके साथ ऐसा हो रहा है कि जब आप कीबोर्ड पर p दबाते हैं, तो स्क्रीन पर आउटपुट कुछ और अक्षर या प्रतीक होता है? यह आपको अजीब लग सकता है लेकिन यह समझ लें कि वर्षों तक इतने विश्वसनीय होने के बावजूद कीबोर्ड कुछ समय बाद हार्डवेयर की खराबी का अनुभव कर सकते हैं। कुछ कुंजियाँ परेशान करने वाली आवाजें उत्प