Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[12.x] Add Eloquent memory optimization features for large datasets #55215

Draft
wants to merge 2 commits into
base: 12.x
Choose a base branch
from

Conversation

Amirhf1
Copy link
Contributor

@Amirhf1 Amirhf1 commented Mar 30, 2025

Memory Optimization in Eloquent

This page introduces new methods for memory optimization in Eloquent that help reduce memory consumption when working with large datasets.

Memory-Efficient Queries

chunkByMemory

The new chunkByMemory method allows you to process data by automatically adjusting chunk sizes based on memory usage:

User::query()->chunkByMemory(100, function ($users, $page) {
    // Process each batch of users
    foreach ($users as $user) {
        // Perform operations on each user
    }
});

The first parameter specifies the maximum allowed memory consumption in megabytes. The method automatically adjusts chunk sizes based on real-time memory usage.

lazyByMemory

The lazyByMemory method works similarly to lazy, but with automatic chunk size adjustment based on memory:

User::query()->lazyByMemory(50)->each(function ($user) {
    // Process each user
});

This method creates a LazyCollection that loads records lazily with smart memory management.

Relationship Optimization

withLimited

The new withLimited method allows you to load relationships with limits:

User::withLimited('posts', ['posts' => 10, 'posts.comments' => 5])->get();

This loads users with a maximum of 10 latest posts and 5 latest comments per post.

Model Optimization

optimizeMemory

The optimizeMemory method lets you remove unnecessary relationships and attributes from models:

$user = User::with('posts', 'comments', 'profile')->first();

// Keep only posts relationship and specified attributes
$user->optimizeMemory(['posts'], ['id', 'name', 'email']);

cleanup

The cleanup method removes all relationships and extra attributes from a model:

$user = User::with('posts', 'comments', 'profile')->first();

// Clean up after working with relationships
$user->cleanup();

optimizedCollection

The static optimizedCollection method creates a memory-optimized collection of models:

$users = User::optimizedCollection($usersArray, ['posts'], ['id', 'name']);

Practical Examples

Large Data Processing

// Process millions of records with low memory usage
User::query()->lazyByMemory(100)
    ->filter(function ($user) {
        return $user->is_active;
    })
    ->each(function ($user) {
        ProcessUserJob::dispatch($user->id);
    });

Memory-Efficient Relationship Loading

// Load shop data with multiple optimized relationships
$shop = Shop::withLimited([
    'products',
    'orders',
    'customers',
], [
    'products' => 100,       // Last 100 products
    'orders' => 50,          // Last 50 orders
    'orders.items' => 10,    // Last 10 items per order
    'customers' => 200,      // Last 200 customers
])->findOrFail($shopId);

Big Data Reporting

// Generate reports from large datasets with optimized memory
$report = [];

Order::query()->chunkByMemory(50, function ($orders, $page) use (&$report) {
    foreach ($orders as $order) {
        // Perform report calculations
        $report[$order->status] = ($report[$order->status] ?? 0) + $order->total;
        
        // Clean up memory after processing
        $order->cleanup();
    }
});

Benefits and Comparisons

Feature Standard Method Optimized Method Memory Reduction
Batch Processing chunk() chunkByMemory() Up to 50%
Lazy Loading lazy() lazyByMemory() Up to 60%
Relationship Loading with() withLimited() Up to 70%
Model Management Standard optimizeMemory() Up to 40%

Tips and Recommendations

  1. Use chunkByMemory for cron jobs and long-running tasks
  2. Use lazyByMemory for APIs returning large datasets
  3. Use withLimited when working with complex relationships
  4. Use cleanup after processing large datasets to free memory
  5. Always use select() to remove unnecessary columns in complex queries

@Amirhf1 Amirhf1 marked this pull request as draft March 30, 2025 16:29
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant