目次
キャッシュ削除戦略
タイトル LRU 原則
Redis乐观锁实现秒杀
ホームページ データベース Redis Redis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法

Redis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法

Jun 03, 2023 pm 04:05 PM
redis

    キャッシュ削除戦略

    タイトル LRU 原則

    LRU (最も最近使用されていない、最も最近使用されていない) アルゴリズムは、履歴アクセスに基づいています。データの記録 データを排除するための中心的な考え方は、「データに最近アクセスされた場合、将来アクセスされる可能性も高い」ということです。

    最も一般的な実装は、リンクされたリストを使用してキャッシュされたデータを保存することです。詳細なアルゴリズムは次のように実装されます:

    Redis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法

    • ## 新しいデータがリンク リストの先頭に挿入されます;

    • キャッシュがヒットする (つまり、キャッシュされたデータがアクセスされる) たびに、データはリンク リストの先頭;

    • リンク リストがいっぱいの場合、リンク リストの末尾のデータを破棄します。

    Java では、LinkHashMap を使用して、ハッシュ リンク リストの実装を使用して LRU を実装できます。

    Redis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法

    Title Redis キャッシュの削除戦略

    最大キャッシュの設定

    Redis では、ユーザーは最大メモリ サイズ maxmemory を設定できます。デフォルトは 0 です。最大キャッシュは指定されていません。新しいデータが追加されると、最大メモリを超えると Redis がクラッシュするため、設定する必要があります。

    redis メモリ データ セットのサイズが特定のサイズに増加すると、データ削除戦略が実装されます。

    排除戦略

    redis 排除戦略の構成: maxmemory-policy voltile-lru、ホット構成をサポート

    redis は 6 種類のデータを提供します削除戦略:

    • volatile-lru: 有効期限が設定されている、最も最近使用されていないデータ セット (server.db[i].expires) を選択します。データ削除

    • volatile-ttl:有効期限を設定したデータセット(server.db[i].expires)から期限切れとなるデータを選択します.

    • volatile-random: 有効期限が設定されているデータセット (server.db[i].expires) から削除するデータをランダムに選択します

    • allkeys-lru: データ セット (server.db[i].dict) から最も最近使用されていないデータを選択して、

    • を削除します
    • allkeys-random: データ セット (server.db[i].dict) からデータをランダムに選択して、

    • no-enviction を排除します。 (エビクション): データのエビクションを禁止します

    Redis トランザクション

    Redis トランザクションの概要

    • Redis トランザクションは終了しますMULTI、EXEC、DISCARD、WATCH、UNWATCH の 5 つのコマンドを完了します。

    • Redis の個々のコマンドはアトミックであるため、トランザクション オブジェクトがコマンド セットであることを確認する必要があります。

    • Redis はコマンド セットをシリアル化し、同じトランザクション内でコマンド セットが継続的かつ中断なく実行されるようにします

    • Redis は return Roll をサポートしていません手術。トランザクション コマンド

    MULTI

    は、トランザクション ブロックの開始をマークするために使用されます。 Redis は後続のコマンドを 1 つずつキューに入れ、アトミック EXEC コマンドを使用してこのコマンド シーケンスを実行します。

    構文:

    multi
    ログイン後にコピー

    EXEC

    トランザクション内で以前にキューに入れられたすべてのコマンドを実行し、通常の接続状態を復元します

    構文:

    exec
    ログイン後にコピー

    DISCARD

    トランザクション内で以前にキューに入れられたすべてのコマンドをクリアし、通常の接続状態を復元します。

    構文:

    discard
    ログイン後にコピー

    WATCH

    [トランザクションを条件付きで実行する必要がある]場合、このコマンドを使用して、指定された [ キーが設定されているを変更します[監視対象へ] ステータス。

    構文:

    watch key [key…]
    ログイン後にコピー

    注: このコマンドを使用して、Redis の楽観的ロックを実装します。

    UNWATCH

    トランザクションに対して以前に監視されていたすべてのキーをクリアします

    構文:

    unwatch
    ログイン後にコピー

    コマンドの図:

    Redis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法

    トランザクションのデモ:

    127.0.0.1:6379> multi
    OK
    127.0.0.1:6379> set s1 111
    QUEUED
    127.0.0.1:6379> hset set1 name zhangsan
    QUEUED
    127.0.0.1:6379> exec
    1) OK
    2) (integer) 1
    127.0.0.1:6379> multi
    OK
    127.0.0.1:6379> set s2 222
    QUEUED
    127.0.0.1:6379> hset set2 age 20
    QUEUED
    127.0.0.1:6379> discard
    OK
    127.0.0.1:6379> exec (error) ERR EXEC without MULTI 
    127.0.0.1:6379> watch s1
    OK
    127.0.0.1:6379> multi
    OK
    127.0.0.1:6379> set s1 555
    QUEUED 127.0.0.1:6379> exec # 此时在没有exec之前,通过另一个命令窗口对监控的s1字段进行修改 
    (nil)
    127.0.0.1:6379> get s1
    111
    ログイン後にコピー

    Redis はトランザクションのロールバックをサポートしていません (理由)

    大多数のトランザクション失敗の原因は構文エラーまたは型エラーです。これらのエラーはどちらも開発段階で予見可能です。Redis はパフォーマンス上の理由からトランザクションのロールバックを無視します。

    Redis Optimistic Lock

    Optimistic Lock は CAS (Compare And Swap) のアイデア (比較と置換) に基づいており、相互排他的ではなく、ロック待機を引き起こしたりリソースを消費したりすることはありませんが、再試行を繰り返す必要がありますが、再試行メカニズムにより、より高速に応答できます。したがって、redis を使用して

    オプティミスティック ロックを実装できます。具体的なアイデアは次のとおりです。

    • redis の watch 関数を使用して、この redisKey のステータス値を監視します。

    • Get the redisKey の値

    • redis トランザクションの作成

    • このキーの値を 1

      にします

    • 然后去执行这个事务,如果key的值被修改过则回滚,key不加1

    public void watch() {
    	try {
    		String watchKeys = "watchKeys";
    		//初始值 value=1
    		jedis.set(watchKeys, 1);
    		//监听key为watchKeys的值
    		jedis.watch(watchkeys);
    		//开启事务
    		Transaction tx = jedis.multi();
    		//watchKeys自增加一
    		tx.incr(watchKeys);
    		//执行事务,如果其他线程对watchKeys中的value进行修改,则该事务将不会执行
    		//通过redis事务以及watch命令实现乐观锁
    		List<Object> exec = tx.exec();
    		if (exec == null) {
    			System.out.println("事务未执行");
    		} else {
    			System.out.println("事务成功执行,watchKeys的value成功修改");
    		}
    	} catch (Exception e) {
    		e.printStackTrace();
    	} finally {
    		jedis.close();
    	}
    }
    ログイン後にコピー

    Redis乐观锁实现秒杀

    public class RedisLock {
        public static void main(String[] arg) {
            //库存key 
            String redisKey = "stock";
            ExecutorService executorService = Executors.newFixedThreadPool(20);
            try {
                Jedis jedis = new RedisProperties.Jedis("127.0.0.1", 6378);
                // 可以被秒杀的库存的初始值,库存总共20个
                jedis.set(redisKey, "0");
                jedis.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
            for (int i = 0; i < 1000; i++) {
                executorService.execute(() -> {
                    Jedis jedis1 = new Jedis("127.0.0.1", 6378);
                    try {
                        jedis1.watch(redisKey);
                        String redisValue = jedis1.get(redisKey);
                        int valInteger = Integer.valueOf(redisValue);
                        String userInfo = UUID.randomUUID().toString();
                        // 没有秒完
                        if (valInteger < 20) {
                            Transaction tx = jedis1.multi();
                            tx.incr(redisKey);
                            List list = tx.exec();
                            // 秒成功 失败返回空list而不是空
                            if (list != null && list.size() > 0) {
                                System.out.println("用户:" + userInfo + ",秒杀成 功!当前成功人数:" + (valInteger + 1));
                            }
                            // 版本变化,被别人抢了。
                            else {
                                System.out.println("用户:" + userInfo + ",秒杀失 败");
                            }
                        }
                        // 秒完了
                        else {
                            System.out.println("已经有20人秒杀成功,秒杀结束");
                        }
                    } catch (Exception e) {
                        e.printStackTrace();
                    } finally {
                        jedis1.close();
                    }
                });
            }
            executorService.shutdown();
        }
    }
    ログイン後にコピー

    以上がRedis キャッシュ削除戦略とトランザクションを使用して楽観的ロックを実装する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

    このウェブサイトの声明
    この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

    ホットAIツール

    Undresser.AI Undress

    Undresser.AI Undress

    リアルなヌード写真を作成する AI 搭載アプリ

    AI Clothes Remover

    AI Clothes Remover

    写真から衣服を削除するオンライン AI ツール。

    Undress AI Tool

    Undress AI Tool

    脱衣画像を無料で

    Clothoff.io

    Clothoff.io

    AI衣類リムーバー

    Video Face Swap

    Video Face Swap

    完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

    ホットツール

    メモ帳++7.3.1

    メモ帳++7.3.1

    使いやすく無料のコードエディター

    SublimeText3 中国語版

    SublimeText3 中国語版

    中国語版、とても使いやすい

    ゼンドスタジオ 13.0.1

    ゼンドスタジオ 13.0.1

    強力な PHP 統合開発環境

    ドリームウィーバー CS6

    ドリームウィーバー CS6

    ビジュアル Web 開発ツール

    SublimeText3 Mac版

    SublimeText3 Mac版

    神レベルのコード編集ソフト(SublimeText3)

    Redisクラスターモードの構築方法 Redisクラスターモードの構築方法 Apr 10, 2025 pm 10:15 PM

    Redisクラスターモードは、シャードを介してRedisインスタンスを複数のサーバーに展開し、スケーラビリティと可用性を向上させます。構造の手順は次のとおりです。異なるポートで奇妙なRedisインスタンスを作成します。 3つのセンチネルインスタンスを作成し、Redisインスタンスを監視し、フェールオーバーを監視します。 Sentinel構成ファイルを構成し、Redisインスタンス情報とフェールオーバー設定の監視を追加します。 Redisインスタンス構成ファイルを構成し、クラスターモードを有効にし、クラスター情報ファイルパスを指定します。各Redisインスタンスの情報を含むnodes.confファイルを作成します。クラスターを起動し、CREATEコマンドを実行してクラスターを作成し、レプリカの数を指定します。クラスターにログインしてクラスター情報コマンドを実行して、クラスターステータスを確認します。作る

    Redisデータをクリアする方法 Redisデータをクリアする方法 Apr 10, 2025 pm 10:06 PM

    Redisデータをクリアする方法:Flushallコマンドを使用して、すべての重要な値をクリアします。 FlushDBコマンドを使用して、現在選択されているデータベースのキー値をクリアします。 [選択]を使用してデータベースを切り替え、FlushDBを使用して複数のデータベースをクリアします。 DELコマンドを使用して、特定のキーを削除します。 Redis-CLIツールを使用してデータをクリアします。

    Redisキューの読み方 Redisキューの読み方 Apr 10, 2025 pm 10:12 PM

    Redisのキューを読むには、キュー名を取得し、LPOPコマンドを使用して要素を読み、空のキューを処理する必要があります。特定の手順は次のとおりです。キュー名を取得します:「キュー:キュー」などの「キュー:」のプレフィックスで名前を付けます。 LPOPコマンドを使用します。キューのヘッドから要素を排出し、LPOP Queue:My-Queueなどの値を返します。空のキューの処理:キューが空の場合、LPOPはnilを返し、要素を読む前にキューが存在するかどうかを確認できます。

    Redisロックの使用方法 Redisロックの使用方法 Apr 10, 2025 pm 08:39 PM

    Redisを使用して操作をロックするには、setnxコマンドを介してロックを取得し、有効期限を設定するために有効期限コマンドを使用する必要があります。特定の手順は次のとおりです。(1)SETNXコマンドを使用して、キー価値ペアを設定しようとします。 (2)expireコマンドを使用して、ロックの有効期限を設定します。 (3)Delコマンドを使用して、ロックが不要になったときにロックを削除します。

    Redisコマンドの使用方法 Redisコマンドの使用方法 Apr 10, 2025 pm 08:45 PM

    Redis指令を使用するには、次の手順が必要です。Redisクライアントを開きます。コマンド(動詞キー値)を入力します。必要なパラメーターを提供します(指示ごとに異なります)。 Enterを押してコマンドを実行します。 Redisは、操作の結果を示す応答を返します(通常はOKまたは-ERR)。

    Redisのソースコードを読み取る方法 Redisのソースコードを読み取る方法 Apr 10, 2025 pm 08:27 PM

    Redisソースコードを理解する最良の方法は、段階的に進むことです。Redisの基本に精通してください。開始点として特定のモジュールまたは機能を選択します。モジュールまたは機能のエントリポイントから始めて、行ごとにコードを表示します。関数コールチェーンを介してコードを表示します。 Redisが使用する基礎となるデータ構造に精通してください。 Redisが使用するアルゴリズムを特定します。

    Redisコマンドラインの使用方法 Redisコマンドラインの使用方法 Apr 10, 2025 pm 10:18 PM

    Redisコマンドラインツール(Redis-Cli)を使用して、次の手順を使用してRedisを管理および操作します。サーバーに接続し、アドレスとポートを指定します。コマンド名とパラメーターを使用して、コマンドをサーバーに送信します。ヘルプコマンドを使用して、特定のコマンドのヘルプ情報を表示します。 QUITコマンドを使用して、コマンドラインツールを終了します。

    Centos RedisでLUAスクリプト実行時間を構成する方法 Centos RedisでLUAスクリプト実行時間を構成する方法 Apr 14, 2025 pm 02:12 PM

    Centosシステムでは、Redis構成ファイルを変更するか、Redisコマンドを使用して悪意のあるスクリプトがあまりにも多くのリソースを消費しないようにすることにより、LUAスクリプトの実行時間を制限できます。方法1:Redis構成ファイルを変更し、Redis構成ファイルを見つけます:Redis構成ファイルは通常/etc/redis/redis.confにあります。構成ファイルの編集:テキストエディター(VIやNANOなど)を使用して構成ファイルを開きます:sudovi/etc/redis/redis.conf luaスクリプト実行時間制限を設定します。

    See all articles