热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

php实现国密算法_使用PHP实现LRU缓存淘汰算法

LRU(cache)LRU介绍缓存是一种提高数据读取性能的技术。但是对于计算机来说,并不可能缓存所有的数据,在达到它的临界空间时,我们需要

LRU(cache)

LRU 介绍

缓存是一种提高数据读取性能的技术。但是对于计算机来说,并不可能缓存所有的数据,在达到它的临界空间时,我们需要通过一些规则用新的数据取代掉一部分的缓存数据。这时候你会如果选择替换呢?

替换的策略有很多种,常用的有以下几种:

● FIFO (先进先出策略)

● LFU (最少使用策略)

● LRU (最近最少使用策略)

● NMRU (在最近没有使用的缓存中随机选择一个替换)

介于我这篇主要实现 LRU,所以就不去介绍其他的了,可以自行去了解。

假设你已经有 5 个女朋友了,此时你成功勾搭上一个新女朋友,在你沉迷女色的同时,你惊奇的发现,你已经不能像年轻时一样以一敌六了,你必须舍弃若干个女朋友,这时候,身拥六个女朋友的渣男你,彻底展示出你的渣男本色,和最近最少秀恩爱的小姐姐说再见:“对不起,国篮此时需要我挺身发边线球,我楠辞琦咎,再见。”,就这样在你成功勾搭一个新小姐姐,你的身体临界点的同时,你就必须舍弃其他的小姐姐。

下面来张实际点的图搞清楚他的原理。

397ffe00d51f42be71781516b66758c1.png

基于上述图片,我们知道,对于 LRU 的操作,无非在于插入 (insert), 删除 (delete),以及替换,针对替换来说,如果缓存空间满了,那么就是 insert to head and delete for tail。如果未满,也分为两种,一种是缓存命中的话,只需要把缓存的值 move to head。如果之前不存在,那么就是 insert to head。

实现过程

接下来就是数据结构的选择了。数组的存储是连续的内存空间,虽然查询的时间复杂度是 O (1), 但是删除和插入为了保存内存空间的连续性,需要进行搬移,那么时间复杂度就是 O (n), 为了实现能快速删除,故而采用双向链表。但是链表的查询时间复杂度是 O (n), 那么就需要 hash table。屁话说了这么多,代码实现。其实之前刷过这道题目。特地拿出来讲一下。

class LRUCache {

private $capacity;

private $list;

/**

* @param Integer $capacity

*/

function __construct($capacity) {

$this->capacity=$capacity;

$this->list=new HashList();

}

/**

* @param Integer $key

* @return Integer

*/

function get($key) {

if($key<0) return -1;

return $this->list->get($key);

}

/**

* &#64;param Integer $key

* &#64;param Integer $value

* &#64;return NULL

*/

function put($key, $value) {

$size&#61;$this->list->size;

$isHas&#61;$this->list->checkIndex($key);

if($isHas || $size&#43;1 > $this->capacity){

$this->list->removeNode($key);

}

$this->list->addAsHead($key,$value);

}

}

class HashList{

public $head;

public $tail;

public $size;

public $buckets&#61;[];

public function __construct(Node $head&#61;null,Node $tail&#61;null){

$this->head&#61;$head;

$this->tail&#61;$tail;

$this->size&#61;0;

}

//检查键是否存在

public function checkIndex($key){

$res&#61;$this->buckets[$key];

if($res){

return true;

}

return false;

}

public function get($key){

$res&#61;$this->buckets[$key];

if(!$res) return -1;

$this->moveToHead($res);

return $res->val;

}

//新加入的节点

public function addAsHead($key,$val)

{

$node&#61;new Node($val);

if($this->tail&#61;&#61;null && $this->head !&#61;null){

$this->tail&#61;$this->head;

$this->tail->next&#61;null;

$this->tail->pre&#61;$node;

}

$node->pre&#61;null;

$node->next&#61;$this->head;

$this->head->pre&#61;$node;

$this->head&#61;$node;

$node->key&#61;$key;

$this->buckets[$key]&#61;$node;

$this->size&#43;&#43;;

}

//移除指针(已存在的键值对或者删除最近最少使用原则)

public function removeNode($key)

{

$current&#61;$this->head;

for($i&#61;1;$isize;$i&#43;&#43;){

if($current->key&#61;&#61;$key) break;

$current&#61;$current->next;

}

unset($this->buckets[$current->key]);

//调整指针

if($current->pre&#61;&#61;null){

$current->next->pre&#61;null;

$this->head&#61;$current->next;

}else if($current->next &#61;&#61;null){

$current->pre->next&#61;null;

$current&#61;$current->pre;

$this->tail&#61;$current;

}else{

$current->pre->next&#61;$current->next;

$current->next->pre&#61;$current->pre;

$current&#61;null;

}

$this->size--;

}

//把对应的节点应到链表头部(最近get或者刚刚put进去的node节点)

public function moveToHead(Node $node)

{

if($node&#61;&#61;$this->head) return ;

//调整前后指针指向

$node->pre->next&#61;$node->next;

$node->next->pre&#61;$node->pre;

$node->next&#61;$this->head;

$this->head->pre&#61;$node;

$this->head&#61;$node;

$node->pre&#61;null;

}

}

class Node{

public $key;

public $val;

public $next;

public $pre;

public function __construct($val)

{

$this->val&#61;$val;

}

}

/**

* Your LRUCache object will be instantiated and called as such:

* $obj &#61; LRUCache($capacity);

* $ret_1 &#61; $obj->get($key);

* $obj->put($key, $value);

Github 整理地址:https://github.com/wuqinqiang/leetcode-php

更多PHP知识&#xff0c;请访问&#xff01;



推荐阅读
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 深入理解Redis的数据结构与对象系统
    本文详细探讨了Redis中的数据结构和对象系统的实现,包括字符串、列表、集合、哈希表和有序集合等五种核心对象类型,以及它们所使用的底层数据结构。通过分析源码和相关文献,帮助读者更好地理解Redis的设计原理。 ... [详细]
  • 本文详细介绍了在企业级项目中如何优化 Webpack 配置,特别是在 React 移动端项目中的最佳实践。涵盖资源压缩、代码分割、构建范围缩小、缓存机制以及性能优化等多个方面。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 从零开始构建完整手机站:Vue CLI 3 实战指南(第一部分)
    本系列教程将引导您使用 Vue CLI 3 构建一个功能齐全的移动应用。我们将深入探讨项目中涉及的每一个知识点,并确保这些内容与实际工作中的需求紧密结合。 ... [详细]
  • Scala 实现 UTF-8 编码属性文件读取与克隆
    本文介绍如何使用 Scala 以 UTF-8 编码方式读取属性文件,并实现属性文件的克隆功能。通过这种方式,可以确保配置文件在多线程环境下的一致性和高效性。 ... [详细]
  • MySQL索引详解与优化
    本文深入探讨了MySQL中的索引机制,包括索引的基本概念、优势与劣势、分类及其实现原理,并详细介绍了索引的使用场景和优化技巧。通过具体示例,帮助读者更好地理解和应用索引以提升数据库性能。 ... [详细]
  • 在 Flutter 开发过程中,开发者经常会遇到 Widget 构造函数中的可选参数 Key。对于初学者来说,理解 Key 的作用和使用场景可能是一个挑战。本文将详细探讨 Key 的概念及其应用场景,并通过实例帮助你更好地掌握这一重要工具。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 网络运维工程师负责确保企业IT基础设施的稳定运行,保障业务连续性和数据安全。他们需要具备多种技能,包括搭建和维护网络环境、监控系统性能、处理突发事件等。本文将探讨网络运维工程师的职业前景及其平均薪酬水平。 ... [详细]
  • PHP 5.5.0rc1 发布:深入解析 Zend OPcache
    2013年5月9日,PHP官方发布了PHP 5.5.0rc1和PHP 5.4.15正式版,这两个版本均支持64位环境。本文将详细介绍Zend OPcache的功能及其在Windows环境下的配置与测试。 ... [详细]
  • 深入解析 Apache Shiro 安全框架架构
    本文详细介绍了 Apache Shiro,一个强大且灵活的开源安全框架。Shiro 专注于简化身份验证、授权、会话管理和加密等复杂的安全操作,使开发者能够更轻松地保护应用程序。其核心目标是提供易于使用和理解的API,同时确保高度的安全性和灵活性。 ... [详细]
  • 本文介绍如何在PostgreSQL数据库中正确插入和处理JSON数据类型,确保数据完整性和避免常见错误。 ... [详细]
  • 不确定性|放入_华为机试题 HJ9提取不重复的整数
    不确定性|放入_华为机试题 HJ9提取不重复的整数 ... [详细]
author-avatar
书友36296361
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有