最近ビジネス開発に夢中になっていて、しばらくブログを更新していませんでした。ビジネス シナリオでの実践的なソリューションや、より良いデザインのアイデアを今後の記事で取り上げる予定です。ブログ投稿。共有は、以前のトピックに関連するコンテンツを整理するために多くの時間を費やすようなものではありません (究極のトリックを保留します)。その後の記事の内容はそれほど豊富ではないかもしれませんが、次のような 1 つの点についてより詳しく説明できます。ブログ共有の頻度を増やしながら、継続的な共有と自己レビュー、経験の蓄積を通じて、可能な限り多くのことを行う、またはより詳細な分析を行う
シナリオ
シナリオ 1
留言功能限制,30秒 内只能评论 10次,超出次数不让能再评论,并提示:过于频繁
シナリオ 2
点赞功能限制,10秒 内只能点赞 10次,超出次数后不能再点赞,并禁止操作 1个小时,提示:过于频繁,被禁止操作1小时
シナリオ 3
上传记录功能,限制一天只能上传 100次,超出次数不让能再上传,并提示:超出今日上线
本質を抽象化する
事業開発の過程で、私たちは常にさまざまなビジネスシナリオの設計に携わり、非常によく似た問題シナリオに遭遇することがよくありますが、現在属しているビジネス モジュールは異なります。実際、これらの要件の本質は同じ問題を解決することです。このシナリオに遭遇した場合、本質的な問題を抽出する必要があります。独自の経験分析に基づいて要件を抽出し、普遍的なソリューションを実装することで、ソリューションの価値を高めることができます。これが、あなたが魂を持ったエンジニアになるか、最強のCP(コピペ)王になるかの違いかもしれません。
上記の 3 つのビジネス シナリオを分析すると、類似したロジックがあり、類似した問題と呼ばれていることがわかります。次に、この問題を分離し、一般的な解決策を設計し、同じロジック フローチャートの概要を説明します。
##上記の需要シナリオを分析することで、すべての需要シナリオが必要とする条件を抽出できます。 制限対象: ユーザーが制限された操作 (コメント、いいね!)、レコード、 ...) 時間範囲内で(最小时间单位用秒:天/小时/分钟都可换算成秒,用秒可以解决更多的场景)
関数が一般関数に抽出される場合、次のようになりますか:
<?php/** * 频率限制 * @param string $action 操作动作 * @param int $userId 发起操作的用户ID * @param int $time 时间范围X秒内 * @param int $number 限制操作数Y次 * @param array $expire 超出封印时间Z ['type'=>1,'ttl'=>过期时间/秒] ['type'=>2,'ttl'=>具体过期时间戳] 二选一 * @return bool * @throws \Exception */public static function frequencyLimit(string $action, int $userId, int $time, int $number, $expire = []){ // todo 根据用户操作动作时间范围,进行频率的控制和失效释放}
ソリューションの実装
関数は、ユーザーが開始した操作と時間、ストレージの累積回数、有効期限をクリーンアップする必要があるこの時点でストレージを mysql に依存する場合、それを考えるのは非常に面倒になります。ついに登場したredis redisの特性を踏まえ、incrとkeyのアトミックな操作により有効期限機構とメモリストレージをサポートし、効率性の利点により比較的簡単かつ柔軟かつ効率的に目的を達成することができます。 これは、一般的な機能を実装するための簡単なコードです:<?php/** * 频率限制 * @param string $action 操作动作 * @param int $userId 发起操作的用户ID * @param int $time 时间范围X秒内 * @param int $number 限制操作数Y次 * @param array $expire 超出封印时间Z ['type'=>1,'ttl'=>过期时间/秒] ['type'=>2,'ttl'=>具体过期时间戳] 二选一 * @return bool * @throws \Exception */public function frequencyLimit(string $action, int $userId, int $time, int $number, $expire = []){ if (empty($action) || $userId <= 0 || $time <= 0 || $number <= 0) { throw new \Exception('非法参数'); } $key = 'act:limit:' . $action . ':' . $userId; $r = RedisClient::connect(); //获取当前累计次数 $current = intval($r->get($key)); if ($current >= $number) return false; //累计并返回最新值 $current = $r->incr($key); //第一次累加,设置控制操作频率的有效时间 if ($current === 1) $r->expire($key, $time); //未超出限制次数先放过 if ($current < $number) return true; //超出后根据需要重新设置过期失效时间 $current === $number 判断保证只重新设置一次 $type = empty($expire['type']) ? 0 : intval($expire['type']); $ttl = empty($expire['ttl']) ? 0 : intval($expire['ttl']); if ($current === $number && $ttl > 0 && in_array($type, [1, 2])) { if ($type === 1) $r->expire($key, $ttl); if ($type === 2) $r->expireAt($key, $ttl); } return false; }//场景1/** * 评论限制 * @param int $userId * @return bool|string */public function doComment(int $userId){ try { $pass = FrequencyLimit::doHandle('comment', $userId, 30, 10); if (!$pass) return '过于频繁'; // todo 评论逻辑 return true; } catch (\Exception $e) { return $e->getMessage(); } }//场景2/** * 点赞限制 * @param int $userId * @return bool|string */public function doLike(int $userId){ try { $pass = FrequencyLimit::doHandle('like', $userId, 10, 10, ['type' => 1, 'ttl' => 1 * 60 * 60]); if (!$pass) return '过于频繁,被禁止操作1小时'; // todo 点赞逻辑 return true; } catch (\Exception $e) { return $e->getMessage(); } }//场景3/** * 上传限制 * @param int $userId * @return bool|string */public function doUpload(int $userId){ try { $expire = strtotime(date('Y-m-d', strtotime(+1 . 'days'))); $pass = FrequencyLimit::doHandle('upload', $userId, 1 * 24 * 60 * 60, 100, ['type' => 2, 'ttl' => $expire]); if (!$pass) return '超出今日上线'; // todo 上传逻辑 return true; } catch (\Exception $e) { return $e->getMessage(); } }//场景N
编码上可以根据你设计这个通用方案的复杂度进行进一步抽象,如抽象成频率限制的功能类等
RedisTutorial## をご覧ください。 # 学べるコラム!
以上がRedis実戦制限動作頻度の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。