`
cjf068
  • 浏览: 34230 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

LRUCache

 
阅读更多
MyLRUCache 缓存类
package org.jf.alg.lru;

import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;


/**
 * 限定容量二叉堆+HashMap实现LRU
 * 
 * @author junfeng.chen
 *
 */
public class MyLRUCache {
		
	private Map<Object,ValueEntry> container = Collections.synchronizedMap( new HashMap<Object,ValueEntry>());
	
	private List<Object> visistList = Collections.synchronizedList(new LinkedList<Object>());//key值的访问历史
	
	private KeyVisitCounter[] heap_array ;
	
	private int increaseStep = 500 ;
	
	private int initSize = 2000;
	
//访问记录列表  每次访问元素,将其访问记录放入列表,并有后台线程处理
	private long reset_interval = 2000*60;//刷新时间默认为2分钟
	
	private int capacity = 100000;
	
	private int size = 0;
	
	private boolean bln_started = false;
	
	private PerlocateThread perlocateThd ;
	
	public MyLRUCache()
	{
		init();
	}
	
	public MyLRUCache(int capacity)
	{
		this.capacity = capacity;
		init();
	}
	
	/**
	 * 
	 * @param capacity 最大容量
	 * @param time_cycle 计数时间周期
	 */
	public MyLRUCache(int capacity,int time_cycle)
	{
		this.capacity = capacity;
		this.reset_interval = time_cycle;
		init();
	}
	
	public void start()
	{
		this.perlocateThd = new PerlocateThread();
		perlocateThd.start();
		this.bln_started = true;
	}
	
	public void stop()
	{
		if(perlocateThd != null)
			perlocateThd.exit();
		this.bln_started = false;
		this.heap_array = new KeyVisitCounter[this.initSize+1];
		this.visistList.clear();
		this.container.clear();
	}
	private void init()
	{
		this.heap_array = new KeyVisitCounter[this.initSize+1];
	}
	
	public void resetCount()
	{
		for(KeyVisitCounter counter:heap_array)
		{
			if(counter != null)
				counter.reSet();
			else
				 break;
		}
	}
	
	
//	public void remove(Object key)
//	{
//		not support remove explicity
//	}
//	
	public Object get(Object key)
	{
		assertStatus();
		ValueEntry valueEntry = this.container.get(key);
		if(valueEntry != null)
		{
			//在堆中查找当前key,并对其count加1
			visistList.add(key);
			return valueEntry.getData();
		}
		return null;
	}
	
	public void put(Object key,Object value)
	{
		assertStatus();
		ValueEntry entry = new ValueEntry(value);
		this.visistList.add(key);
		this.container.put(key, entry);//实际上 保存的元素可能比capacity多
	}
	
	public int getSize()
	{
		return this.size;
	}
	
	
	/***********************some help methods****************************************************/
	
	private void assertStatus()
	{
		if(!this.bln_started)
			throw new RuntimeException("Illegle Cache Staus,not started");
	}

	/**
	 * 
	 * @param key
	 * @return 该key值对应的索引
	 */
	private int  perlocateUp(Object key/* , ValueEntry entry*/)
	{
		int indx = -1;
		ValueEntry entry = container.get(key);
		if(entry.getKeyIndx()==-1)
		{
			if(size<this.capacity&&size==this.heap_array.length-1)
			{
				int step = this.capacity - this.size;
				step = step<=this.increaseStep?step:increaseStep;
				KeyVisitCounter []newArray = new KeyVisitCounter[heap_array.length+step];
				System.arraycopy(heap_array, 1, newArray,1, size);
				heap_array = newArray;
			}
			
			KeyVisitCounter key_counter = new KeyVisitCounter(key);
			key_counter.visit();//缓存已满时,防止最后一个元素总是被删除 刚插入的元素给予特殊待遇
			//新加入一个记录值
			if(this.size==this.capacity)//heap满
			{
				if(heap_array[size].compareTo(key_counter)<=0)
				{
					heap_array[size] = key_counter;
					indx = size;
				}//else indx = -1
				
			}else
			{
				heap_array[size+1] = key_counter;
				indx = size+1;
				size++;
			}
			entry.setKeyIndex(indx);
		}
		else
		{
			indx = entry.getKeyIndx();
		}
		
		if(indx == -1)//
			return -1;
		
		//fix up heap_array to ensure it's heap character
			int parent = indx/2;
			KeyVisitCounter counter = this.heap_array[indx];
			counter.visit();
			while(parent!=0)
			{
				if(this.heap_array[indx].compareTo(this.heap_array[parent])>0)
				{				
					KeyVisitCounter tmp = heap_array[indx];
					heap_array[indx] = heap_array[parent];
					heap_array[parent] = tmp;
					indx = parent;
					
					
					//设置key对应的索引
					this.container.get(heap_array[indx].getKey()).setKeyIndex(indx);
					this.container.get(heap_array[parent].getKey()).setKeyIndex(parent);
					
					parent = indx/2;
				}else
				{
					break;
				}
			}
					
	  return indx;
	}
	
	

	private class PerlocateThread extends Thread
	{
		boolean run = true;
		public void run()
		{
			int count = 200;
			long last_reset_time = System.currentTimeMillis();
			while(run)
			{
				while(count>0&&visistList.size()>0)
				{
					Object key = visistList.remove(0);
					int indx = perlocateUp(key);
					if(indx == -1)
						container.remove(key);
				}
				
				if(System.currentTimeMillis() - last_reset_time >reset_interval)
				{
					resetCount();
					last_reset_time = System.currentTimeMillis();
				}
				try {
					this.sleep(500);
				} catch (InterruptedException e) {
					e.printStackTrace();
					break;
				}	
			}
		}
		
		public void exit()
		{
			run = false;
		}
		
	}
}


访问次数记录类
package org.jf.alg.lru;

public class KeyVisitCounter implements Comparable{

	private Object key;
	private int count;
	
	public KeyVisitCounter(Object key)
	{
		this.key = key;
	}
	
	public void visit()
	{
		count++;
	}
	
	public void visit(int times)
	{
		this.count += times ;
	}
	
	public int getVisitTimes()
	{
		return count;
	}
	
	public Object getKey()
	{
		return this.key;
	}

	public void reSet()
	{
		this.count = 0;
	}
	
	@Override
	public int compareTo(Object o) 
	{

		if(!(o instanceof KeyVisitCounter))
			return 1;
		KeyVisitCounter counter = (KeyVisitCounter) o;
		if(this.count > counter.count)
			return 1;
		if(this.count == counter.count)
			return 0;
		return -1;

	}
}



package org.jf.alg.lru;

public class ValueEntry {

	private int keyIndx = -1;//key在heap数组中的下标
	private Object data;
	
	public ValueEntry(Object data)
	{
		this.data = data;
	}
	
	public Object getData()
	{
		return this.data;
	}
	
	public void setKeyIndex(int indx)
	{
		this.keyIndx = indx;
	}
	
	public int getKeyIndx()
	{
		return this.keyIndx;
	}
}

分享到:
评论

相关推荐

    PHP语言基础知识详解及常见功能应用.docx

    本文详细介绍了PHP的基本语法、变量类型、运算符号以及文件上传和发邮件功能的实现方法,适合初学者了解和掌握PHP的基础知识。

    公司金融课程期末考试题目

    公司金融整理的word文档

    适用于 Python 应用程序的 Prometheus 检测库.zip

    Prometheus Python客户端Prometheus的官方 Python 客户端。安装pip install prometheus-client这个包可以在PyPI上找到。文档文档可在https://prometheus.github.io/client_python上找到。链接发布发布页面显示项目的历史记录并充当变更日志。吡啶甲酸

    DFC力控系统维护及使用

    DFC力控系统维护及使用

    Spring Data的书籍项目,含多数据库相关内容.zip

    1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。

    2019-2023GESP,CSP,NOIP真题.zip

    2019-2023GESP,CSP,NOIP真题.zip

    基于 Gin + Element 实现的春联生成平台

    博文链接 https://blog.csdn.net/weixin_47560078/article/details/127712877?spm=1001.2014.3001.5502

    zetero7实测可用插件

    包含: 1、jasminum茉莉花 2、zotero-style 3、greenfrog 4、zotero-reference 5、translate-for-zotero 用法参考:https://zhuanlan.zhihu.com/p/674602898

    简单的 WSN 动画制作器 matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。 替换数据可以直接使用,注释清楚,适合新手

    毕业设计&课设_仿知乎社区问答类 App 项目:吉林大学毕业设计,含代码、截图及相关说明.zip

    1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。

    python技巧学习.zip

    python技巧学习.zip

    2023 年“泰迪杯”数据分析技能赛 A 题 档案数字化加工流程数据分析

    2023 年“泰迪杯”数据分析技能赛 A 题 档案数字化加工流程数据分析 完整代码

    life-expectancy-table.json

    echarts 折线图数据源文件

    此扩展现在由 Microsoft fork 维护 .zip

    Visual Studio Code 的 Python 扩展Visual Studio Code 扩展对Python 语言提供了丰富的支持(针对所有积极支持的 Python 版本),为扩展提供了访问点,以无缝集成并提供对 IntelliSense(Pylance)、调试(Python 调试器)、格式化、linting、代码导航、重构、变量资源管理器、测试资源管理器等的支持!支持vscode.devPython 扩展在vscode.dev (包括github.dev )上运行时确实提供了一些支持。这包括编辑器中打开文件的部分 IntelliSense。已安装的扩展Python 扩展将默认自动安装以下扩展,以在 VS Code 中提供最佳的 Python 开发体验Pylance - 提供高性能 Python 语言支持Python 调试器- 使用 debugpy 提供无缝调试体验这些扩展是可选依赖项,这意味着如果无法安装,Python 扩展仍将保持完全功能。可以禁用或卸载这些扩展中的任何一个或全部,但会牺牲一些功能。通过市场安装的扩展受市场使用条款的约束。可

    Centos6.x通过RPM包升级OpenSSH9.7最新版 升级有风险,前务必做好快照,以免升级后出现异常影响业务

    Centos6.x通过RPM包升级OpenSSH9.7最新版 升级有风险,前务必做好快照,以免升级后出现异常影响业务

    5 总体设计.pptx

    5 总体设计.pptx

    用于执行 RPA 的 Python 包.zip

    Python 版 RPAv1.50  • 使用案例•  API  参考 • 关于 和制作人员 • 试用云 •  PyCon 视频 •  Telegram 聊天 • 中文 •  हिन्दी  • 西班牙语 • 法语 •  বাংলা  •  Русский  • 葡萄牙语 • 印尼语 • 德语 • 更多..要为 RPA(机器人流程自动化)安装此 Python 包 -pip install rpa要在 Jupyter 笔记本、Python 脚本或交互式 shell 中使用它 -import rpa as r有关操作系统和可选可视化自动化模式的说明 -️‍ Windows -如果视觉自动化有故障,请尝试将显示缩放级别设置为推荐的 % 或 100% macOS -由于安全性更加严格,请手动安装 PHP并查看PhantomJS和Java 弹出窗口的解决方案 Linux -视觉自动化模式需要在 Linux 上进行特殊设置,请参阅如何安装 OpenCV 和 Tesseract Raspberry Pi - 使用此设置指南在 Raspberry Pies(低成本自

    原生js识别手机端或电脑端访问代码.zip

    原生js识别手机端或电脑端访问代码.zip

    极速浏览器(超快速运行)

    浏览器

    基于SpringBoot和Vue的旅游可视化系统设计与实现

    内容概要:本文介绍了基于Spring Boot和Vue开发的旅游可视化系统的设计与实现。该系统集成了用户管理、景点信息、路线规划、酒店预订等功能,通过智能算法根据用户偏好推荐景点和路线,提供旅游攻略和管理员后台,支持B/S架构,使用Java语言和MySQL数据库,提高了系统的扩展性和维护性。 适合人群:具有一定编程基础的技术人员,特别是熟悉Spring Boot和Vue框架的研发人员。 使用场景及目标:适用于旅游行业,为企业提供一个高效的旅游推荐平台,帮助用户快速找到合适的旅游信息和推荐路线,提升用户旅游体验。系统的智能化设计能够满足用户多样化的需求,提高旅游企业的客户满意度和市场竞争力。 其他说明:系统采用现代化的前后端分离架构,具备良好的可扩展性和维护性,适合在旅游行业中推广应用。开发过程中需要注意系统的安全性、稳定性和用户体验。

Global site tag (gtag.js) - Google Analytics