246 lines
6.3 KiB
Perl
246 lines
6.3 KiB
Perl
|
# ==================================================================
|
||
|
# Gossamer Threads Module Library - http://gossamer-threads.com/
|
||
|
#
|
||
|
# GT::Cache
|
||
|
# Author : Scott Beck
|
||
|
# CVS Info : 087,071,086,086,085
|
||
|
# $Id: Cache.pm,v 1.13 2004/01/13 01:35:15 jagerman Exp $
|
||
|
#
|
||
|
# Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
|
||
|
# ==================================================================
|
||
|
#
|
||
|
# Description:
|
||
|
# Implements a tied hash cache that will not grow forever, but expire
|
||
|
# old/unused entries. Useful under mod_perl.
|
||
|
#
|
||
|
|
||
|
package GT::Cache;
|
||
|
# ===============================================================
|
||
|
use vars qw /$DEBUG $VERSION $CACHE_SIZE/;
|
||
|
use strict;
|
||
|
|
||
|
$DEBUG = 0;
|
||
|
$VERSION = sprintf "%d.%03d", q$Revision: 1.13 $ =~ /(\d+)\.(\d+)/;
|
||
|
$CACHE_SIZE = 500;
|
||
|
|
||
|
##
|
||
|
# tie %cache, 'GT::Cache', $size, \&function;
|
||
|
# ----------------------------
|
||
|
# Is called when you tie a hash to this
|
||
|
# class. The size should be the size limit
|
||
|
# you want on your hash. If not specified
|
||
|
# this will default to the CLASS variable
|
||
|
# $CACH_SIZE which is initialized to 500
|
||
|
##
|
||
|
sub TIEHASH {
|
||
|
my $this = shift;
|
||
|
my $size = shift || $CACHE_SIZE;
|
||
|
my $code = shift || sub {undef};
|
||
|
my $class = ref $this || $this;
|
||
|
my $self = bless {
|
||
|
cache_size => $size,
|
||
|
popularity => [],
|
||
|
content => {},
|
||
|
indices => {},
|
||
|
is_indexed => 0,
|
||
|
size => 0,
|
||
|
code => $code,
|
||
|
}, $class;
|
||
|
$#{$self->{popularity}} = $size;
|
||
|
return $self;
|
||
|
}
|
||
|
|
||
|
sub FETCH {
|
||
|
my ($self, $key) = @_;
|
||
|
if (ref $key) {
|
||
|
require GT::Dumper;
|
||
|
my $dmp = new GT::Dumper (
|
||
|
{
|
||
|
data => $key,
|
||
|
sort => 1
|
||
|
}
|
||
|
);
|
||
|
my $new = $dmp->dump;
|
||
|
$key = $new;
|
||
|
}
|
||
|
unless (exists $self->{content}->{$key}) {
|
||
|
my $val = $self->{code}->($key);
|
||
|
defined $val or return undef;
|
||
|
$self->STORE ($key, $val);
|
||
|
return $val;
|
||
|
}
|
||
|
if ($self->{is_indexed}) {
|
||
|
my ($pos1, $pos2, $replace);
|
||
|
|
||
|
$pos1 = $self->{content}->{$key}->[1];
|
||
|
$pos2 = $pos1 + (int (rand( ($self->{cache_size} - $pos1) / 2) )) || 1;
|
||
|
|
||
|
$replace = ${$self->{popularity}}[$pos2];
|
||
|
|
||
|
${$self->{popularity}}[$pos2] = $key;
|
||
|
$self->{content}->{$key}->[1] = $pos2;
|
||
|
if (defined $replace) {
|
||
|
${$self->{popularity}}[$pos1] = $replace;
|
||
|
$self->{content}->{$replace}->[1] = $pos1;
|
||
|
}
|
||
|
}
|
||
|
return $self->{content}->{$key}->[0];
|
||
|
}
|
||
|
|
||
|
##
|
||
|
# %cash = (key1 => $field1, key2 => $val2);
|
||
|
# -----------------------------------------
|
||
|
# $cash{key} = $val;
|
||
|
# ------------------
|
||
|
# Called when you store something in the hash.
|
||
|
# This will check the number of elements in the
|
||
|
# hash and delete the oldest one if the limit.
|
||
|
# is reached.
|
||
|
##
|
||
|
sub STORE {
|
||
|
my ($self, $key, $value) = @_;
|
||
|
if (ref $key) {
|
||
|
require GT::Dumper;
|
||
|
my $dmp = new GT::Dumper (
|
||
|
{
|
||
|
data => $key,
|
||
|
sort => 1
|
||
|
}
|
||
|
);
|
||
|
my $new = $dmp->dump;
|
||
|
$key = $new;
|
||
|
}
|
||
|
my ($replace, $insid);
|
||
|
if ($self->{is_indexed}) {
|
||
|
$insid = int (rand($self->{cache_size} / 2)) || 1;
|
||
|
if (defined ($replace = ${$self->{popularity}}[$insid])) {
|
||
|
delete $self->{content}->{$replace};
|
||
|
undef ${$self->{popularity}}[$insid];
|
||
|
}
|
||
|
${$self->{popularity}}[$insid] = $key;
|
||
|
$self->{content}->{$key} = [$value, $insid];
|
||
|
}
|
||
|
else {
|
||
|
${$self->{popularity}}[$self->{size}] = $key;
|
||
|
$self->{content}->{$key} = [$value, $self->{size}];
|
||
|
if ($self->{size} == $self->{cache_size}) {
|
||
|
for (0 .. $#{$self->{popularity}}) {
|
||
|
next unless defined $self->{popularity}[$_];
|
||
|
$self->{content}{$self->{popularity}[$_]}[1] = $_;
|
||
|
}
|
||
|
$self->{is_indexed} = 1;
|
||
|
}
|
||
|
$self->{size}++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sub DELETE {
|
||
|
my ($self, $key) = @_;
|
||
|
if (ref $key) {
|
||
|
require GT::Dumper;
|
||
|
my $dmp = new GT::Dumper (
|
||
|
{
|
||
|
data => $key,
|
||
|
sort => 1
|
||
|
}
|
||
|
);
|
||
|
my $new = $dmp->dump;
|
||
|
$key = $new;
|
||
|
}
|
||
|
exists $self->{content}->{$key} or return undef;
|
||
|
$self->{size}--;
|
||
|
my $aref = delete $self->{content}->{$key};
|
||
|
undef $self->{popularity}->[$aref->[1]];
|
||
|
return $aref->[0];
|
||
|
}
|
||
|
|
||
|
sub CLEAR {
|
||
|
my $self = shift;
|
||
|
$self->{content} = {};
|
||
|
$self->{size} = 0;
|
||
|
$self->{popularity} = [];
|
||
|
$self->{is_indexed} = 0;
|
||
|
}
|
||
|
|
||
|
sub EXISTS {
|
||
|
my ($self, $key) = @_;
|
||
|
if (ref $key) {
|
||
|
require GT::Dumper;
|
||
|
my $dmp = new GT::Dumper (
|
||
|
{
|
||
|
data => $key,
|
||
|
sort => 1
|
||
|
}
|
||
|
);
|
||
|
my $new = $dmp->dump;
|
||
|
$key = $new;
|
||
|
}
|
||
|
return exists $self->{content}->{$key} ? 1 : 0;
|
||
|
}
|
||
|
|
||
|
sub FIRSTKEY {
|
||
|
my $self = shift;
|
||
|
my $c = keys %{$self->{content}};
|
||
|
return scalar each %{$self->{content}};
|
||
|
}
|
||
|
|
||
|
sub NEXTKEY {return scalar each %{shift()->{content}}}
|
||
|
|
||
|
1;
|
||
|
|
||
|
__END__
|
||
|
|
||
|
=head1 NAME
|
||
|
|
||
|
GT::Cache - Tied hash which caches output of functions.
|
||
|
|
||
|
=head1 SYNOPSIS
|
||
|
|
||
|
use GT::Cache;
|
||
|
my %cache;
|
||
|
tie %cache, 'GT::Cache', $size, \&function;
|
||
|
|
||
|
=head1 DESCRIPTION
|
||
|
|
||
|
GT::Cache implements a simple but quick caching scheme for remembering
|
||
|
the results of functions. It also implements a max size to prevent
|
||
|
the cache from growing and drops least frequently requested entries
|
||
|
first, making it very useful under mod_perl.
|
||
|
|
||
|
=head1 EXAMPLE
|
||
|
|
||
|
use GT::Cache;
|
||
|
my %cache;
|
||
|
tie %cache, 'GT::Cache', 100, \&complex_func;
|
||
|
|
||
|
while (<>) {
|
||
|
print "RESULT: ", $cache{$_}, "\n";
|
||
|
}
|
||
|
|
||
|
sub complex_func {
|
||
|
my $input = shift;
|
||
|
# .. do complex work.
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
This will cache the results of complex_func, and only run it when
|
||
|
the input is different. It stores a max of 100 entries at a time,
|
||
|
with the least frequently requested getting dropped first.
|
||
|
|
||
|
=head1 NOTES
|
||
|
|
||
|
Currently, you can only pass as input to the function a single
|
||
|
scalar, and the output must be a single scalar. See the
|
||
|
Memoize module in CPAN for a much more robust implementation.
|
||
|
|
||
|
=head1 COPYRIGHT
|
||
|
|
||
|
Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
|
||
|
http://www.gossamer-threads.com/
|
||
|
|
||
|
=head1 VERSION
|
||
|
|
||
|
Revision: $Id: Cache.pm,v 1.13 2004/01/13 01:35:15 jagerman Exp $
|
||
|
|
||
|
=cut
|