Pascal - คลาส

คุณได้เห็นแล้วว่า Pascal Objects แสดงลักษณะบางประการของกระบวนทัศน์เชิงวัตถุ พวกเขาใช้การห่อหุ้มการซ่อนข้อมูลและการสืบทอด แต่ก็มีข้อ จำกัด เช่นกัน ตัวอย่างเช่น Pascal Objects ไม่ได้มีส่วนร่วมในความหลากหลาย ดังนั้นคลาสจึงถูกใช้กันอย่างแพร่หลายเพื่อใช้พฤติกรรมเชิงวัตถุที่เหมาะสมในโปรแกรมโดยเฉพาะซอฟต์แวร์ที่ใช้ GUI

คลาสถูกกำหนดในลักษณะเดียวกับ Object แต่เป็นตัวชี้ไปที่ Object แทนที่จะเป็น Object เอง ในทางเทคนิคหมายความว่าคลาสถูกจัดสรรบน Heap ของโปรแกรมในขณะที่ Object ถูกจัดสรรบน Stack กล่าวอีกนัยหนึ่งคือเมื่อคุณประกาศตัวแปรประเภทอ็อบเจ็กต์มันจะใช้พื้นที่บนสแต็กเท่ากับขนาดของอ็อบเจ็กต์ แต่เมื่อคุณประกาศตัวแปรประเภทคลาสมันจะใช้ขนาดของพอยน์เตอร์เสมอ บนกอง ข้อมูลคลาสจริงจะอยู่ในฮีป

การกำหนดคลาส Pascal

คลาสถูกประกาศในลักษณะเดียวกับอ็อบเจ็กต์โดยใช้การประกาศประเภท รูปแบบทั่วไปของการประกาศคลาสมีดังนี้ -

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ในคลาสรูท คลาสนามธรรมและคลาสคอนกรีตทุกคลาสเป็นลูกหลานของรูทดังนั้นคลาสทั้งหมดจึงมีผู้ทำลายอย่างน้อยหนึ่งคน

ให้เรากำหนดคลาส Rectangle ที่มีสมาชิกข้อมูลประเภทจำนวนเต็มสองตัว - ความยาวและความกว้างและฟังก์ชันสมาชิกบางอย่างเพื่อจัดการกับสมาชิกข้อมูลเหล่านี้และขั้นตอนในการวาดสี่เหลี่ยมผืนผ้า

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;

ให้เราเขียนโปรแกรมที่สมบูรณ์ที่จะสร้างอินสแตนซ์ของคลาสสี่เหลี่ยมผืนผ้าและวาดรูปสี่เหลี่ยมผืนผ้า นี่เป็นตัวอย่างเดียวกับที่เราใช้ในการพูดคุยเกี่ยวกับ Pascal Objects คุณจะพบว่าทั้งสองโปรแกรมเกือบจะเหมือนกันโดยมีข้อยกเว้นดังต่อไปนี้ -

  • คุณจะต้องใส่คำสั่ง {$ mode 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
* * * * * * 
* * * * * * 
* * * * * * 
* * * * * *

การเปิดเผยของสมาชิกชั้นเรียน

การมองเห็นบ่งบอกถึงความสามารถในการเข้าถึงของสมาชิกชั้นเรียน สมาชิกชั้นเรียน Pascal มีการมองเห็นห้าประเภท -

ซีเนียร์ No การมองเห็นและการเข้าถึง
1

Public

สมาชิกเหล่านี้สามารถเข้าถึงได้ตลอดเวลา

2

Private

สมาชิกเหล่านี้สามารถเข้าถึงได้เฉพาะในโมดูลหรือหน่วยที่มีข้อกำหนดคลาส สามารถเข้าถึงได้จากภายในวิธีการเรียนหรือจากภายนอก

3

Strict Private

สมาชิกเหล่านี้สามารถเข้าถึงได้จากเมธอดของคลาสเท่านั้น คลาสอื่น ๆ หรือคลาสที่สืบเชื้อสายมาในหน่วยเดียวกันไม่สามารถเข้าถึงได้

4

Protected

สิ่งนี้เหมือนกับไพรเวตยกเว้นสมาชิกเหล่านี้สามารถเข้าถึงได้ในประเภทที่ลดหลั่นกันแม้ว่าจะถูกนำไปใช้ในโมดูลอื่นก็ตาม

5

Published

สิ่งนี้เหมือนกับ Public แต่คอมไพลเลอร์จะสร้างข้อมูลประเภทที่จำเป็นสำหรับการสตรีมอัตโนมัติของคลาสเหล่านี้หากคอมไพเลอร์อยู่ในสถานะ {$ M +} ฟิลด์ที่กำหนดในส่วนที่เผยแพร่ต้องเป็นประเภทคลาส

ตัวสร้างและตัวทำลายสำหรับคลาส Pascal

ตัวสร้างเป็นวิธีการพิเศษซึ่งจะเรียกโดยอัตโนมัติเมื่อใดก็ตามที่สร้างวัตถุ ดังนั้นเราจึงใช้ประโยชน์อย่างเต็มที่จากพฤติกรรมนี้โดยการเริ่มต้นหลาย ๆ สิ่งผ่านฟังก์ชันตัวสร้าง

Pascal มีฟังก์ชันพิเศษที่เรียกว่า create () เพื่อกำหนดคอนสตรัคเตอร์ คุณสามารถส่งผ่านอาร์กิวเมนต์ได้มากเท่าที่คุณต้องการในฟังก์ชันตัวสร้าง

ตัวอย่างต่อไปนี้จะสร้างตัวสร้างหนึ่งตัวสำหรับคลาสชื่อหนังสือและจะเริ่มต้นราคาและชื่อหนังสือในเวลาที่สร้างวัตถุ

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

เช่นเดียวกับตัวสร้างโดยนัยที่มีชื่อว่า create นอกจากนี้ยังมีวิธีการทำลายโดยนัยที่ใช้ทำลายซึ่งคุณสามารถปล่อยทรัพยากรทั้งหมดที่ใช้ในคลาสได้

มรดก

นิยามคลาส Pascal สามารถเลือกที่จะสืบทอดจากนิยามคลาสพาเรนต์ ไวยากรณ์มีดังนี้ -

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 ตัวดำเนินการใช้สำหรับการทำงานมากเกินไป

  • มีการประกาศขั้นตอนการแสดงหนังสือ virtualเพื่อให้วิธีการเดียวกันจากคลาสนวนิยายสามารถ override มัน.

  • Novels.Create constructor เรียกตัวสร้างคลาสพื้นฐานโดยใช้ 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

คุณต้องใช้คำสั่ง {$ static on} เพื่อใช้สมาชิกแบบคงที่