Groovy - การเขียนโปรแกรม Meta Object

การเขียนโปรแกรม Meta object หรือ MOP สามารถใช้เพื่อเรียกใช้เมธอดแบบไดนามิกและยังสร้างคลาสและเมธอดได้ทันที

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

Def myStudent = new Student() 
myStudent.Name = ”Joe”; 
myStudent.Display()

ขณะนี้อยู่ในการเขียนโปรแกรม meta object แม้ว่าคลาสจะไม่มีชื่อตัวแปรสมาชิกหรือวิธีการแสดง () โค้ดด้านบนจะยังคงใช้งานได้

วิธีการทำงานนี้? เพื่อให้ได้ผลเราต้องใช้อินเทอร์เฟซ GroovyInterceptable เพื่อเชื่อมต่อกับกระบวนการดำเนินการของ Groovy ต่อไปนี้เป็นวิธีการที่ใช้ได้สำหรับอินเทอร์เฟซนี้

Public interface GroovyInterceptable { 
   Public object invokeMethod(String methodName, Object args) 
   Public object getproperty(String propertyName) 
   Public object setProperty(String propertyName, Object newValue) 
   Public MetaClass getMetaClass() 
   Public void setMetaClass(MetaClass metaClass) 
}

ดังนั้นในคำอธิบายอินเทอร์เฟซด้านบนสมมติว่าคุณต้องใช้ invokeMethod () มันจะถูกเรียกสำหรับทุกเมธอดที่มีอยู่หรือไม่มีอยู่

ไม่มีคุณสมบัติ

ลองดูตัวอย่างวิธีที่เราสามารถใช้ Meta Object Programming สำหรับคุณสมบัติที่ขาดหายไป ควรสังเกตสิ่งสำคัญต่อไปนี้เกี่ยวกับรหัสต่อไปนี้

  • คลาส Student ไม่มีตัวแปรสมาชิกที่เรียกว่า Name หรือ ID ที่กำหนด

  • นักเรียนชั้นเรียนใช้อินเทอร์เฟซ GroovyInterceptable

  • มีพารามิเตอร์ที่เรียกว่า dynamicProps ซึ่งจะใช้เพื่อเก็บค่าของตัวแปรสมาชิกที่สร้างขึ้นทันที

  • มีการใช้เมธอด getproperty และ setproperty เพื่อรับและตั้งค่าคุณสมบัติของคลาสที่รันไทม์

class Example {
   static void main(String[] args) {
      Student mst = new Student();
      mst.Name = "Joe";
      mst.ID = 1;
		
      println(mst.Name);
      println(mst.ID);
   }
}

class Student implements GroovyInterceptable { 
   protected dynamicProps=[:]
	
   void setProperty(String pName,val) {
      dynamicProps[pName] = val
   }
   
   def getProperty(String pName) {
      dynamicProps[pName]
   } 
}

ผลลัพธ์ของรหัสต่อไปนี้จะเป็น -

Joe 
1

ไม่มีวิธีการ

ลองดูตัวอย่างวิธีที่เราสามารถใช้ Meta Object Programming สำหรับคุณสมบัติที่ขาดหายไป ควรสังเกตสิ่งสำคัญต่อไปนี้เกี่ยวกับรหัสต่อไปนี้ -

  • ขณะนี้คลาส Student ใช้เมธอด invokeMethod ซึ่งเรียกโดยไม่คำนึงว่าเมธอดนั้นมีอยู่จริงหรือไม่

class Example {
   static void main(String[] args) {
      Student mst = new Student();
      mst.Name = "Joe";
      mst.ID = 1;
		
      println(mst.Name);
      println(mst.ID);
      mst.AddMarks();
   } 
}
 
class Student implements GroovyInterceptable {
   protected dynamicProps = [:]  
    
   void setProperty(String pName, val) {
      dynamicProps[pName] = val
   } 
   
   def getProperty(String pName) {
      dynamicProps[pName]
   }
   
   def invokeMethod(String name, Object args) {
      return "called invokeMethod $name $args"
   }
}

ผลลัพธ์ของรหัสต่อไปนี้จะแสดงด้านล่าง โปรดทราบว่าไม่มีข้อผิดพลาดของ Method Exception ที่หายไปแม้ว่าจะไม่มี Method Display อยู่ก็ตาม

Joe 
1

Metaclass

ฟังก์ชันนี้เกี่ยวข้องกับการใช้งาน MetaClass ในการใช้งานเริ่มต้นคุณสามารถเข้าถึงฟิลด์โดยไม่ต้องเรียกใช้ getters และ setters ตัวอย่างต่อไปนี้แสดงให้เห็นว่าการใช้ฟังก์ชัน metaClass ทำให้เราสามารถเปลี่ยนค่าของตัวแปรส่วนตัวในคลาสได้อย่างไร

class Example {
   static void main(String[] args) {
      Student mst = new Student();
      println mst.getName()
      mst.metaClass.setAttribute(mst, 'name', 'Mark')
      println mst.getName()
   } 
} 

class Student {
   private String name = "Joe";
	
   public String getName() {
      return this.name;
   } 
}

ผลลัพธ์ของรหัสต่อไปนี้จะเป็น -

Joe 
Mark

ไม่มีวิธีการ

Groovy สนับสนุนแนวคิดของ methodMissing เมธอดนี้แตกต่างจาก invokeMethod ตรงที่เรียกใช้เฉพาะในกรณีที่มีการจัดส่งเมธอดล้มเหลวเมื่อไม่พบเมธอดสำหรับชื่อที่กำหนดและ / หรืออาร์กิวเมนต์ที่กำหนด ตัวอย่างต่อไปนี้แสดงให้เห็นว่าสามารถใช้ methodMissing ได้อย่างไร

class Example {
   static void main(String[] args) {
      Student mst = new Student();
      mst.Name = "Joe";
      mst.ID = 1;
		
      println(mst.Name);
      println(mst.ID);
      mst.AddMarks();
   } 
} 

class Student implements GroovyInterceptable {
   protected dynamicProps = [:]  
    
   void setProperty(String pName, val) {
      dynamicProps[pName] = val
   }
   
   def getProperty(String pName) {
      dynamicProps[pName]
   }
   
   def methodMissing(String name, def args) {         
      println "Missing method"
   }  
}

ผลลัพธ์ของรหัสต่อไปนี้จะเป็น -

Joe 
1 
Missing method