JavaScriptを有効にしてください

【PHP】共有メモリの活用: キャッシュクラスの作成

 ·  ☕ 3 分で読めます

【PHP】共有メモリの活用: キャッシュクラスの作成

PHPでは、共有メモリというメモリの一部を複数のプロセスが共有する機能を利用ができます。
これを利用することで、プロセス間でデータをやり取りしたり、一時的なデータを保存したりすることが可能です。
今回は、この共有メモリを用いてキャッシュの役割を果たすクラスを作成してみました。

注意事項

先に言っておきますが、PHPの場合メモリをローカルメモリに入れる意味はほぼないです。
これは複数台構成などになった場合、各サーバーに別々にキャッシュされるためです。
普通に MemcachedRedis などを使用しましょう。

またキャッシュを永続化してしまった場合、PHPが再起動するまで開放されなくなるのでメモリをずっと確保され続けることになります。
そのためメモリ管理がうまくできないと大変なことになりますのでご注意ください。

環境

PHP8

環境構築

面倒なことにメモリを触る Shmop は通常を使用することができません。
そのため環境にインストールする必要があります。

僕はDocker環境だったので以下をDockerfileに追記するだけですみました。

1
2
# Install shmop
RUN docker-php-ext-install shmop

キャッシュクラスのサンプルコード

実際のメモリキャッシュするサンプルコードになります。

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
<?php

/**
 * 共有メモリを使用したキャッシュクラス
 */
class Cache
{
    /**
     * @var Shmop 共有メモリのID
     */
    private $shm_id;

    /**
     * @var bool 共有メモリの永続性フラグ
     */
    private $is_persistent;

    /**
     * @var bool 共有メモリの永続性フラグ ON
     */
    const PERSISTENT_ON = true;

    /**
     * @var bool 共有メモリの永続性フラグ OFF
     */
    const PERSISTENT_OFF = false;

    /**
     * コンストラクタ
     *
     * @param string $key 共有メモリのキー
     * @param bool $is_persistent 共有メモリの永続性フラグ
     * @param int $max_size 共有メモリの最大サイズ
     */
    public function __construct(string $key, bool $is_persistent = self::PERSISTENT_OFF, int $max_size = 1000)
    {
        $key = ftok(__FILE__, $key);
        $this->shm_id = shmop_open($key, "c", 0644, $max_size);
        $this->is_persistent = $is_persistent;

        if (!$this->shm_id) {
            throw new Exception('Unable to create the shared memory segment.');
        }
    }

    /**
     * 共有メモリにデータと有効期限を書き込む
     *
     * @param mixed $data 書き込むデータ
     * @param int|null $expiry 有効期限(秒数) nullの場合は有効期限なし
     * @throws Exception
     */
    public function put($data, int $expiry = null): void
    {
        $expiryTimestamp = $expiry ? time() + $expiry : null;
        $dataWithExpiry = serialize(['data' => $data, 'expiry' => $expiryTimestamp]);

        if (shmop_write($this->shm_id, $dataWithExpiry, 0) != strlen($dataWithExpiry)) {
            throw new Exception('Was not able to write all of the data into shared memory.');
        }
    }

    /**
     * 共有メモリからデータを取得する
     *
     * @return mixed|null 有効期限内のデータと有効期限の日時秒、データがない場合はnull
     */
    public function get()
    {
        $size = shmop_size($this->shm_id);
        $dataWithExpiry = @unserialize(shmop_read($this->shm_id, 0, $size));

        if ($dataWithExpiry && $dataWithExpiry['expiry'] !== null && $dataWithExpiry['expiry'] < time()) {
            return null;
        }

        return $dataWithExpiry['data'] ?? null;
    }

    /**
     * 共有メモリを削除する
     */
    public function delete(): void
    {
        shmop_delete($this->shm_id);
    }

    /**
     * デストラクタ
     */
    public function __destruct()
    {
        if (!$this->is_persistent) {
            $this->delete();
        }
    }
}

// キャッシュクラスの使用例
$cache = new Cache('a', Cache::PERSISTENT_ON);
$cache->put("Hello, World!", 60);
var_dump($cache->get());

// 他の場所での使用例
$cache2 = new Cache('b', Cache::PERSISTENT_OFF);
var_dump($cache2->get());

参考

  • ftok: この関数は、ファイルのパス名とプロジェクトIDからシステムIDを生成します。このIDは共有メモリセグメントの生成や、セマフォの作成などに使用されます。
  • shmop_open: この関数は、共有メモリセグメントを作成または開きます。既に存在するセグメントを開くには、ftokで生成したシステムIDと同じIDを指定します。
  • shmop_write: この関数は、共有メモリセグメントにデータを書き込みます。
  • shmop_read: この関数は、共有メモリセグメントからデータを読み込みます。
  • shmop_size: この関数は、共有メモリセグメントのサイズを取得します。
  • shmop_delete: この関数は、共有メモリセグメントを削除します。
  • serialize: この関数は、PHPの値を保存や送信可能な文字列の表現に変換します。
  • unserialize: この関数は、serializeで生成された文字列表現からPHPの値を再構築します。
共有

こぴぺたん
著者
こぴぺたん
Copy & Paste Engineer