シェア
//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

今すぐ応募







    福利厚生

    給料・ボーナス制度

    社員の感情・願望を理解しているので、リバークレーンベトナムは特に年2回の定期昇給制度を設けています。毎年6月と12月に評価を行い、毎年1月と7月に給与が変更されます。また、社員は月次と年次の優秀な個人には定期的な業績賞与が別で支給されます。

    日本研修制度

    世界中の新しい技術分野に触れるために、社員を日本にオンサイトさせる方針があります。さらに、技術分野か管理分野かのキャリアパスは社員の決定次第です。

    毎年社員旅行

    リバークレイン・ベトナムは、スタッフに挑戦の機会を提供するだけでなく、年に一度の魅力的な旅行で彼らを楽しませています。エキサイティングなガラディナーやチームビルディングゲームは、リバークレインのメンバー同士の絆をさらに深める手助けをします。

    リバークレイン イベント

    チームビルディング・ファミリーデー・お夏休み・中秋節などのイベントはチーム内のメンバーが接続出来るしお互いに自分のことを共有出来る機会です。ご家族員に連携する際にはそれも誇りに言われています。

    社会保険制度

    リバークレーンベトナムは従業員に社会保険、医療保険、失業手当などの社会保険制度があります。当社は、これらの保険に関するあらゆる手続きをスタッフに必ずサポートしています。さらに、他の保険契約も考慮され、検討されています。

    他福利

    社員向けの活動をサポートすることもあります。 ・文化・芸術・スポーツクラブの運営費用 ・技術研究の教科書を購入する金額 ・エンジニア試験・言語能力試験を受験料 ・ソフトスキルのセミナー・コースの参加費 ・等 また会社政策通り、他のベネフィットもあります。

    © 2012 RiverCrane Vietnam. All rights reserved.

    Close