Lazy Loading এবং Eager Loading

আইবাটিস (iBATIS) - Java Technologies

307

iBATIS (বর্তমানে MyBatis) একটি powerful Object-Relational Mapping (ORM) ফ্রেমওয়ার্ক যা SQL কুয়েরি এবং Java objects এর মধ্যে সম্পর্ক স্থাপন করে। iBATIS মূলত data mapper প্যাটার্নে কাজ করে, যেখানে আপনি SQL কুয়েরি লিখেন এবং সেগুলি JavaBeans এর সাথে ম্যাপ করা হয়। যদিও iBATIS একটি full-fledged ORM নয়, তবে এটি Lazy Loading এবং Eager Loading সমর্থন করে, যার মাধ্যমে আপনি ডেটাবেস থেকে সম্পর্কিত ডেটা লোড করার প্রক্রিয়া নিয়ন্ত্রণ করতে পারেন।

Lazy Loading এবং Eager Loading এর ধারণা

  • Lazy Loading: Lazy loading হল একটি কৌশল যেখানে সম্পর্কিত ডেটা শুধুমাত্র তখনই লোড করা হয় যখন এটি প্রথমবার অ্যাক্সেস করা হয়। এটি ডেটার প্রাথমিক লোডিংয়ের সময় performance উন্নত করতে সাহায্য করে।
  • Eager Loading: Eager loading এর মাধ্যমে সম্পর্কিত সমস্ত ডেটা একসাথে লোড করা হয়, যাতে ডেটা অ্যাক্সেসের সময় আরও কমপ্লেক্স কুয়েরি না করতে হয়। যদিও এটি performance কমাতে পারে, তবে এটি অনেক ক্ষেত্রেই সহজ এবং দ্রুত ডেটা অ্যাক্সেসের সুযোগ দেয়।

Lazy Loading এবং Eager Loading iBATIS এ

iBATIS (MyBatis) এ Lazy Loading এবং Eager Loading হ্যান্ডল করার জন্য কিছু কৌশল এবং কনফিগারেশন থাকে, যা SQL statements এর মাধ্যমে সম্পর্কিত ডেটা লোড করার সময় ব্যবহৃত হয়।

1. Lazy Loading in iBATIS

iBATIS-এ Lazy Loading সক্ষম করতে, resultMap এবং association বা collection ব্যবহার করা হয়, যেখানে সম্পর্কিত ডেটা (যেমন, একাধিক child বা nested objects) শুধুমাত্র যখন প্রথমবারের মতো অ্যাক্সেস করা হয় তখন লোড করা হয়।

Example: Lazy Loading with association

<resultMap id="employeeResultMap" type="com.example.model.Employee">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <result property="salary" column="salary"/>
    <association property="department" column="department_id" javaType="com.example.model.Department" select="com.example.mapper.DepartmentMapper.getDepartmentById"/>
</resultMap>

<select id="getEmployeeById" resultMap="employeeResultMap">
    SELECT id, name, salary, department_id FROM employees WHERE id = #{id}
</select>

Explanation:

  • Lazy Loading with association: এখানে, department অ্যাসোসিয়েশনটি lazy loading এর মাধ্যমে লোড হবে, কারণ এটি select স্টেটমেন্টের মধ্যে ব্যবহৃত হয় এবং যখন প্রথমবারের মতো employee রেকর্ডটি অ্যাক্সেস করা হবে তখনই department তথ্য লোড হবে।
  • select পদ্ধতি দিয়ে, একটি subquery কে DepartmentMapper.getDepartmentById হিসাবে কল করা হয়েছে। এটি সম্পর্কিত department ডেটা লোড করে।

2. Eager Loading in iBATIS

Eager Loading চালানোর জন্য iBATIS-এ আপনাকে join বা fetch পদ্ধতির মাধ্যমে সম্পর্কিত ডেটা একসাথে লোড করতে হবে, যাতে related entities একসাথে সিলেক্ট করা হয়।

Example: Eager Loading with join

<resultMap id="employeeResultMap" type="com.example.model.Employee">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <result property="salary" column="salary"/>
    <result property="department.name" column="department_name"/>
</resultMap>

<select id="getEmployeeWithDepartment" resultMap="employeeResultMap">
    SELECT e.id, e.name, e.salary, d.name AS department_name 
    FROM employees e
    JOIN departments d ON e.department_id = d.id
    WHERE e.id = #{id}
</select>

Explanation:

  • Eager Loading with join: এখানে, JOIN অপারেশন ব্যবহার করা হয়েছে, যা employees এবং departments টেবিলের ডেটা একসাথে লোড করে। এই পদ্ধতিতে, যখন employee ডেটা রিটার্ন করা হয়, তখন department সম্পর্কিত ডেটাও একসাথে লোড হয়ে যায়।

3. Using fetchType for Eager and Lazy Loading

iBATIS-এ আপনি fetchType সেটিংস ব্যবহার করে Lazy এবং Eager লোডিং এর মধ্যে পার্থক্য করতে পারেন। Lazy loading সাধারণত fetchType.LAZY দ্বারা কনফিগার করা হয়, এবং Eager loading সাধারণত fetchType.EAGER ব্যবহার করে।

Example: Eager and Lazy Loading with fetchType

<resultMap id="employeeResultMap" type="com.example.model.Employee">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <result property="salary" column="salary"/>
    <association property="department" column="department_id" fetchType="EAGER"/>
</resultMap>

<select id="getEmployeeById" resultMap="employeeResultMap">
    SELECT id, name, salary, department_id FROM employees WHERE id = #{id}
</select>

Explanation:

  • Eager Loading: এখানে, fetchType="EAGER" ব্যবহার করা হয়েছে, যার মাধ্যমে department সম্পর্কিত ডেটা একসাথে লোড করা হবে।
<resultMap id="employeeResultMap" type="com.example.model.Employee">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <result property="salary" column="salary"/>
    <association property="department" column="department_id" fetchType="LAZY"/>
</resultMap>

Explanation:

  • Lazy Loading: এখানে, fetchType="LAZY" ব্যবহার করা হয়েছে, যার মাধ্যমে department সম্পর্কিত ডেটা শুধুমাত্র প্রথমবারের মতো অ্যাক্সেস করার সময় লোড হবে।

Best Practices for Lazy and Eager Loading in iBATIS

  1. Use Lazy Loading for Large Datasets: যখন সম্পর্কিত ডেটা খুব বড় হয় এবং সব সময় ব্যবহৃত হয় না, তখন lazy loading ব্যবহার করুন। এটি ডেটাবেসের লোডের পরিমাণ কমাতে সাহায্য করে এবং পারফরম্যান্স উন্নত করে।
  2. Use Eager Loading for Frequently Accessed Data: যখন সম্পর্কিত ডেটা সবসময় ব্যবহার করা হয় এবং একসাথে লোড হওয়া দরকার, তখন eager loading ব্যবহার করুন। এটি একাধিক ডেটাবেস কলের সংখ্যা কমাতে সাহায্য করে।
  3. Avoid N+1 Query Problem: Lazy loading প্রয়োগ করার সময় N+1 query problem হতে পারে, যেখানে অনেক অতিরিক্ত SQL কুয়েরি তৈরি হয়। Eager loading বা join ব্যবহার করে এটি এড়ানো যেতে পারে।
  4. Use fetchType Appropriately: fetchType কনফিগারেশন ব্যবহার করুন, যা lazy বা eager লোডিং সুবিধা দেয় এবং resultMap এবং association এর মধ্যে লোডিং কৌশল নির্বাচন করতে সাহায্য করে।
  5. Limit the Use of join for Large Data: join ব্যবহার করার সময়, বড় ডেটাসেটের জন্য এটি ভালো হতে পারে না। join সঠিকভাবে ব্যবহৃত না হলে এটি পারফরম্যান্স ইস্যু তৈরি করতে পারে।

Lazy Loading এবং Eager Loading iBATIS (MyBatis)-এ data fetching strategy হিসাবে ব্যবহৃত হয়। Lazy Loading কেবল তখনই সম্পর্কিত ডেটা লোড করে যখন এটি প্রথমবারের মতো প্রয়োজন, যেখানে Eager Loading সম্পর্কিত সমস্ত ডেটা একসাথে লোড করে। iBATIS এ dynamic SQL, fetchType, এবং association ট্যাগ ব্যবহার করে আপনি আপনার ডেটার লোডিং কৌশল কাস্টমাইজ করতে পারেন।

সঠিকভাবে Lazy এবং Eager Loading ব্যবহার করলে আপনি ডেটাবেস অপারেশনগুলির কার্যক্ষমতা এবং স্কেলেবিলিটি উন্নত করতে সক্ষম হবেন।

Content added By

iBATIS (বর্তমানে MyBatis নামে পরিচিত) একটি SQL Mapping Framework যা Java objects এবং SQL স্টেটমেন্টগুলির মধ্যে সম্পর্ক স্থাপন করে। iBATIS (MyBatis) ORM-এর তুলনায় SQL-ভিত্তিক, যেখানে আপনি ম্যানুয়ালি SQL স্টেটমেন্ট লিখে ডেটাবেসের সাথে যোগাযোগ করেন। তবে, MyBatis-এ Lazy Loading এবং Eager Loading এর মতো ফিচার রয়েছে, যা ডেটাবেসের অপটিমাইজড এক্সেস এবং সম্পর্কিত ডেটা লোডিং পরিচালনা করতে সহায়তা করে।

Lazy Loading এবং Eager Loading হল দুটি ভিন্ন পদ্ধতি যা সম্পর্কিত ডেটা লোড করার ক্ষেত্রে ব্যবহৃত হয়। এখানে, আমরা এই দুটি পদ্ধতির ধারণা এবং এগুলির কিভাবে কাজ করে তা আলোচনা করব।


Lazy Loading: ধারণা

Lazy Loading এমন একটি লোডিং পদ্ধতি যেখানে সম্পর্কিত ডেটা তখনই লোড করা হয় যখন তা প্রথমবার প্রয়োজন হয়। অর্থাৎ, কোনো সম্পর্কিত অবজেক্ট (যেমন, Many-to-One, One-to-Many) শুরুতে লোড করা হয় না, বরং access করার সময় তা লোড হয়। এই পদ্ধতিতে performance optimization করা যায় কারণ অপ্রয়োজনীয় ডেটা প্রথমেই লোড করা হয় না।

Lazy LoadingSQL query তখনই রান হয়, যখন ডেটার প্রয়োজন হয়। এটি N+1 Query Problem সৃষ্টি করতে পারে, যেখানে একাধিক SELECT কুয়েরি চালাতে হয় সম্পর্কিত ডেটা লোড করার জন্য।

Lazy Loading উদাহরণ:

ধরা যাক, আমাদের Employee এবং Department এর মধ্যে একটি সম্পর্ক আছে এবং Department সম্পর্কটি Lazy Loading মোডে কনফিগার করা হয়েছে।

@Entity
public class Employee {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    private String name;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "department_id")
    private Department department;

    // Getters and Setters
}

এখানে, FetchType.LAZY ব্যবহৃত হয়েছে, যার মানে হল যে Department অবজেক্টটি প্রথমে লোড হবে না। এটি তখনই লোড হবে যখন employee.getDepartment() কল করা হবে।

Lazy Loading এর সুবিধা:

  • Performance Optimization: ডেটা লোডের সময় কেবলমাত্র প্রয়োজনীয় ডেটা লোড করা হয়, যা পারফরম্যান্স উন্নত করতে সাহায্য করে।
  • Reduced Memory Usage: শুধুমাত্র প্রয়োজনীয় সম্পর্কিত ডেটাই মেমরিতে লোড হয়, যার ফলে মেমরি সঞ্চয়ের সুবিধা পাওয়া যায়।

Lazy Loading এর সমস্যা:

  • N+1 Query Problem: যখন সম্পর্কিত ডেটা লোড করার জন্য একাধিক SELECT কুয়েরি চলে, তখন N+1 Query Problem সৃষ্টি হতে পারে।
  • Performance Issues: যদি ডেটা অ্যাক্সেস করার জন্য অনেক বার লোড করা হয়, তবে এটি পারফরম্যান্সকে খারাপ করতে পারে।

Eager Loading: ধারণা

Eager Loading হল একটি লোডিং পদ্ধতি যেখানে সম্পর্কিত ডেটা প্রথম থেকেই লোড করা হয়, অর্থাৎ যখন প্রধান অবজেক্টটি লোড হয়, তখন তার সম্পর্কিত অবজেক্টগুলোও লোড হয়ে যায়। এই পদ্ধতিতে SQL query একসাথে চলে এবং সমস্ত সম্পর্কিত ডেটা একসাথে লোড হয়।

Eager Loading ব্যবহার করলে, সম্পর্কিত সব ডেটা একসাথে লোড হয়ে যায় এবং এতে N+1 Query সমস্যা দূর করা যায়। তবে, এটি memory এবং performance এর ওপর নেতিবাচক প্রভাব ফেলতে পারে যদি আপনার প্রয়োজন না হয় এমন সমস্ত সম্পর্ক লোড হয়।

Eager Loading উদাহরণ:

ধরা যাক, আমাদের Employee এবং Department এর মধ্যে একটি সম্পর্ক আছে এবং Department সম্পর্কটি Eager Loading মোডে কনফিগার করা হয়েছে।

@Entity
public class Employee {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    private String name;

    @ManyToOne(fetch = FetchType.EAGER)
    @JoinColumn(name = "department_id")
    private Department department;

    // Getters and Setters
}

এখানে, FetchType.EAGER ব্যবহৃত হয়েছে, যার মানে হল যে যখন Employee অবজেক্টটি লোড হবে, তখন Department সম্পর্কিত ডেটাও সাথে সাথে লোড হবে।

Eager Loading এর সুবিধা:

  • No N+1 Query Problem: সমস্ত সম্পর্কিত ডেটা একসাথে লোড হওয়ায় N+1 Query সমস্যা এড়ানো যায়।
  • Simpler Code: সম্পর্কিত ডেটা আগে থেকেই লোড হয়ে থাকে, তাই বারবার ডেটা লোড করার প্রয়োজন নেই, যা কোডের সরলতা বাড়ায়।

Eager Loading এর সমস্যা:

  • Performance Issues: যদি খুব বেশি সম্পর্কিত ডেটা থাকে, তাহলে সমস্ত সম্পর্ক একসাথে লোড হওয়ার ফলে পারফরম্যান্সে খারাপ প্রভাব ফেলতে পারে।
  • Memory Consumption: একাধিক সম্পর্ক একসাথে লোড হলে অতিরিক্ত মেমরি খরচ হতে পারে।

Lazy Loading এবং Eager Loading এর মধ্যে পার্থক্য

CriteriaLazy LoadingEager Loading
Definitionসম্পর্কিত ডেটা তখনই লোড হয় যখন তা প্রাথমিকভাবে প্রয়োজন হয়।সম্পর্কিত ডেটা অবজেক্ট লোড করার সাথে সাথে লোড হয়।
Performanceপারফরম্যান্স উন্নত হতে পারে, কারণ অপ্রয়োজনীয় ডেটা লোড হয় না।প্রথমে ডেটা একসাথে লোড হওয়ায় বেশি মেমরি ব্যবহার হতে পারে।
Memory Usageকম মেমরি ব্যবহার হয় কারণ ডেটা কেবলমাত্র প্রয়োজনীয় সময়ে লোড হয়।মেমরি বেশি ব্যবহার হতে পারে, কারণ সম্পর্কিত সমস্ত ডেটা একসাথে লোড হয়।
Use Caseযখন সম্পর্কিত ডেটা সবসময় প্রয়োজন হয় না।যখন সম্পর্কিত ডেটা সবসময় প্রয়োজন হয়।
Risk of N+1 QueryN+1 Query সমস্যা হতে পারে।N+1 Query সমস্যা এড়ানো যায়।
Complexityকোডে আরও বেশি fetching লজিক থাকতে পারে।সহজ, কারণ সমস্ত সম্পর্ক একসাথে লোড হয়।

MyBatis (iBATIS) তে Lazy এবং Eager Loading

MyBatis (iBATIS) তে Lazy এবং Eager লোডিং কনফিগার করার জন্য, @FetchType.LAZY বা @FetchType.EAGER অ্যানোটেশন ব্যবহার করা যায়, তবে এটি একটি ORM ফিচার নয়। MyBatis-এ, Lazy Loading এবং Eager Loading সাধারণত association mapping এর মাধ্যমে কনফিগার করা হয়, যেমন:

Lazy Loading Example (MyBatis):

<resultMap id="employeeResultMap" type="com.example.Employee">
    <result property="department" column="department_id" fetchType="lazy"/>
</resultMap>

Eager Loading Example (MyBatis):

<resultMap id="employeeResultMap" type="com.example.Employee">
    <result property="department" column="department_id" fetchType="eager"/>
</resultMap>

  • Lazy Loading তখন ব্যবহার করা হয় যখন সম্পর্কিত ডেটা প্রথমে প্রয়োজন হয় না এবং এটি পারফরম্যান্স অপটিমাইজেশনের জন্য উপযোগী। তবে N+1 Query Problem এড়ানোর জন্য অতিরিক্ত কৌশল প্রয়োগ করা হতে পারে।
  • Eager Loading তখন ব্যবহৃত হয় যখন সম্পর্কিত ডেটা সবসময়ই প্রয়োজন এবং আপনি চাইছেন যে ডেটা একসাথে লোড হোক। এটি N+1 Query সমস্যা সমাধান করতে সাহায্য করে, তবে মেমরি ব্যবহারে অতিরিক্ত হতে পারে।

MyBatis (iBATIS) তে Lazy Loading এবং Eager Loading ব্যবহারের মাধ্যমে আপনার অ্যাপ্লিকেশনটি ডেটাবেসের ডেটা অ্যাক্সেসের পারফরম্যান্স এবং মেমরি ব্যবস্থাপনা উন্নত হতে পারে।

Content added By

iBATIS (বর্তমানে MyBatis) একটি জনপ্রিয় ORM ফ্রেমওয়ার্ক যা Lazy Loading-এর সমর্থন প্রদান করে, অর্থাৎ একমাত্র তখনই সম্পর্কিত ডেটা লোড হয় যখন তার প্রয়োজন হয়। Lazy Loading কার্যকরভাবে আপনার অ্যাপ্লিকেশনের পারফরম্যান্স উন্নত করতে সাহায্য করে, কারণ শুধুমাত্র প্রয়োজনীয় ডেটা লোড করা হয় এবং অপ্রয়োজনীয় ডেটা লোড করা এড়ানো হয়।

MyBatis-এ Lazy Loading কনফিগার করার জন্য কিছু গুরুত্বপূর্ণ পদক্ষেপ রয়েছে, যা নীচে বিস্তারিতভাবে আলোচনা করা হয়েছে।


Lazy Loading কী?

Lazy Loading এমন একটি কৌশল, যেখানে একাধিক সম্পর্কিত অবজেক্টগুলোকে একযোগে লোড করার পরিবর্তে, শুধুমাত্র তখনই লোড করা হয় যখন সেই সম্পর্কিত অবজেক্টটি অ্যাক্সেস করা হয়। উদাহরণস্বরূপ, যদি আপনার Order এবং Customer নামক দুটি সম্পর্কিত অবজেক্ট থাকে, তবে Order অবজেক্টটি লোড করার সময় Customer অবজেক্টটি লোড করা হবে না যতক্ষণ না আপনি Customer অবজেক্টটি অ্যাক্সেস না করেন।


Lazy Loading কনফিগারেশন MyBatis-এ

MyBatis-এ Lazy Loading কার্যকর করার জন্য lazyLoadingEnabled কনফিগারেশন ব্যবহার করা হয়।

Step 1: mybatis-config.xml ফাইল কনফিগার করা

MyBatis-এ Lazy Loading কনফিগার করার জন্য, প্রথমে mybatis-config.xml ফাইলে কিছু সেটিংস করতে হবে।

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>

    <!-- Enable Lazy Loading -->
    <settings>
        <setting name="lazyLoadingEnabled" value="true"/>
        <setting name="lazyLoadTriggerMethods" value="get|is"/>
    </settings>

    <!-- DataSource Configuration -->
    <dataSource type="POOLED">
        <property name="driver" value="org.h2.Driver"/>
        <property name="url" value="jdbc:h2:mem:test;DB_CLOSE_DELAY=-1"/>
        <property name="username" value="sa"/>
        <property name="password" value=""/>
    </dataSource>

    <!-- Transaction Manager -->
    <transactionManager type="JDBC"/>

    <!-- SQL Map Files -->
    <sqlMap resource="com/example/mapper/EmployeeMapper.xml"/>
    <sqlMap resource="com/example/mapper/DepartmentMapper.xml"/>

</configuration>

Explanation:

  • lazyLoadingEnabled: এটি true সেট করলে Lazy Loading চালু হয়, অর্থাৎ সম্পর্কিত অবজেক্টগুলির জন্য ডেটা তখনই লোড হবে যখন তার প্রয়োজন হবে।
  • lazyLoadTriggerMethods: এই সেটিংটি নির্ধারণ করে কোন মেথডের মাধ্যমে Lazy Loading ট্রিগার হবে। এখানে get এবং is মেথডগুলো দিয়ে Lazy Loading ট্রিগার হবে।

Step 2: MyBatis Mapper XML Configuration

Lazy Loading ব্যবহার করতে, আপনাকে Mapper XML ফাইলের মধ্যে সম্পর্কিত অবজেক্টগুলো কনফিগার করতে হবে।

EmployeeMapper.xml Example (Lazy Loading):

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//iBATIS//DTD Mapper 3.0//EN" "http://ibatis.apache.org/dtd/sql-map-3.dtd">
<mapper namespace="com.example.mapper.EmployeeMapper">

    <!-- Employee Select Query -->
    <select id="getEmployeeById" resultType="com.example.model.Employee">
        SELECT id, name, department_id
        FROM employee
        WHERE id = #{id}
    </select>

    <!-- Lazy Loading for Department -->
    <select id="getDepartment" resultType="com.example.model.Department">
        SELECT id, name
        FROM department
        WHERE id = #{id}
    </select>

</mapper>

Explanation:

  • Lazy Loading নিশ্চিত করার জন্য, getDepartment মেথডটি Employee অবজেক্টের মধ্যে Department অবজেক্টের লোডিং এর জন্য ব্যবহৃত হবে।
  • এখানে Employee অবজেক্টের মধ্যে department_id কেবল একটি ফিল্ড হিসেবে থাকবে এবং Department অবজেক্টটি Lazy Loaded হবে।

Step 3: Employee এবং Department Model Classes

এখন, আমাদের Java Model Classes তৈরি করতে হবে, যেখানে Employee ক্লাসের মধ্যে Department সম্পর্কিত ডেটা থাকবে এবং Lazy Loading বাস্তবায়ন হবে।

Employee.java:

public class Employee {
    private int id;
    private String name;
    private Department department;  // Lazy-loaded relationship

    // Getters and Setters
    public Department getDepartment() {
        return department;
    }

    public void setDepartment(Department department) {
        this.department = department;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Department.java:

public class Department {
    private int id;
    private String name;

    // Getters and Setters
    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Step 4: Java Code to Test Lazy Loading

এখন, Java কোড ব্যবহার করে Lazy Loading পরীক্ষা করতে হবে।

Lazy Loading Test - EmployeeApp.java:

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;

public class EmployeeApp {

    public static void main(String[] args) {

        // Step 1: Build SqlSessionFactory from configuration file
        SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder()
                .build(EmployeeApp.class.getResourceAsStream("/sql-map-config.xml"));

        // Step 2: Open SqlSession
        try (SqlSession session = sessionFactory.openSession()) {

            // Step 3: Get Employee by ID, this will lazily load the department
            Employee employee = session.selectOne("com.example.mapper.EmployeeMapper.getEmployeeById", 1);

            // Step 4: Lazy load the department when accessed
            System.out.println("Employee Name: " + employee.getName());
            System.out.println("Department Name: " + employee.getDepartment().getName());

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

Explanation:

  • Lazy Loading: যখন আপনি employee.getDepartment() কল করবেন, তখনই Department অবজেক্টটি লোড হবে।
  • getDepartment() মেথডটি Lazy Loading ট্রিগার করে, কারণ MyBatis-এর কনফিগারেশনে আমরা Lazy Loading এনাবল করেছি।

Step 5: Gradle বা Maven দিয়ে প্রোজেক্ট বিল্ড এবং রান করা

Gradle বা Maven দিয়ে আপনার প্রোজেক্টটি বিল্ড এবং রান করতে পারেন।

For Gradle:

# Build the project
gradle build

# Run the project
gradle run

For Maven:

# Build the project
mvn clean install

# Run the project
mvn exec:java

MyBatis (iBATIS) এ Lazy Loading ব্যবহার করার জন্য lazyLoadingEnabled কনফিগারেশন এবং সম্পর্কিত অবজেক্টগুলির জন্য সঠিক কনফিগারেশন প্রয়োজন। Lazy Loading ব্যবহার করলে, আপনি শুধু তখনই সম্পর্কিত অবজেক্টগুলো লোড করবেন যখন সেগুলো অ্যাক্সেস করা হবে, যা অ্যাপ্লিকেশনটির পারফরম্যান্স এবং মেমরি ব্যবস্থাপনা উন্নত করতে সাহায্য করে।

  • lazyLoadingEnabled সেটিংটি কনফিগার করার মাধ্যমে Lazy Loading চালু করা যায়।
  • Mapper XML এবং Java Models এর মাধ্যমে সম্পর্কিত অবজেক্টের Lazy Loading বাস্তবায়ন করা যায়।
Content added By

Lazy Loading একটি ORM (Object-Relational Mapping) কৌশল যা ডেটাবেস সম্পর্কিত অবজেক্টগুলিকে দেরিতে লোড করার জন্য ব্যবহৃত হয়। এটি ডেটাবেসের মধ্যে সম্পর্কিত অবজেক্টগুলো এগজিকিউশন বা লোডিং এর সময় সেকেন্ডারি query চালানোর পরিবর্তে, সেগুলো প্রয়োজন হওয়া পর্যন্ত বিলম্বিত করে। iBATIS (বর্তমানে MyBatis নামে পরিচিত) তে Lazy Loading এর ব্যবহারে কিছু প্রভাব হতে পারে এবং এর পারফরম্যান্স টিউনিংয়ের জন্য বিভিন্ন কৌশলও আছে।

এখানে Lazy Loading এর প্রভাব, সুবিধা ও অসুবিধা এবং পারফরম্যান্স টিউনিং সম্পর্কিত কিছু গুরুত্বপূর্ণ ধারণা তুলে ধরা হয়েছে।


1. Lazy Loading এর প্রভাব

Lazy Loading-এর সুবিধা হল এটি এক সময়ে অপ্রয়োজনীয় ডেটা লোড না করে, শুধুমাত্র যখন প্রয়োজন হয় তখনই ডেটা লোড করে। তবে, এর কিছু নেগেটিভ প্রভাবও থাকতে পারে যদি এটি সঠিকভাবে ব্যবহৃত না হয়।

সুবিধা:

  • Memory Efficiency: Lazy loading অপ্রয়োজনীয় ডেটা লোড না করার মাধ্যমে মেমরি সঞ্চয় করতে সাহায্য করে।
  • Performance: যদি সম্পর্কিত ডেটার সবগুলো কল একসাথে লোড করা না হয়, তবে সিস্টেমের কর্মক্ষমতা বাড়তে পারে কারণ প্রাথমিক সময়ে কেবল প্রয়োজনীয় ডেটাই লোড হবে।
  • Reduce Initial Load Time: ডেটা লোডের সময় কমে, বিশেষ করে বড় ডেটাবেসে, যখন সম্পর্কিত অবজেক্টগুলি বিলম্বিতভাবে লোড হয়।

অসুবিধা:

  • N+1 Query Problem: যখন Lazy Loading ব্যবহার করা হয়, তখন N+1 Query সমস্যা হতে পারে। উদাহরণস্বরূপ, যদি একটি parent অবজেক্টে Nটি সম্পর্কিত child অবজেক্ট থাকে, তবে প্রথমে একটি parent কুয়েরি চালানো হবে এবং তারপর প্রতিটি child এর জন্য আলাদা আলাদা কুয়েরি চালানো হবে।
  • Performance Degradation: যদি খুব বেশি ডেটার উপর Lazy loading প্রয়োগ করা হয়, তবে প্রতিবার অ্যাক্সেস করার সময় অতিরিক্ত SQL কুয়েরি চালাতে হতে পারে, যা পারফরম্যান্সে নেতিবাচক প্রভাব ফেলতে পারে।

2. Lazy Loading এর পারফরম্যান্স টিউনিং

Lazy Loading এর পারফরম্যান্সে প্রভাব কমাতে বা এটি আরও কার্যকরী করতে কিছু টিউনিং কৌশল অনুসরণ করা যেতে পারে।

2.1. Eager Loading with FetchType

এটি একটি জনপ্রিয় কৌশল যেখানে আপনি প্রয়োজনীয় সম্পর্কিত ডেটাগুলো শুরু থেকেই Eagerly লোড করে নেন।

  • Eager Loading: এতে সম্পর্কিত সমস্ত ডেটা একসাথে লোড হয়, যা Lazy loading এর থেকে পারফরম্যান্সের দিক থেকে দ্রুত হতে পারে, তবে এটি মেমরি ব্যবহারে বাড়তি চাপ সৃষ্টি করতে পারে।

Example: MyBatis Mapper XML তে Eager Loading ব্যবহার করা।

<resultMap id="EmployeeResultMap" type="Employee">
    <result property="id" column="id"/>
    <result property="name" column="name"/>
    <result property="department" column="department"/>
</resultMap>

<select id="getEmployeeWithDetails" resultMap="EmployeeResultMap">
    SELECT * FROM employee
</select>

2.2. Use fetchType Attribute:

MyBatis-এ Lazy Loading সক্রিয় বা নিষ্ক্রিয় করার জন্য fetchType সেট করা যায়। সাধারণভাবে, LAZY মান ব্যবহার করা হয়, তবে আপনি চাইলে EAGER ব্যবহার করতে পারেন।

<resultMap id="employeeResultMap" type="Employee">
    <association property="department" column="department_id" fetchType="EAGER"/>
</resultMap>
  • EAGER: সম্পর্কিত ডেটা একসাথে লোড হবে।
  • LAZY: সম্পর্কিত ডেটা পরে লোড হবে যখন তা অ্যাক্সেস করা হবে।

2.3. Use select or join wisely to avoid N+1 Query Problem

Lazy loading এর ফলে N+1 query সমস্যা হতে পারে, যেখানে একাধিক সম্পর্কিত অবজেক্টের জন্য অতিরিক্ত কুয়েরি চালাতে হয়। এই সমস্যা এড়াতে, <join> বা <select> ট্যাগ ব্যবহার করতে পারেন যা একসাথে একাধিক ডেটা ফেচ করে।

<select id="getEmployeeAndDepartment" resultType="Employee">
    SELECT e.id, e.name, d.name AS department
    FROM employee e
    LEFT JOIN department d ON e.department_id = d.id
    WHERE e.id = #{id}
</select>

2.4. Lazy Loading with SqlSession and ResultHandler

Lazy loading এর জন্য SqlSession এর মাধ্যমে ডেটা অ্যাক্সেস করা যায়, এবং আপনি ResultHandler ব্যবহার করে আরও কাস্টম লোডিং কৌশল প্রয়োগ করতে পারেন।

SqlSession session = sqlSessionFactory.openSession();
try {
    Employee employee = session.selectOne("com.example.mapper.EmployeeMapper.getEmployeeWithLazyLoading", id);
    Department department = employee.getDepartment();  // Lazy-loaded
} finally {
    session.close();
}

2.5. Enable Caching

MyBatis তে caching সক্ষম করে, আপনি পুনরায় একই ডেটা লোড করার সময় কর্মক্ষমতা বাড়াতে পারেন।

<settings>
    <setting name="cacheEnabled" value="true"/>
</settings>

এটি MyBatis-এ ক্যাশিং সক্রিয় করে এবং ডেটা পুনরায় অ্যাক্সেস করার সময় ক্যাশে থেকে সরাসরি ডেটা পাওয়া যাবে, ফলে পারফরম্যান্সের উন্নতি হবে।


3. N+1 Query Problem এবং এর সমাধান

Lazy loading ব্যবহারের সময় N+1 Query Problem একটি সাধারণ সমস্যা হতে পারে, যেখানে একটি একক parent কুয়েরি চালানোর পর সম্পর্কিত Nটি child অবজেক্টের জন্য পৃথক পৃথক কুয়েরি চালানো হয়। এই সমস্যার সমাধানে:

Solution 1: Use EAGER fetching for relationships

যখন আপনি জানেন যে সম্পর্কিত অবজেক্টগুলো ব্যবহার হবে, তখন EAGER fetching ব্যবহার করা যেতে পারে, যাতে একসাথে সব ডেটা লোড করা যায়।

Solution 2: Use JOIN instead of Lazy Loading

একটি বড় পরিমাণ ডেটা লোড করার সময় JOIN ব্যবহার করতে পারেন, যাতে সম্পর্কিত ডেটাগুলিকে একসাথে ফেচ করা যায়।

Solution 3: Use Batch operations

যখন একটি বড় লিস্টে লোড করার প্রয়োজন, তখন Batch অপারেশন ব্যবহার করতে পারেন, যাতে একাধিক ডেটাবেস রাউন্ড ট্রিপের সংখ্যা কমিয়ে আনা যায়।

<settings>
    <setting name="jdbc.batchSize" value="50"/>
</settings>

এটি একাধিক INSERT, UPDATE, DELETE অপারেশনকে একসাথে ব্যাচের মধ্যে পরিচালনা করে।


4. Monitoring and Profiling

Lazy loading এর পারফরম্যান্সের প্রভাব দেখতে হলে SQL Profiler ব্যবহার করা যেতে পারে যা ডেটাবেসের কর্মক্ষমতা পর্যবেক্ষণ করতে সহায়ক।

Solution: Profiling Tool

  • MyBatis Logging: আপনি MyBatis এর logging সক্রিয় করে দেখতে পারেন কতগুলি কুয়েরি চালানো হচ্ছে এবং কোন অংশে বিলম্ব হচ্ছে।
  • SQL Profiler: যেমন, JProfiler, Hibernate Profiler, এবং YourKit-এর মতো টুলস ব্যবহার করতে পারেন যাতে সঠিক কুয়েরি এবং সময়মতো ডেটাবেস অপারেশন সনাক্ত করা যায়।
<settings>
    <setting name="logImpl" value="STDOUT_LOGGING"/>
</settings>

Lazy loading একটি শক্তিশালী কৌশল যা মেমরি এবং পারফরম্যান্স অপটিমাইজেশন করতে সহায়ক হতে পারে, তবে এটি কিছু সমস্যাও সৃষ্টি করতে পারে যেমন N+1 Query Problem। iBATIS (MyBatis)-এ Lazy Loading ব্যবহারের সময়, EAGER fetching, JOIN queries, Batch operations, এবং Caching এর মতো কৌশলগুলি ব্যবহার করে আপনি পারফরম্যান্স টিউনিং করতে পারেন এবং ডেটাবেস অপারেশনগুলির কার্যকারিতা উন্নত করতে পারবেন।

Content added By

iBATIS (MyBatis), যা বর্তমানে MyBatis নামে পরিচিত, একটি SQL ম্যাপিং ফ্রেমওয়ার্ক যা SQL কুয়েরি এবং Java objects-এর মধ্যে সম্পর্ক স্থাপন করে। Eager Loading হল একটি ORM (Object-Relational Mapping) কৌশল, যেখানে সম্পর্কিত ডেটা (যেমন ফোরেন কীগুলির মাধ্যমে সম্পর্কিত অবজেক্ট) লোড করা হয় ডেটাবেস থেকে যখন মূল অবজেক্টটি লোড করা হয়। iBATIS/MyBatis-এ Eager Loading সাধারণত join বা fetch কৌশল ব্যবহার করে হয়, যা সম্পর্কিত ডেটা একসাথে লোড করে।

iBATIS (MyBatis)-এ Eager Loading কৌশলের ব্যবহার এবং সেটআপ সম্পর্কে এখানে বিস্তারিত আলোচনা করা হবে।


1. What is Eager Loading?

Eager Loading হল একটি কৌশল যেখানে যখন একটি অবজেক্ট লোড করা হয়, তখন তার সম্পর্কিত অবজেক্টগুলো (যেমন One-to-Many, Many-to-One, বা Many-to-Many) একই সাথে ডেটাবেস থেকে লোড করা হয়। অর্থাৎ, Lazy Loading-এর বিপরীতে, যেখানে সম্পর্কিত অবজেক্টগুলো শুধুমাত্র প্রয়োজন হলে লোড হয়, Eager Loading সম্পর্কিত সমস্ত ডেটা একসাথে লোড করে।

Eager Loading এর সুবিধা:

  • একাধিক সম্পর্কিত অবজেক্টের ডেটা একসাথে লোড করা হয়, ফলে পরবর্তীতে ওই অবজেক্টগুলো রিট্রিভ করতে অতিরিক্ত SQL কোয়েরি চালাতে হয় না।
  • যখন আপনি জানেন যে, সম্পর্কিত ডেটাগুলির প্রয়োজন হবে, তখন এটি পারফরম্যান্সের জন্য কার্যকর হতে পারে।

Eager Loading এর অসুবিধা:

  • যদি সম্পর্কিত ডেটা অনেক বড় বা জটিল হয়, তবে একসাথে সব কিছু লোড করলে performance bottlenecks হতে পারে।
  • অতিরিক্ত ডেটা লোড হওয়া উক্ত কেসে memory ব্যবহার বাড়িয়ে দিতে পারে।

2. iBATIS (MyBatis) Eager Loading Implementation

iBATIS (MyBatis)-এ Eager Loading অর্জন করতে সাধারণত join কুয়েরি ব্যবহার করা হয়, যেখানে সম্পর্কিত টেবিলগুলির ডেটা একসাথে লোড করা হয়।

a) One-to-Many Relationship Eager Loading

ধরা যাক, আমাদের একটি Student ক্লাস রয়েছে, এবং সেই Student ক্লাসের সাথে সম্পর্কিত একাধিক Course অবজেক্টের তথ্য রিট্রিভ করতে হবে। এই ধরনের সম্পর্কের জন্য Eager Loading ব্যবহার করতে পারেন।

Example: SQL Query for One-to-Many Relationship
<select id="getStudentWithCourses" resultMap="studentWithCoursesMap">
    SELECT s.id AS student_id, s.name AS student_name, c.id AS course_id, c.name AS course_name
    FROM student s
    LEFT JOIN course c ON s.id = c.student_id
    WHERE s.id = #{id}
</select>

<resultMap id="studentWithCoursesMap" type="com.example.Student">
    <id property="id" column="student_id"/>
    <result property="name" column="student_name"/>
    <collection property="courses" ofType="com.example.Course">
        <id property="id" column="course_id"/>
        <result property="name" column="course_name"/>
    </collection>
</resultMap>

Explanation:

  • LEFT JOIN: student টেবিলের সাথে course টেবিলের ডেটা যোগ করা হচ্ছে student_id এর মাধ্যমে।
  • <collection>: এটি student এর সাথে সম্পর্কিত সমস্ত course অবজেক্ট লোড করে। এটি Eager Loading এর উদাহরণ, যেখানে একসাথে ছাত্র এবং তার কোর্সের সব তথ্য রিট্রিভ করা হচ্ছে।

b) Many-to-One Relationship Eager Loading

একটি Course অবজেক্টের সাথে সম্পর্কিত Instructor অবজেক্টের তথ্য লোড করার জন্য Many-to-One সম্পর্কের জন্য Eager Loading ব্যবহার করতে হবে।

Example: SQL Query for Many-to-One Relationship
<select id="getCourseWithInstructor" resultType="com.example.Course">
    SELECT c.id AS course_id, c.name AS course_name, i.id AS instructor_id, i.name AS instructor_name
    FROM course c
    LEFT JOIN instructor i ON c.instructor_id = i.id
    WHERE c.id = #{id}
</select>

Explanation:

  • LEFT JOIN: course টেবিলের সাথে instructor টেবিলের ডেটা যোগ করা হচ্ছে instructor_id এর মাধ্যমে।
  • LEFT JOIN ব্যবহারের মাধ্যমে Eager Loading নিশ্চিত করা হচ্ছে, যা course এবং তার সম্পর্কিত instructor এর তথ্য একসাথে লোড করে।

3. Using @FetchType.EAGER Annotation in JPA (When using JPA with iBATIS)

যদি আপনি iBATIS এর সাথে JPA ব্যবহার করছেন, তবে আপনি FetchType.EAGER অ্যানোটেশন ব্যবহার করতে পারেন। এটি Eager Loading নির্ধারণ করতে সাহায্য করে।

Example: Using @ManyToOne with FetchType.EAGER
@Entity
public class Course {

    @ManyToOne(fetch = FetchType.EAGER)
    @JoinColumn(name = "instructor_id")
    private Instructor instructor;

    // Getters and Setters
}

Explanation:

  • @ManyToOne(fetch = FetchType.EAGER): এটি নির্দেশ করে যে, Course অবজেক্টের সাথে সম্পর্কিত Instructor অবজেক্টকে Eagerly লোড করতে হবে, অর্থাৎ Course অবজেক্ট লোড করার সময় তার সম্পর্কিত Instructor অবজেক্টও একসাথে লোড হবে।

4. Eager Loading in iBATIS Using ResultMaps

iBATIS (MyBatis)-এ ResultMap ব্যবহার করে সম্পর্কিত অবজেক্টগুলির Eager Loading করা যায়। আপনি <resultMap> ব্যবহার করে সম্পর্কিত টেবিলের ডেটা ম্যাপ করতে পারেন এবং একটি সম্পর্কিত Java object (বা collection) রিটার্ন করতে পারেন।

Example: Eager Loading with ResultMap

<resultMap id="studentMap" type="com.example.Student">
    <id property="id" column="student_id"/>
    <result property="name" column="student_name"/>
    <collection property="courses" ofType="com.example.Course">
        <id property="id" column="course_id"/>
        <result property="name" column="course_name"/>
    </collection>
</resultMap>

<select id="getStudentWithCourses" resultMap="studentMap">
    SELECT s.id AS student_id, s.name AS student_name, c.id AS course_id, c.name AS course_name
    FROM student s
    LEFT JOIN course c ON s.id = c.student_id
    WHERE s.id = #{id}
</select>

Explanation:

  • <collection>: এটি student অবজেক্টের সাথে সম্পর্কিত course অবজেক্টের একটি সংগ্রহকে লোড করবে। এটি Eager Loading ব্যবহার করে।

5. Advantages and Disadvantages of Eager Loading

Advantages:

  1. Fewer Database Queries: সম্পর্কিত ডেটা একসাথে লোড করা হয়, ফলে পরবর্তীতে আলাদা আলাদা কোয়েরি পাঠানোর প্রয়োজন হয় না।
  2. Performance Boost: যদি সম্পর্কিত ডেটা ব্যবহার করা হয়, তাহলে একসাথে লোড করার ফলে network latency কমে যায় এবং পারফরম্যান্স বাড়ে।

Disadvantages:

  1. Memory Usage: অতিরিক্ত ডেটা একসাথে লোড করার কারণে memory ব্যবহারে বৃদ্ধি হতে পারে।
  2. Complexity in Handling Large Data Sets: যদি সম্পর্কিত ডেটা খুব বড় হয়, তবে একসাথে সব কিছু লোড করলে পারফরম্যান্স সমস্যা তৈরি হতে পারে।

Eager Loading iBATIS (MyBatis)-এ সম্পর্কিত টেবিলের ডেটা একসাথে লোড করার একটি কৌশল যা বিশেষ করে One-to-Many এবং Many-to-One সম্পর্কের ক্ষেত্রে উপকারী। JOIN এবং ResultMap ব্যবহার করে আপনি সহজেই Eager Loading বাস্তবায়ন করতে পারেন। যদিও Eager Loading অনেক ক্ষেত্রে পারফরম্যান্সের জন্য উপকারী, তবে যখন সম্পর্কিত ডেটার পরিমাণ অনেক বেশি হয়, তখন এটি মেমরি সমস্যা সৃষ্টি করতে পারে, সেক্ষেত্রে Lazy Loading ব্যবহার করা হতে পারে বেশি কার্যকর।

Content added By
Promotion

Are you sure to start over?

Loading...