डी प्रोग्रामिंग - रंग

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

रंग डी। डी। के स्लाइस का एक अभिन्न हिस्सा हैं जो सबसे शक्तिशाली रेंज रैंडमऐसट्रींगे का कार्यान्वयन होता है, और फोबोस में कई रेंज फीचर हैं। कई फ़ोबोस एल्गोरिदम अस्थायी श्रेणी की वस्तुएँ लौटाते हैं। उदाहरण के लिए, फ़िल्टर () उन तत्वों को चुनता है जो निम्नलिखित कोड में 10 से अधिक हैं, वास्तव में एक श्रेणी वस्तु देता है, न कि एक सरणी।

संख्या सीमा

नंबर रेंज का उपयोग आमतौर पर किया जाता है और ये नंबर रेंज टाइप इंट की होती है। संख्या सीमाओं के लिए कुछ उदाहरण नीचे दिखाए गए हैं -

// Example 1 
foreach (value; 3..7)  

// Example 2 
int[] slice = array[5..10];

फोबोस रेंजेस

संरचना और वर्ग इंटरफेस से संबंधित रेंज फ़ोबोस रेंज हैं। फोबोस आधिकारिक रनटाइम और मानक पुस्तकालय है जो डी भाषा संकलक के साथ आता है।

विभिन्न प्रकार की श्रेणियां हैं जिनमें शामिल हैं -

  • InputRange
  • ForwardRange
  • BidirectionalRange
  • RandomAccessRange
  • OutputRange

निवेश सीमा

सबसे सरल रेंज इनपुट रेंज है। अन्य श्रेणियां उस सीमा के ऊपर अधिक आवश्यकताएं लाती हैं जो वे पर आधारित हैं। तीन कार्य हैं जिन्हें InputRange की आवश्यकता है -

  • empty- यह निर्दिष्ट करता है कि क्या रेंज खाली है; जब सीमा खाली मानी जाती है तो इसे सही लौटना चाहिए; गलत है अन्यथा।

  • front - यह सीमा की शुरुआत में तत्व तक पहुंच प्रदान करता है।

  • popFront() - यह पहले तत्व को हटाकर शुरू से ही सीमा को छोटा कर देता है।

उदाहरण

import std.stdio; 
import std.string; 
 
struct Student { 
   string name; 
   int number; 
   
   string toString() const { 
      return format("%s(%s)", name, number); 
   } 
}
  
struct School { 
   Student[] students; 
}
struct StudentRange {
   Student[] students; 
   
   this(School school) { 
      this.students = school.students; 
   } 
   @property bool empty() const { 
      return students.length == 0; 
   } 
   @property ref Student front() { 
      return students[0]; 
   } 
   void popFront() { 
      students = students[1 .. $]; 
   } 
}

void main() { 
   auto school = School([ Student("Raj", 1), Student("John", 2), Student("Ram", 3)]);
   auto range = StudentRange(school); 
   writeln(range);  
   
   writeln(school.students.length);
   
   writeln(range.front); 
   
   range.popFront;  
   
   writeln(range.empty); 
   writeln(range); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[Raj(1), John(2), Ram(3)] 
3 
Raj(1) 
false 
[John(2), Ram(3)]

ForwardRange

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

import std.array; 
import std.stdio; 
import std.string; 
import std.range;

struct FibonacciSeries { 
   int first = 0; 
   int second = 1; 
   enum empty = false;   //  infinite range  
   
   @property int front() const { 
      return first; 
   } 
   void popFront() { 
      int third = first + second; 
      first = second; 
      second = third; 
   }
   @property FibonacciSeries save() const { 
      return this; 
   } 
}
  
void report(T)(const dchar[] title, const ref T range) {
   writefln("%s: %s", title, range.take(5)); 
} 

void main() { 
   auto range = FibonacciSeries(); 
   report("Original range", range);
   
   range.popFrontN(2); 
   report("After removing two elements", range); 
   
   auto theCopy = range.save; 
   report("The copy", theCopy);
   
   range.popFrontN(3); 
   report("After removing three more elements", range); 
   report("The copy", theCopy); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Original range: [0, 1, 1, 2, 3] 
After removing two elements: [1, 2, 3, 5, 8] 
The copy: [1, 2, 3, 5, 8] 
After removing three more elements: [5, 8, 13, 21, 34] 
The copy: [1, 2, 3, 5, 8]

BidirectionalRange

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

उदाहरण

import std.array; 
import std.stdio; 
import std.string; 

struct Reversed { 
   int[] range; 
   
   this(int[] range) { 
      this.range = range; 
   } 
   @property bool empty() const { 
      return range.empty; 
   }
   @property int front() const { 
      return range.back;  //  reverse 
   }
   @property int back() const { 
      return range.front; // reverse 
   } 
   void popFront() { 
      range.popBack(); 
   }
   void popBack() { 
      range.popFront(); 
   } 
} 
 
void main() { 
   writeln(Reversed([ 1, 2, 3])); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[3, 2, 1]

अनंत रैंडमअगलेरेंज

फ़ॉरवर्डरेंज की तुलना में opIndex () अतिरिक्त रूप से आवश्यक है। इसके अलावा, एक खाली फ़ंक्शन का मान संकलन समय पर गलत के रूप में जाना जाता है। एक सरल उदाहरण के साथ समझाया गया है कि वर्गों को नीचे दिखाया गया है।

import std.array; 
import std.stdio; 
import std.string; 
import std.range; 
import std.algorithm; 

class SquaresRange { 
   int first;  
   this(int first = 0) { 
      this.first = first; 
   }
   enum empty = false; 
   @property int front() const { 
      return opIndex(0); 
   }
   void popFront() { 
      ++first; 
   }
   @property SquaresRange save() const { 
      return new SquaresRange(first); 
   }
   int opIndex(size_t index) const { 
      /* This function operates at constant time */ 
      immutable integerValue = first + cast(int)index; 
      return integerValue * integerValue; 
   } 
}
  
bool are_lastTwoDigitsSame(int value) { 
   /* Must have at least two digits */ 
   if (value < 10) { 
      return false; 
   } 
   
   /* Last two digits must be divisible by 11 */ 
   immutable lastTwoDigits = value % 100; 
   return (lastTwoDigits % 11) == 0; 
} 
 
void main() { 
   auto squares = new SquaresRange(); 
   
   writeln(squares[5]);
   
   writeln(squares[10]); 
   
   squares.popFrontN(5); 
   writeln(squares[0]); 
   
   writeln(squares.take(50).filter!are_lastTwoDigitsSame); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

25 
100 
25 
[100, 144, 400, 900, 1444, 1600, 2500]

परिमित रैंडम असफलता

द्विदिश श्रेणी की तुलना में opIndex () और लंबाई अतिरिक्त रूप से आवश्यक है। यह विस्तृत उदाहरण की मदद से समझाया गया है जो पहले इस्तेमाल किए गए फाइबोनैचि श्रृंखला और वर्गों रेंज उदाहरण का उपयोग करता है। यह उदाहरण सामान्य डी संकलक पर अच्छा काम करता है लेकिन ऑनलाइन संकलक पर काम नहीं करता है।

उदाहरण

import std.array; 
import std.stdio; 
import std.string; 
import std.range; 
import std.algorithm; 

struct FibonacciSeries { 
   int first = 0; 
   int second = 1; 
   enum empty = false;   //  infinite range  
   
   @property int front() const { 
      return first;
   }
   void popFront() { 
      int third = first + second; 
      first = second; 
      second = third; 
   }
   @property FibonacciSeries save() const { 
      return this; 
   } 
}
  
void report(T)(const dchar[] title, const ref T range) { 
   writefln("%40s: %s", title, range.take(5)); 
}
  
class SquaresRange { 
   int first;  
   this(int first = 0) { 
      this.first = first; 
   } 
   enum empty = false; 
   @property int front() const { 
      return opIndex(0); 
   }
   void popFront() { 
      ++first; 
   }
   @property SquaresRange save() const { 
      return new SquaresRange(first); 
   } 
   int opIndex(size_t index) const { 
      /* This function operates at constant time */ 
      immutable integerValue = first + cast(int)index; 
      return integerValue * integerValue; 
   } 
}
  
bool are_lastTwoDigitsSame(int value) { 
   /* Must have at least two digits */ 
   if (value < 10) { 
      return false; 
   }
   
   /* Last two digits must be divisible by 11 */ 
   immutable lastTwoDigits = value % 100; 
   return (lastTwoDigits % 11) == 0; 
}
  
struct Together { 
   const(int)[][] slices;  
   this(const(int)[][] slices ...) { 
      this.slices = slices.dup;  
      clearFront(); 
      clearBack(); 
   }
   private void clearFront() { 
      while (!slices.empty && slices.front.empty) { 
         slices.popFront(); 
      } 
   } 
   private void clearBack() { 
      while (!slices.empty && slices.back.empty) { 
         slices.popBack(); 
      } 
   }
   @property bool empty() const { 
      return slices.empty; 
   } 
   @property int front() const { 
      return slices.front.front; 
   }
   void popFront() { 
      slices.front.popFront(); 
      clearFront(); 
   }
   @property Together save() const { 
      return Together(slices.dup); 
   } 
   @property int back() const { 
      return slices.back.back; 
   } 
   void popBack() { 
      slices.back.popBack(); 
      clearBack(); 
   }
   @property size_t length() const { 
      return reduce!((a, b) => a + b.length)(size_t.init, slices); 
   }
   int opIndex(size_t index) const { 
      /* Save the index for the error message */ 
      immutable originalIndex = index;  

      foreach (slice; slices) { 
         if (slice.length > index) { 
            return slice[index];  
         } else { 
            index -= slice.length; 
         } 
      } 
      throw new Exception( 
         format("Invalid index: %s (length: %s)", originalIndex, this.length));
   } 
}
void main() { 
   auto range = Together(FibonacciSeries().take(10).array, [ 777, 888 ],
      (new SquaresRange()).take(5).array); 
   writeln(range.save); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 777, 888, 0, 1, 4, 9, 16]

OutputRange

OutputRange स्ट्रीम किए गए तत्व आउटपुट का प्रतिनिधित्व करता है, जो स्टडआउट को वर्ण भेजने के समान है। OutputRange को पुट (श्रेणी, तत्व) ऑपरेशन के लिए समर्थन की आवश्यकता होती है। put () std.range मॉड्यूल में परिभाषित एक फ़ंक्शन है। यह संकलन समय पर सीमा और तत्व की क्षमताओं को निर्धारित करता है और तत्वों का उत्पादन करने के लिए उपयोग करने के लिए सबसे उपयुक्त विधि का उपयोग करता है। एक सरल उदाहरण नीचे दिखाया गया है।

import std.algorithm; 
import std.stdio; 
 
struct MultiFile { 
   string delimiter;
   File[] files;
   
   this(string delimiter, string[] fileNames ...) { 
      this.delimiter = delimiter; 

      /* stdout is always included */ 
      this.files ~= stdout; 

      /* A File object for each file name */ 
      foreach (fileName; fileNames) { 
         this.files ~= File(fileName, "w"); 
      } 
   }
   void put(T)(T element) { 
      foreach (file; files) { 
         file.write(element, delimiter); 
      } 
   }
}
void main() { 
   auto output = MultiFile("\n", "output_0", "output_1"); 
   copy([ 1, 2, 3], output);  
   copy([ "red", "blue", "green" ], output); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[1, 2, 3] 
["red", "blue", "green"]