Database Tutorials SQLAlchemy তে Inheritance এবং Polymorphism গাইড ও নোট

323

SQLAlchemy তে ইনহেরিটেন্স (Inheritance) এবং পলিমরফিজম (Polymorphism) সম্পর্কিত ধারণাগুলি জেনারেল অবজেক্ট-ওরিয়েন্টেড প্রোগ্রামিং (OOP) ধারণা থেকে এসেছে, এবং এগুলি ডাটাবেস মডেলিং এবং ক্লাস মডেলিংয়ের মধ্যে সম্পর্ক স্থাপন করতে ব্যবহৃত হয়। SQLAlchemy তে ইনহেরিটেন্স এবং পলিমরফিজম ব্যবহার করলে, একাধিক শ্রেণি (classes) বা টেবিলের মধ্যে সম্পর্ক স্থাপন করা যায় যা ক্লাসের বৈশিষ্ট্য এবং আচরণ শেয়ার করতে সক্ষম।


Inheritance Types in SQLAlchemy

SQLAlchemy তে বিভিন্ন ধরনের ইনহেরিটেন্স স্ট্রাটেজি রয়েছে, এবং প্রতিটি স্ট্রাটেজি ডাটাবেস টেবিলের মধ্যে ভিন্নভাবে সংরক্ষণ হয়। SQLAlchemy তে তিনটি প্রধান ইনহেরিটেন্স স্ট্রাটেজি রয়েছে:

  1. Single Table Inheritance (STI): এই পদ্ধতিতে, সমস্ত ইনহেরিটেড ক্লাসের তথ্য এক টেবিলেই রাখা হয়। একটি কলাম নির্ধারণ করা হয় যাতে জানা যায় কোন সাবক্লাসের অবজেক্ট সেই রেকর্ডের জন্য প্রযোজ্য।
  2. Joined Table Inheritance (JTI): এই পদ্ধতিতে, প্রতিটি ক্লাসের জন্য আলাদা টেবিল থাকে এবং একটি যোগফল (join) ব্যবহার করে বিভিন্ন টেবিল থেকে ডেটা একত্রিত করা হয়।
  3. Concrete Table Inheritance (CTI): এই পদ্ধতিতে, প্রতিটি সাবক্লাসের জন্য পৃথক পৃথক টেবিল তৈরি করা হয় এবং সেগুলোর মধ্যে সম্পর্কের প্রয়োজন হয় না।

Polymorphism in SQLAlchemy

Polymorphism একটি অবজেক্ট-ওরিয়েন্টেড কনসেপ্ট যেখানে একাধিক অবজেক্ট বিভিন্ন ধরণের মেথড বা আচরণ শেয়ার করতে পারে। SQLAlchemy তে পলিমরফিজম সুনির্দিষ্ট ইনহেরিটেন্স স্ট্রাটেজি ব্যবহার করে সৃষ্ট হয়।

উদাহরণ: Single Table Inheritance (STI)

এখানে আমরা দেখব কিভাবে SQLAlchemy তে Single Table Inheritance (STI) ব্যবহার করতে হয়।

স্টেপ ১: প্রয়োজনীয় প্যাকেজ ইনস্টল করা

pip install sqlalchemy

স্টেপ ২: SQLAlchemy কনফিগারেশন

from sqlalchemy import create_engine, Column, Integer, String
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker

Base = declarative_base()

# Base Class
class Animal(Base):
    __tablename__ = 'animals'

    id = Column(Integer, primary_key=True)
    name = Column(String)
    type = Column(String)  # Single column to differentiate types
    
    __mapper_args__ = {
        'polymorphic_identity': 'animal',
        'polymorphic_on': type
    }

# Subclass for Dog
class Dog(Animal):
    __tablename__ = 'dogs'
    
    id = Column(Integer, primary_key=True)
    breed = Column(String)
    
    __mapper_args__ = {
        'polymorphic_identity': 'dog',
    }

# Subclass for Cat
class Cat(Animal):
    __tablename__ = 'cats'
    
    id = Column(Integer, primary_key=True)
    color = Column(String)
    
    __mapper_args__ = {
        'polymorphic_identity': 'cat',
    }

# Create engine and session
engine = create_engine('sqlite:///:memory:')
Base.metadata.create_all(engine)

Session = sessionmaker(bind=engine)
session = Session()

# Creating instances
dog = Dog(name='Rex', breed='German Shepherd')
cat = Cat(name='Whiskers', color='White')

session.add(dog)
session.add(cat)
session.commit()

# Querying with polymorphism
animals = session.query(Animal).all()
for animal in animals:
    print(f"{animal.name} is a {animal.type}")

কোডের ব্যাখ্যা:

  1. Base Class (Animal):
    • Animal হল মূল শ্রেণি, যার মধ্যে id, name, এবং type নামে কলাম রয়েছে। type কলামটি আলাদা করে প্রতিটি সাবক্লাসকে চিহ্নিত করতে ব্যবহার হয়।
    • polymorphic_on এর মাধ্যমে এক টেবিলে সব ধরনের অবজেক্ট ধারণ করা হচ্ছে।
    • polymorphic_identity সাবক্লাসের জন্য আলাদা আলাদা চিহ্ন দেয়।
  2. Subclasses (Dog, Cat):
    • Dog এবং Cat ক্লাসগুলো Animal ক্লাস থেকে ইনহেরিট করেছে, এবং তাদের নিজস্ব অতিরিক্ত বৈশিষ্ট্য (যেমন breed এবং color) যুক্ত করা হয়েছে।
    • এগুলোর polymorphic_identity আলাদা আলাদা 'dog' এবং 'cat' হিসেবে সেট করা হয়েছে।
  3. Querying with Polymorphism:
    • session.query(Animal).all() ব্যবহার করে সব ধরনের অবজেক্ট (যা Animal ক্লাসের সাবক্লাস) একসাথে পাওয়া যায়।

আউটপুট:

Rex is a dog
Whiskers is a cat

Joined Table Inheritance (JTI) এর উদাহরণ:

স্টেপ ১: কোড

# Parent Class
class Animal(Base):
    __tablename__ = 'animals'

    id = Column(Integer, primary_key=True)
    name = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'animal',
        'polymorphic_on': type
    }

# Dog subclass with Joined Table Inheritance
class Dog(Animal):
    __tablename__ = 'dogs'
    
    id = Column(Integer, ForeignKey('animals.id'), primary_key=True)
    breed = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'dog',
    }

# Cat subclass with Joined Table Inheritance
class Cat(Animal):
    __tablename__ = 'cats'
    
    id = Column(Integer, ForeignKey('animals.id'), primary_key=True)
    color = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'cat',
    }

# Create tables
Base.metadata.create_all(engine)

# Create session and insert data
session = Session()
dog = Dog(name='Rex', breed='German Shepherd')
cat = Cat(name='Whiskers', color='White')

session.add(dog)
session.add(cat)
session.commit()

স্টেপ ২: Querying

animals = session.query(Animal).all()
for animal in animals:
    print(f"{animal.name} is a {animal.type}")

Conclusion:

  • Single Table Inheritance (STI) একটি টেবিলের মধ্যে সব ডেটা রাখতে সাহায্য করে, যেটি পারফরম্যান্সের দিক থেকে ভাল, তবে অনেক বড় ডেটাবেসে এটি ব্যয়বহুল হতে পারে।
  • Joined Table Inheritance (JTI) রিলেশনাল টেবিলগুলো আলাদা রাখে এবং এইভাবে বেশি ফ্লেক্সিবিলিটি পাওয়া যায়।
  • Polymorphism ব্যবহারের মাধ্যমে আপনি একাধিক শ্রেণির অবজেক্টকে একসাথে পলিমর্ফিক্যালি কাজ করতে পারেন, যা সফটওয়্যারের কোডে উন্নত প্রবাহ এবং কার্যকারিতা তৈরি করে।

এই ধরনের ইনহেরিটেন্স এবং পলিমরফিজমের সাহায্যে আপনার অ্যাপ্লিকেশন আরও মডুলার, স্কেলেবল এবং কার্যকর হতে পারে।

Content added By

Table-per-class Inheritance

271

SQLAlchemy তে Table-per-Class Inheritance (TPCI) প্যাটার্ন ব্যবহার করা হয় যখন আমরা একাধিক ক্লাসের জন্য আলাদা টেবিল তৈরি করতে চাই, তবে প্রতিটি টেবিলেই একই ফিল্ডগুলি থাকবে। এর মানে হল যে প্রতিটি সাবক্লাসের জন্য একটি আলাদা টেবিল থাকবে, এবং প্রতি টেবিলের মধ্যে শুধু সেই সাবক্লাসের নিজস্ব ফিল্ডগুলি থাকবে, যেখানে অভ্যন্তরীণভাবে বাইরের টেবিলের প্রাথমিক কীগুলি ব্যবহৃত হবে।


এটি মূলত যখন আপনি যদি আপনার ওবজেক্ট মডেলগুলোকে আলাদা টেবিলগুলিতে সংরক্ষণ করতে চান, এবং সম্পর্কিত ডেটা একসাথে থাকুক।

SQLAlchemy তে Table-per-Class Inheritance ব্যবহারের উদাহরণ:

ধরা যাক, আমাদের একটি Person এবং তার দুটি সাবক্লাস Employee এবং Manager রয়েছে।

কোড উদাহরণ:

from sqlalchemy import create_engine, Column, Integer, String, ForeignKey
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import relationship, sessionmaker

Base = declarative_base()

# Parent class
class Person(Base):
    __tablename__ = 'person'
    id = Column(Integer, primary_key=True)
    name = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'person',
        'polymorphic_on': id
    }

# Subclass: Employee
class Employee(Person):
    __tablename__ = 'employee'
    id = Column(Integer, ForeignKey('person.id'), primary_key=True)
    job_title = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'employee'
    }

# Subclass: Manager
class Manager(Person):
    __tablename__ = 'manager'
    id = Column(Integer, ForeignKey('person.id'), primary_key=True)
    department = Column(String)

    __mapper_args__ = {
        'polymorphic_identity': 'manager'
    }

# Database setup
engine = create_engine('sqlite:///:memory:')
Base.metadata.create_all(engine)

Session = sessionmaker(bind=engine)
session = Session()

# Adding some data
manager1 = Manager(name='Alice', department='HR')
employee1 = Employee(name='Bob', job_title='Engineer')

session.add(manager1)
session.add(employee1)
session.commit()

# Querying the data
for person in session.query(Person).all():
    print(f"{person.name} - {person.__class__.__name__}")

কোডের ব্যাখ্যা:

  1. Person হল প্যারেন্ট ক্লাস। এর একটি টেবিল থাকবে person নামে।
  2. Employee এবং Manager হল সাবক্লাস। প্রতিটি সাবক্লাসের জন্য আলাদা টেবিল হবে: employee এবং manager
  3. ForeignKey ব্যবহার করা হয়েছে, যাতে Employee এবং Manager ক্লাসগুলি তাদের প্যারেন্ট Person ক্লাসের id সুত্রের মাধ্যমে যুক্ত থাকে।
  4. __mapper_args__ তে polymorphic_identity নির্ধারণ করে দেওয়া হয়েছে, যাতে প্রতিটি টেবিল আলাদা আলাদা ইনস্ট্যান্সের জন্য সঠিক সাবক্লাস সংজ্ঞায়িত করতে পারে।

ফলাফল:

Alice - Manager
Bob - Employee

উপকারিতা:

  1. ডেটা বিচ্ছিন্নতা: প্রতিটি সাবক্লাসের জন্য আলাদা টেবিল থাকে, ফলে ডেটা কাঠামো এবং সম্পর্কগুলি পরিষ্কার থাকে।
  2. প্রতিটি টেবিলের জন্য প্রপার্টি আলাদা রাখা: প্রতিটি সাবক্লাসের নিজস্ব কাস্টম ফিল্ড থাকবে, যা প্রধান টেবিল থেকে আলাদা থাকে।
  3. পর্যাপ্ত ফ্লেক্সিবিলিটি: আপনি প্রয়োজন অনুযায়ী টেবিলের মাঝে সম্পর্ক নির্ধারণ করতে পারেন এবং ডেটাবেসের গঠন কাস্টমাইজ করতে পারেন।

সীমাবদ্ধতা:

  1. কঠিন জটিলতা: টেবিলগুলো আলাদা হওয়ায় জটিলতা বেড়ে যেতে পারে। যখন একাধিক টেবিলের ডেটা একত্রিত করতে হবে, তখন কমপ্লেক্স কোয়েরি তৈরি হতে পারে।
  2. সম্পর্কের জটিলতা: টেবিলগুলির মধ্যে সম্পর্কের জটিলতা বাড়তে পারে যখন অনেকগুলো সাবক্লাস থাকে।

এটি Table-per-Class Inheritance এর একটি সাধারণ উদাহরণ যা SQLAlchemy তে ব্যবহৃত হতে পারে।

Content added By

Joined-table Inheritance

265

Joined-table Inheritance হল একটি ORM (Object-Relational Mapping) প্রযুক্তির একটি প্যাটার্ন, যা একটি ক্লাস হায়ারার্কিকে একাধিক টেবিলে ভাগ করে রাখা হয়। Flask-SQLAlchemy ব্যবহার করে, Joined-table Inheritance ডিজাইন করতে গেলে, সিংহভাগ ক্লাস মডেলগুলির জন্য আলাদা আলাদা টেবিল তৈরি করা হয় এবং ক্লাসগুলির মধ্যে সম্পর্ক বজায় রাখতে একটি মূল টেবিল থেকে ডেটা যুক্ত করা হয়।

এটি মূলত নিম্নলিখিত তিনটি ধাপে কাজ করে:

  1. Parent Class: একটি মৌলিক বা অভ্যন্তরীণ ক্লাস যা সাধারণ ক্ষেত্র বা অঙ্গসংস্থান ধারণ করে।
  2. Child Classes: এই ক্লাসগুলি মূল Parent ক্লাস থেকে বংশানুক্রমে লাভ করা হয় এবং এগুলি parent class এর জন্য অতিরিক্ত ক্ষেত্র ধারণ করে। এগুলি পৃথক টেবিল হতে পারে কিন্তু তাদের মধ্যে সম্পর্ক Parent ক্লাসের সাথে থাকে।
  3. JOIN: যখন query করা হয়, তখন Child ক্লাসের সাথে Parent ক্লাসের ডেটা সংযুক্ত (joined) হয়ে আসে।

এখন, Flask-SQLAlchemy-তে Joined-table Inheritance বাস্তবায়নের জন্য কিছু উদাহরণ দেখানো যাক:


Step 1: Flask Setup and Required Libraries

প্রথমে Flask এবং Flask-SQLAlchemy ইন্সটল করতে হবে যদি তা না থাকে।

pip install flask flask-sqlalchemy

Step 2: Define the Base Model and Child Models

এখন, আমরা Parent ক্লাস এবং দুটি Child ক্লাস ডিফাইন করব।

from flask import Flask
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///example.db'
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
db = SQLAlchemy(app)

# Base Model (Parent Class)
class Animal(db.Model):
    __tablename__ = 'animals'
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(80))
    
    __mapper_args__ = {
        'polymorphic_identity': 'animal',
        'polymorphic_on': id
    }

    def __init__(self, name):
        self.name = name

# Dog Model (Child Class)
class Dog(Animal):
    __tablename__ = 'dogs'
    id = db.Column(db.Integer, db.ForeignKey('animals.id'), primary_key=True)
    breed = db.Column(db.String(50))

    __mapper_args__ = {
        'polymorphic_identity': 'dog'
    }

    def __init__(self, name, breed):
        super().__init__(name)
        self.breed = breed

# Cat Model (Child Class)
class Cat(Animal):
    __tablename__ = 'cats'
    id = db.Column(db.Integer, db.ForeignKey('animals.id'), primary_key=True)
    color = db.Column(db.String(50))

    __mapper_args__ = {
        'polymorphic_identity': 'cat'
    }

    def __init__(self, name, color):
        super().__init__(name)
        self.color = color

Step 3: Database Creation

এখন, আমরা ডেটাবেস তৈরি করব।

@app.before_first_request
def create_tables():
    db.create_all()

Step 4: Inserting Data

এখন, Parent ক্লাস এবং Child ক্লাসে ডেটা ইনসার্ট করব।

@app.route('/')
def index():
    # Create new instances of Animal, Dog, and Cat
    animal = Animal(name="Generic Animal")
    dog = Dog(name="Rex", breed="German Shepherd")
    cat = Cat(name="Whiskers", color="Black")

    db.session.add(animal)
    db.session.add(dog)
    db.session.add(cat)
    db.session.commit()

    return "Animals added!"

Step 5: Querying Data

এখন, আমরা কিছু query করব যাতে Parent ক্লাস এবং Child ক্লাসের তথ্য পাওয়া যায়।

@app.route('/show')
def show():
    animals = Animal.query.all()
    output = ""
    for animal in animals:
        if isinstance(animal, Dog):
            output += f"Dog: {animal.name}, Breed: {animal.breed}<br>"
        elif isinstance(animal, Cat):
            output += f"Cat: {animal.name}, Color: {animal.color}<br>"
        else:
            output += f"Animal: {animal.name}<br>"
    return output

Step 6: Running the Application

এখন Flask অ্যাপ্লিকেশন রান করাতে হবে:

if __name__ == '__main__':
    app.run(debug=True)

Conclusion:

Joined-table Inheritance প্যাটার্নের সাহায্যে, Flask-SQLAlchemy-তে Parent এবং Child ক্লাসের মধ্যে সম্পর্ক সংরক্ষিত হয়, যেখানে Child ক্লাসগুলির জন্য আলাদা টেবিল তৈরি হয় এবং Parent ক্লাসের মাধ্যমে তাদের মধ্যে সম্পর্ক তৈরি হয়। এতে কোডের পুনঃব্যবহারযোগ্যতা বাড়ে এবং ডেটাবেসে বিভিন্ন ধরনের সম্পর্কের মধ্যে সংযোগ বজায় রাখা সম্ভব হয়।

এটা আপনার অ্যাপ্লিকেশনে ডেটাবেস সম্পর্কের আরও কার্যকরী এবং প্রসারিত পদ্ধতি হতে পারে।

Content added By

Single-table Inheritance

295

Single-table Inheritance (STI) হল একটি ORM ডিজাইন প্যাটার্ন যেখানে একাধিক সাবক্লাসের ডেটা একটি একক টেবিলে সংরক্ষণ করা হয়। এই প্যাটার্নটি ব্যবহৃত হয় যখন একাধিক ক্লাসে একই ধরনের ডেটা থাকে, কিন্তু তাদের মধ্যে কিছু ভিন্নতা থাকতে পারে। STI সাধারণত ORM (Object-Relational Mapping) সিস্টেমে ব্যবহৃত হয়, যেখানে একাধিক ইনহেরিটেড ক্লাসের ডেটা একটিই টেবিলের মধ্যে রাখা হয় এবং একটি অতিরিক্ত কলাম দ্বারা পৃথকীকরণ করা হয়।

এই প্যাটার্নটি বিশেষভাবে কার্যকর যখন আপনি ক্লাস হায়ারার্কির মধ্যে শেয়ার করা প্রপার্টি গুলি একত্রে সংরক্ষণ করতে চান।


উদাহরণ:

ধরা যাক আমাদের একটি Employee নামক মূল ক্লাস আছে, এবং এর দুটি সাবক্লাস রয়েছে: Manager এবং Developer। সবগুলো ক্লাসের কিছু সাধারণ প্রপার্টি (যেমন name, email, salary) শেয়ার করা হয়, কিন্তু Manager ক্লাসে অতিরিক্ত department প্রপার্টি থাকতে পারে এবং Developer ক্লাসে programming_language প্রপার্টি থাকতে পারে।

মডেল কাঠামো:

# Base Class
class Employee(models.Model):
    name = models.CharField(max_length=100)
    email = models.EmailField()
    salary = models.DecimalField(max_digits=10, decimal_places=2)

    class Meta:
        abstract = True  # This ensures that Employee itself isn't directly instantiated

# Derived Class 1: Manager
class Manager(Employee):
    department = models.CharField(max_length=100)

# Derived Class 2: Developer
class Developer(Employee):
    programming_language = models.CharField(max_length=50)

এই ক্ষেত্রে, Employee, Manager, এবং Developer ক্লাসগুলির মধ্যে মূল পার্থক্য হল যে Manager এবং Developer টেবিলের জন্য department এবং programming_language ফিল্ডগুলি আলাদা, কিন্তু সবগুলো ক্লাসের জন্য name, email, এবং salary ফিল্ডগুলি একই।

ডেটাবেজ টেবিলের কাঠামো:

এখন, STI প্যাটার্ন অনুযায়ী ডেটাবেজে এই সমস্ত ডেটা একটি একক টেবিলে সংরক্ষিত হবে, এবং সবগুলো ক্লাসের জন্য আলাদা রেকর্ড থাকবে। এই টেবিলটি দেখতে এরকম হতে পারে:

idnameemailsalarydepartmentprogramming_languagetype
1John Doejohn@example.com60000MarketingNULLManager
2Alice Smithalice@example.com80000NULLPythonDeveloper

বিশেষ দৃষ্টি:

  • type কলামটি তৈরি হয় (এটি প্রায়শই ORM স্বয়ংক্রিয়ভাবে তৈরি করে) যাতে প্রতিটি রেকর্ডের জন্য জানানো যায় কোন সাবক্লাসটি তা উপস্থাপন করছে।
  • department কলামটি শুধুমাত্র Manager রেকর্ডের জন্য প্রযোজ্য, এবং programming_language শুধুমাত্র Developer রেকর্ডের জন্য প্রযোজ্য।

সুবিধা:

  • সহজ ডেটাবেস ডিজাইন: সমস্ত ডেটা একটি টেবিলেই সংরক্ষিত হয়, তাই ডেটাবেজে অনেক টেবিল না রেখে একটি টেবিলের মধ্যে সব কিছু রাখতে পারবেন।
  • কমপ্লেক্স জোইন অপারেশন এড়ানো: যদি বিভিন্ন সাবক্লাস থেকে ডেটা একসাথে দরকার হয়, তাহলে জোইন করার দরকার পড়বে না।

অসুবিধা:

  • নির্দিষ্ট কলামগুলির উপস্থিতি: কিছু সাবক্লাসের জন্য কিছু কলাম অপ্রয়োজনীয় হতে পারে (যেমন department কলাম শুধুমাত্র Manager ক্লাসে প্রযোজ্য)। এটা কিছু অপ্রয়োজনীয় ডেটা সংরক্ষণ হতে পারে, যা ডেটাবেজের পারফরমেন্সের উপর প্রভাব ফেলতে পারে।
  • ডেটাবেজের পরিবর্তন করা কঠিন হতে পারে: যখন ক্লাসের হায়ারার্কি পরিবর্তিত হয়, তখন ডেটাবেজের মডেলও আপডেট করতে হয়, যা কিছু সময়ের জন্য জটিল হতে পারে।

Conclusion:

Single-table Inheritance একটি শক্তিশালী প্যাটার্ন যখন আপনি একই টেবিলে একাধিক ইনহেরিটেড ক্লাসের ডেটা রাখতে চান, তবে এটিতে কিছু সীমাবদ্ধতা থাকতে পারে যেমন অপ্রয়োজনীয় কলামের উপস্থিতি এবং ডেটাবেজের স্কেলিং সমস্যা।

Content added By

Polymorphic Querying Techniques

271

Polymorphic querying হচ্ছে এমন একটি কৌশল যা ডাটাবেসে একাধিক মডেল বা টেবিল থেকে ডেটা আনার জন্য ব্যবহৃত হয়, যেখানে একাধিক সম্পর্কিত মডেল একই ভিত্তিতে কাজ করে এবং তাদের মধ্যে একাধিক ডাটাবেস কনসেপ্টের পার্থক্য থাকে। এটি সাধারণত একাধিক শ্রেণী বা মডেল এক সাথে ব্যবহারের জন্য ব্যবহৃত হয়, যেখানে বিশেষভাবে ফিল্টার বা কোয়েরি করার জন্য একটি কমন ভিত্তি থাকে। Laravel বা অন্যান্য ফ্রেমওয়ার্কে Polymorphic relations অনেকটা ব্যবহৃত হয় যেমন morphTo, morphMany ইত্যাদি।

Polymorphic Relations এর সাধারণ ধারণা

Polymorphic relationships দুই বা ততোধিক মডেলকে এমনভাবে যুক্ত করে, যেগুলি একে অপরের মধ্যে সম্পর্কিত থাকতে পারে। ধরুন, একটি Comment মডেল থাকে, যেটি কোনও Post বা Video বা অন্য কোন মডেলের সাথে সম্পর্কিত হতে পারে, তবে এটি তাদের একটি morphable (অথবা polymorphic) সম্পর্ক।

এ ধরনের সম্পর্কের জন্য সাধারণত তিনটি গুরুত্বপূর্ণ উপাদান থাকে:

  1. Morphable Field - একটি ক্ষেত্র যা মডেলটি শনাক্ত করতে সাহায্য করে।
  2. Morph Type Field - কোন মডেলটিকে সাথে সম্পর্কিত করা হচ্ছে তা নির্ধারণ করার জন্য একটি ধরনের ক্ষেত্র।
  3. Pivot Table (যদি প্রয়োজন হয়) - দুই বা ততোধিক মডেলের সম্পর্কের জন্য একটি মধ্যবর্তী টেবিল।

Polymorphic Querying উদাহরণ

ধরা যাক, আমাদের একটি Comment মডেল আছে, যা Post এবং Video মডেলের সাথে polymorphic সম্পর্কিত। এখন, Comment মডেল থেকে আমরা Post বা Video থেকে কমেন্টগুলি অনুসন্ধান করতে চাই, তাহলে আমরা polymorphic query ব্যবহার করতে পারি।

Migration example:

Schema::create('comments', function (Blueprint $table) {
    $table->id();
    $table->text('body');
    $table->morphs('commentable'); // Adds commentable_id and commentable_type
    $table->timestamps();
});

Model example:

// Comment model
class Comment extends Model
{
    public function commentable()
    {
        return $this->morphTo();
    }
}

// Post model
class Post extends Model
{
    public function comments()
    {
        return $this->morphMany(Comment::class, 'commentable');
    }
}

// Video model
class Video extends Model
{
    public function comments()
    {
        return $this->morphMany(Comment::class, 'commentable');
    }
}

এখন, যখন আমরা Comment মডেলের জন্য কোয়েরি করব, তখন commentable_type কলাম অনুসারে মন্তব্যগুলিকে ফিল্টার করতে পারব।

Query example:

// Get all comments for a specific post
$post = Post::find(1);
$comments = $post->comments;

// Get all comments for a specific video
$video = Video::find(1);
$comments = $video->comments;

Polymorphic Querying Example for Filtering:

// Fetching comments for a specific type (Post or Video)
$comments = Comment::where('commentable_type', 'App\Models\Post')->get();

Polymorphic Querying এর ব্যবহার

  1. Efficient Querying: Polymorphic queries সাধারণত একটি টেবিলের মধ্যে বিভিন্ন ধরনের সম্পর্ক যুক্ত করতে সহায়ক, যা ডেটাবেসের পরিমাণ কমাতে সাহায্য করে।
  2. Dynamic Querying: Polymorphic querying ডাইনামিক ফিল্টারিং এবং অনুসন্ধান প্রদান করতে পারে, যেমন একটি মডেল বা সম্পর্কিত মডেল অনুসারে ডেটা নির্বাচন করা।
  3. Relational Flexibility: এটি ডেটাবেস ডিজাইনে নমনীয়তা প্রদান করে, যেখানে একাধিক সম্পর্কের মধ্যে ফিল্টারিং এবং যোগসূত্র কার্যকর করা যায়।

Polymorphic Querying তে কিছু অতিরিক্ত কৌশল:

  1. Using morphToMany: Polymorphic many-to-many সম্পর্কগুলো ম্যানেজ করার জন্য morphToMany ব্যবহার করা হয়। উদাহরণস্বরূপ, একটি Tag মডেল যা বিভিন্ন মডেল (যেমন Post, Video ইত্যাদি) এর সাথে যুক্ত থাকতে পারে।
  2. Query Optimization: Polymorphic queries সাধারণত with অথবা whereHas ব্যবহার করে দ্রুত কুয়েরি অপটিমাইজেশন করতে সাহায্য করতে পারে। উদাহরণস্বরূপ, polymorphic relation সহ মডেলগুলিতে eager loading।

Eager Loading Example

// Fetching comments with related post and video using eager loading
$comments = Comment::with('commentable')->get();

সারাংশ

Polymorphic querying হচ্ছে এমন একটি শক্তিশালী কৌশল যা একাধিক সম্পর্কযুক্ত মডেলকে একসাথে ব্যবহার করতে দেয়, যা ডাটাবেস ডিজাইন এবং কোডিং সহজ করে। এটি Laravel এবং অন্যান্য ফ্রেমওয়ার্কে সাধারণত ব্যবহৃত হয় এবং ডেটাবেসের অপটিমাইজেশনে সহায়ক।

Content added By
Promotion

Are you sure to start over?

Loading...