पास्कल - कक्षाएं

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

क्लास को ऑब्जेक्ट के रूप में लगभग उसी तरह से परिभाषित किया जाता है, लेकिन ऑब्जेक्ट के बजाय ऑब्जेक्ट के लिए एक पॉइंटर होता है। तकनीकी रूप से, इसका मतलब है कि कक्षा को एक कार्यक्रम के ढेर पर आवंटित किया गया है, जबकि ऑब्जेक्ट को स्टैक पर आवंटित किया गया है। दूसरे शब्दों में, जब आप किसी चर को ऑब्जेक्ट प्रकार घोषित करते हैं, तो यह स्टैक पर ऑब्जेक्ट के आकार के बराबर जगह ले लेगा, लेकिन जब आप क्लास प्रकार के एक वैरिएबल की घोषणा करते हैं, तो यह हमेशा एक पॉइंटर का आकार लेगा ढेर पर। वास्तविक वर्ग का डेटा ढेर पर होगा।

पास्कल कक्षाएं परिभाषित करना

एक वर्ग को ऑब्जेक्ट के रूप में उसी तरह घोषित किया जाता है, जिस प्रकार की घोषणा। वर्ग घोषणा का सामान्य रूप इस प्रकार है -

type class-identifier = class  
   private
      field1 : field-type;  
      field2 : field-type;  
        ...
   
   public
      constructor create();
      procedure proc1;  
      function f1(): function-type;
end;  
var classvar : class-identifier;

महत्वपूर्ण बिंदुओं पर ध्यान देने योग्य है -

  • कक्षा की परिभाषाएं केवल कार्यक्रम के घोषित घोषणा भाग के तहत होनी चाहिए।

  • का उपयोग कर एक वर्ग को परिभाषित किया गया है class कीवर्ड।

  • फ़ील्ड डेटा आइटम हैं जो कक्षा के प्रत्येक उदाहरण में मौजूद हैं।

  • किसी वर्ग की परिभाषा के भीतर तरीके घोषित किए जाते हैं।

  • एक पूर्वनिर्धारित कंस्ट्रक्टर है जिसे कहा जाता है Createजड़ वर्ग में। प्रत्येक अमूर्त वर्ग और प्रत्येक ठोस वर्ग रूट का वंशज है, इसलिए सभी वर्गों में कम से कम एक निर्माता है।

  • एक पूर्वनिर्धारित विध्वंसक कहा जाता है Destroyजड़ वर्ग में। हर अमूर्त वर्ग और हर ठोस वर्ग रूट का वंशज है, इसलिए, सभी वर्गों में कम से कम एक विध्वंसक है।

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

type
   Rectangle = class
   private
      length, width: integer;
   
   public
      constructor create(l, w: integer);
      procedure setlength(l: integer);
      function getlength(): integer;
      procedure setwidth(w: integer);
      function getwidth(): integer;
      procedure draw;
end;

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

  • आपको कक्षाओं का उपयोग करने के लिए {$ मोड objfpc} निर्देश को शामिल करने की आवश्यकता होगी।

  • आपको कंस्ट्रक्टर का उपयोग करने के लिए {$ m +} निर्देश को शामिल करना होगा।

  • क्लास इंस्टेंटेशन ऑब्जेक्ट इंस्टेंटेशन से अलग है। केवल चर घोषित करने से उदाहरण के लिए जगह नहीं बनती है, आप मेमोरी को आवंटित करने के लिए निर्माणकर्ता का उपयोग करेंगे।

यहाँ पूरा उदाहरण है -

{$mode objfpc} // directive to be used for defining classes
{$m+}		   // directive to be used for using constructor

program exClass;
type
   Rectangle = class
   private
      length, width: integer;
   
   public
      constructor create(l, w: integer);
      procedure setlength(l: integer);
      
      function getlength(): integer;
      procedure setwidth(w: integer);
      
      function getwidth(): integer;
      procedure draw;
end;
var
   r1: Rectangle;

constructor Rectangle.create(l, w: integer);
begin
   length := l;
   width := w;
end;

procedure Rectangle.setlength(l: integer);
begin
   length := l;
end;

procedure Rectangle.setwidth(w: integer);
begin
   width :=w;
end;

function Rectangle.getlength(): integer;
begin
   getlength := length;
end;

function Rectangle.getwidth(): integer;
begin
   getwidth := width;
end;

procedure Rectangle.draw;
var
   i, j: integer;
begin
   for i:= 1 to length do
   begin
      for j:= 1 to width do
         write(' * ');
      writeln;
   end;
end;

begin
   r1:= Rectangle.create(3, 7);
   
   writeln(' Draw Rectangle: ', r1.getlength(), ' by ' , r1.getwidth());
   r1.draw;
   r1.setlength(4);
   r1.setwidth(6);
   
   writeln(' Draw Rectangle: ', r1.getlength(), ' by ' , r1.getwidth());
   r1.draw;
end.

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

Draw Rectangle: 3 by 7
* * * * * * *
* * * * * * *
* * * * * * *
Draw Rectangle: 4 by 6
* * * * * * 
* * * * * * 
* * * * * * 
* * * * * *

कक्षा सदस्यों की दृश्यता

दृश्यता कक्षा के सदस्यों की पहुंच को इंगित करती है। पास्कल वर्ग के सदस्यों की दृश्यता पांच प्रकार की होती है -

अनु क्रमांक दृश्यता और पहुंच
1

Public

ये सदस्य हमेशा सुलभ होते हैं।

2

Private

इन सदस्यों को केवल उस मॉड्यूल या यूनिट में एक्सेस किया जा सकता है जिसमें कक्षा की परिभाषा है। उन्हें कक्षा के तरीकों से या उनके बाहर से पहुँचा जा सकता है।

3

Strict Private

इन सदस्यों को केवल कक्षा के तरीकों से ही एक्सेस किया जा सकता है। एक ही इकाई में अन्य वर्ग या अवरोही वर्ग उन तक नहीं पहुँच सकते हैं।

4

Protected

यह निजी के समान है, सिवाय इसके, ये सदस्य अवरोही प्रकारों के लिए सुलभ हैं, भले ही वे अन्य मॉड्यूल में लागू किए गए हों।

5

Published

यह एक सार्वजनिक के समान है, लेकिन कंपाइलर टाइप जानकारी उत्पन्न करता है जो इन वर्गों के स्वचालित स्ट्रीमिंग के लिए आवश्यक है यदि कंपाइलर {$ M +} राज्य में है। प्रकाशित अनुभाग में परिभाषित फ़ील्ड्स वर्ग प्रकार के होने चाहिए।

पास्कल क्लासेस के लिए कंस्ट्रक्टर और डिस्ट्रक्टर्स

कंस्ट्रक्टर विशेष विधियाँ हैं, जो जब भी किसी ऑब्जेक्ट को बनाया जाता है, स्वचालित रूप से कहा जाता है। इसलिए हम निर्माण कार्यों के माध्यम से कई चीजों को शुरू करके इस व्यवहार का पूरा लाभ उठाते हैं।

पास्कल एक रचनाकार को परिभाषित करने के लिए क्रिएट () नामक एक विशेष फ़ंक्शन प्रदान करता है। कंस्ट्रक्टर फंक्शन में आप जितने चाहें उतने तर्क पारित कर सकते हैं।

उदाहरण के बाद, बुक्स नामक एक वर्ग के लिए एक कंस्ट्रक्टर बनाया जाएगा और यह ऑब्जेक्ट निर्माण के समय पुस्तक के लिए मूल्य और शीर्षक को इनिशियलाइज़ करेगा।

program classExample;

{$MODE OBJFPC} //directive to be used for creating classes
{$M+} //directive that allows class constructors and destructors
type
   Books = Class 
   private 
      title : String; 
      price: real;
   
   public
      constructor Create(t : String; p: real); //default constructor
      
      procedure setTitle(t : String); //sets title for a book
      function getTitle() : String; //retrieves title
      
      procedure setPrice(p : real); //sets price for a book
      function getPrice() : real; //retrieves price
      
      procedure Display(); // display details of a book
end;
var
   physics, chemistry, maths: Books;

//default constructor 
constructor Books.Create(t : String; p: real);
begin
   title := t;
   price := p;
end;

procedure Books.setTitle(t : String); //sets title for a book
begin
   title := t;
end;

function Books.getTitle() : String; //retrieves title
begin
   getTitle := title;
end;

procedure Books.setPrice(p : real); //sets price for a book
begin
   price := p;
end;

function Books.getPrice() : real; //retrieves price
begin
   getPrice:= price;
end;

procedure Books.Display();
begin
   writeln('Title: ', title);
   writeln('Price: ', price:5:2);
end;

begin 
   physics := Books.Create('Physics for High School', 10);
   chemistry := Books.Create('Advanced Chemistry', 15);
   maths := Books.Create('Algebra', 7);
   
   physics.Display;
   chemistry.Display;
   maths.Display;
end.

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

Title: Physics for High School
Price: 10
Title: Advanced Chemistry
Price: 15
Title: Algebra
Price: 7

निर्माण नामांकित निहित निर्माता की तरह, एक अंतर्निहित विध्वंसक विधि भी है जिसका उपयोग करके आप कक्षा में उपयोग किए गए सभी संसाधनों को जारी कर सकते हैं।

विरासत

पास्कल श्रेणी की परिभाषाएं मूल रूप से मूल श्रेणी की परिभाषा से ली जा सकती हैं। वाक्य विन्यास इस प्रकार है -

type
childClas-identifier = class(baseClass-identifier) 
< members >
end;

निम्नलिखित उदाहरण एक उपन्यास वर्ग प्रदान करता है, जो पुस्तकें वर्ग को विरासत में देता है और आवश्यकता के आधार पर अधिक कार्यक्षमता जोड़ता है।

program inheritanceExample;

{$MODE OBJFPC} //directive to be used for creating classes
{$M+} //directive that allows class constructors and destructors

type
   Books = Class 
   protected 
      title : String; 
      price: real;
   
   public
      constructor Create(t : String; p: real); //default constructor
      
      procedure setTitle(t : String); //sets title for a book
      function getTitle() : String; //retrieves title
      
      procedure setPrice(p : real); //sets price for a book
      function getPrice() : real; //retrieves price
      
      procedure Display(); virtual; // display details of a book
end;
(* Creating a derived class *)

type
   Novels = Class(Books)
   private
      author: String;
   
   public
      constructor Create(t: String); overload;
      constructor Create(a: String; t: String; p: real); overload;
      
      procedure setAuthor(a: String); // sets author for a book
      function getAuthor(): String; // retrieves author name
      
      procedure Display(); override;
end;
var
   n1, n2: Novels;

//default constructor 
constructor Books.Create(t : String; p: real);
begin
   title := t;
   price := p;
end;

procedure Books.setTitle(t : String); //sets title for a book
begin
   title := t;
end;

function Books.getTitle() : String; //retrieves title
begin
   getTitle := title;
end;

procedure Books.setPrice(p : real); //sets price for a book
begin
   price := p;
end;

function Books.getPrice() : real; //retrieves price
begin
   getPrice:= price;
end;

procedure Books.Display();
begin
   writeln('Title: ', title);
   writeln('Price: ', price);
end;

(* Now the derived class methods  *)
constructor Novels.Create(t: String);
begin
   inherited Create(t, 0.0);
   author:= ' ';
end;

constructor Novels.Create(a: String; t: String; p: real);
begin
   inherited Create(t, p);
   author:= a;
end;

procedure Novels.setAuthor(a : String); //sets author for a book
begin
   author := a;
end;

function Novels.getAuthor() : String; //retrieves author
begin
   getAuthor := author;
end;

procedure Novels.Display();
begin
   writeln('Title: ', title);
   writeln('Price: ', price:5:2);
   writeln('Author: ', author);
end;

begin 
   n1 := Novels.Create('Gone with the Wind');
   n2 := Novels.Create('Ayn Rand','Atlas Shrugged', 467.75);
   n1.setAuthor('Margaret Mitchell');
   n1.setPrice(375.99);
   n1.Display;
   n2.Display;
end.

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

Title: Gone with the Wind
Price: 375.99
Author: Margaret Mitchell
Title: Atlas Shrugged
Price: 467.75
Author: Ayn Rand

महत्वपूर्ण बिंदुओं पर ध्यान देने योग्य है -

  • बुक्स वर्ग के सदस्यों के पास है protected दृश्यता।

  • नॉवेल्स क्लास में दो कंस्ट्रक्टर हैं, इसलिए overload ऑपरेटर का उपयोग फ़ंक्शन ओवरलोडिंग के लिए किया जाता है।

  • Books.Display प्रक्रिया घोषित की गई है virtual, ताकि नोवेल क्लास से एक ही विधि हो सके override यह।

  • नॉवेलसीक्रिएट कंस्ट्रक्टर बेस क्लास कंस्ट्रक्टर का उपयोग करके कॉल करता है inherited कीवर्ड।

इंटरफेस

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

type  
   Mail = Interface  
      Procedure SendMail;  
      Procedure GetMail;  
   end;  
   
   Report = Class(TInterfacedObject,  Mail)  
      Procedure SendMail;  
      Procedure GetMail;  
   end;

कृपया ध्यान दें कि, जब कोई वर्ग इंटरफ़ेस लागू करता है, तो उसे इंटरफ़ेस के सभी तरीकों को लागू करना चाहिए। यदि इंटरफ़ेस का एक तरीका लागू नहीं किया गया है, तो कंपाइलर एक त्रुटि देगा।

सार वर्ग

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

type
   Shape = ABSTRACT CLASS (Root)
      Procedure draw; ABSTRACT;
      ...
   end;

जब एक अमूर्त वर्ग से विरासत में मिला, तो माता-पिता की कक्षा की घोषणा में सार के सभी तरीकों को बच्चे द्वारा परिभाषित किया जाना चाहिए; इसके अलावा, इन तरीकों को समान दृश्यता के साथ परिभाषित किया जाना चाहिए।

स्थैतिक खोजशब्द

कक्षा के सदस्यों या विधियों को स्थैतिक घोषित करने से उन्हें कक्षा की तात्कालिकता की आवश्यकता के बिना सुलभ हो जाता है। स्थैतिक के रूप में घोषित एक सदस्य को तत्काल श्रेणी की वस्तु (हालांकि एक स्थैतिक विधि कर सकते हैं) तक नहीं पहुँचा जा सकता है। निम्नलिखित उदाहरण अवधारणा को दर्शाता है -

program StaticExample;
{$mode objfpc}
{$static on}
type
   myclass=class
      num : integer;static;
   end;
var
   n1, n2 : myclass;
begin
   n1:= myclass.create;
   n2:= myclass.create;
   n1.num := 12;
   writeln(n2.num);
   n2.num := 31;
   writeln(n1.num);
   writeln(myclass.num);
   myclass.num := myclass.num + 20;
   writeln(n1.num);
   writeln(n2.num);
end.

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

12
31
31
51
51

स्थैतिक सदस्यों का उपयोग करने के लिए आपको निर्देश {$ स्थैतिक पर} का उपयोग करना चाहिए।