Share This
//PHP Generators

PHP Generators

Đã bao giờ các bạn gặp phải tình huống này chưa?
Đây là lỗi mà bất cứ lập trình viên PHP nào cũng có thể gặp nếu tiến trình đang thực thi sử dụng bộ nhớ RAM vượt quá giới hạn cho phép.

Chúng ta cùng đi đến 1 ví dụ cụ thể nhé:

private function range() {
    $a = [];
 
    for ($i = 0; $i < PHP_INT_MAX; $i++) {
        $a[] = $i;
    }
 
    return $a;
}
 
foreach ($this->range() as $value) {
    echo $value;
}

Đoạn code trên sẽ tạo một array từ 0 đến PHP_INT_MAX (là giá trị số nguyên cực đại mà version PHP hiện tại cung cấp) và in giá trị ra màn hình. Khi chúng ta chạy đoạn code trên thì sẽ gặp phải lỗi như đã đề cập trước đó.

Vậy làm sao để có thể giải quyết lỗi này nếu như vô tình gặp phải? Có 1 cách là tăng giới hạn bộ nhớ “memory_limit” lên trong config php.ini.

; Maximum amount of memory a script may consume
; https://php.net/memory-limit
 memory_limit = 128M

Nhưng đây không phải là cách tối ưu để giải quyết vấn đề. Chúng ta chỉ mới chạy 1 đoạn code nhỏ đã xảy ra lỗi tràn bộ nhớ thì có vẻ không ổn lắm. Vậy nên, thay vì tăng giới hạn bộ nhớ thì chúng ta hãy thử sử dụng PHP Generators để giảm bộ nhớ được sử dụng xem hiệu quả như thế nào nhé!

PHP GENERATORS LÀ GÌ?

PHP Generators là một tính năng mạnh mẽ được giới thiệu trong PHP 5.5. Tính năng này cho phép bạn tạo ra các hàm generator, các hàm generator không trả về toàn bộ danh sách giá trị cùng một lúc, thay vào đó hàm sẽ tạo và trả về từng giá trị một khi cần thiết. Chính điều này giúp tiết kiệm bộ nhớ đáng kể, cho phép xử lý dữ liệu mà không cần lưu trữ toàn bộ tập dữ liệu trong bộ nhớ.

SỬ DỤNG PHP GENERATORS

Quay lại đoạn code trên chúng ta hãy thay đổi ở hàm “range()”, thay vì tạo 1 array (chuỗi) và return (trả về) tất cả giá trị trong array thì chúng ta chỉ lặp qua các giá trị với từ khóa yield. Có thể hiểu đơn giản yield sẽ trả về giá trị nhưng chỉ khi ta gọi đến nó mà không cần phải lưu tất cả giá trị vào trong bộ nhớ như đoạn code ban đầu và lúc này function của chúng ta sẽ trở thành 1 generator function.

private function range() {
    for ($i = 0; $i < PHP_INT_MAX; $i++) {
        yield $i;
    }
}
 
foreach ($this->range() as $value) {
    echo $value;
}

CÁC PHƯƠNG THỨC CỦA PHP GENERATORS

public current(): mixed
public getReturn(): mixed
public key(): mixed
public next(): void
public rewind(): void
public send(mixed $value): mixed
public throw(Throwable $exception): mixed
public valid(): bool
public __wakeup(): void

CÁCH THỨC HOẠT ĐỘNG

Hãy thử 1 ví dụ khác:

function myGeneratorFunction()
{
      echo 'One' . "<br>";
      yield 'first return value' . "<br>";
 
      echo 'Two' . "<br>";
      yield 'second return value' . "<br>";
 
      echo 'my return value' ;
}
 
$iterator = myGeneratorFunction();
 
$firstValue = $iterator->current();
echo $firstValue;
 
$iterator->next();
$secondValue = $iterator->current();
echo $secondValue;
 
$iterator->next();
echo $iterator->getReturn();

Output

One
first return value
Two
second return value
my return value

Khi chúng ta gọi $iterator->current() thì function sẽ thực thi và hiển thị ra giá trị “One”, sau đó sẽ dừng lại khi gặp từ khóa yield đầu tiên. Giá trị của từ khóa yield đầu tiên sẽ được trả về và gán vào $firstValue.

Kế tiếp khi ta gọi $iterator->next(); thì generator sẽ thực thi function trước đó ngay tại điểm mà chúng ta đã dừng ở từ khóa yield đầu tiên. Lúc này sẽ in ra màn hình giá trị ‘Two’ và trả về giá trị của từ khóa yield thứ 2 vào $secondValue. Tiếp tục như thế, function sẽ lấy giá trị trả về từ keyword return để gọi $iterator->next(); và dùng $iterator->getReturn(); lấy giá trị trả về.

Qua ví dụ trên chúng ta có thể hiểu đơn giản là từ khóa yield như 1 nút pause function, function sẽ tạm dừng thực thi khi gặp từ khóa yield này và sẽ tiếp tục khi chúng ta duyệt qua bằng cách gọi $iterator->next().

RETURNING KEYS

Sử dụng yield trả về key ⇒ value

private function range()
{
    for ($i = 0; $i < PHP_INT_MAX; $i++) {
        yield "key {$i}" => "value {$i}";
    }
}
 
foreach ($this->rangeKeyValue() as $key => $value) {
  echo "{$key} -> {$value} <br>";
}

Output

key 0 -> value 0
key 1 -> value 1
key 2 -> value 2
key 3 -> value 3
key 4 -> value 4
key 5 -> value 5
…

TRUYỀN THAM SỐ VÀO TRONG GENERATORS

Chúng ta có thể truyền tham số vào generators để làm 1 việc gì đó như dừng function lại chẳng hạn, cách làm như sau:

private function range()
{
    for ($i = 0; $i < PHP_INT_MAX; $i++) {
         $value = yield $i;
 
         if ($value === 'stop') {
            return;
         }
    }
}
 
$generator = $this->range();
 
foreach ($generator as $value) {
    if ($value === 5) {
        $generator->send('stop');
    }
 
    echo "current value is {$value} <br>";
}

Output

current value is 0
current value is 1
current value is 2
current value is 3
current value is 4
current value is 5

LỢI ÍCH SỬ DỤNG

Khi ứng dụng của chúng ta cần xử lý một file log với kích thước lên tới hàng gigabyte để lưu vào database thì việc đọc hết nội dung trong file log một lần để xử lý sẽ dẫn đến lỗi bị tràn bộ nhớ. Việc sử dụng PHP Generators để thực hiện đọc lần lượt từng dòng trong file log để xử lý sẽ giảm thiểu tối đa bộ nhớ được sử dụng giúp cho code của chúng ta tránh được việc chiếm quá nhiều tài nguyên của hệ thống.

KẾT LUẬN

Đối với những hệ thống cần xử lý một lượng data lớn, việc cân nhắc và lựa chọn cách làm sao cho an toàn và tốt nhất là điều cần thiết.
PHP Generators cung cấp cho chúng ta 1 giải pháp mạnh mẽ nhằm tiết kiệm bộ nhớ, giúp server của chúng ta chạy mượt mà và ổn định. Cũng như tránh những lỗi quá tải có thể xảy đến với hệ thống, có thể làm gián đoạn trải nghiệm, gây mất thiện cảm của người dùng trong quá trình sử dụng.

Huỳnh Hữu Phát
Web Developer

APPLY NOW






    Benefits

    SALARY & BONUS POLICY

    RiverCrane Vietnam sympathizes staffs' innermost feelings and desires and set up termly salary review policy. Performance evaluation is conducted in June and December and salary change is conducted in January and July every year. Besides, outstanding staffs receive bonus for their achievements periodically (monthly, yearly).

    TRAINING IN JAPAN

    In order to broaden staffs' view about technologies over the world, RiverCrane Vietnam set up policy to send staffs to Japan for study. Moreover, the engineers can develop their career paths in technical or management fields.

    ANNUAL COMPANY TRIP

    Not only bringing chances to the staffs for their challenging, Rivercrane Vietnam also excites them with interesting annual trips. Exciting Gala Dinner with team building games will make the members of Rivercrane connected closer.

    COMPANY'S EVENTS

    Activities such as Team Building, Company Building, Family Building, Summer Holiday, Mid-Autum Festival, etc. will be the moments worthy of remembrance for each individual in the project or the pride when one introduces the company to his or her family, and shares the message "We are One".

    INSURANCE

    Rivercrane Vietnam ensures social insurance, medical insurance and unemployment insurance for staffs. The company commits to support staffs for any procedures regarding these insurances. In addition, other insurance policies are taken into consideration and under review.

    OTHER BENEFITS

    Support budget for activities related to education, entertainment and sports. Support fee for purchasing technical books. Support fee for getting engineering or language certificates. Support fee for joining courses regarding technical management. Other supports following company's policy, etc.

    © 2012 RiverCrane Vietnam. All rights reserved.

    Close