2

我试图通过编写函数来自动处理数组和命名来简化对缓存函数的调用。这应该适用于数据不经常更新的大多数情况。这是我到目前为止得到的:

function save_cache($data, $name) {
    // get id for name of cache
    $id=shmop_open(get_cache_id($name), "c", 0644, get_array_size($data));

    // return int for data size or boolean false for fail
    if ($id) return shmop_write($id, serialize($data), 0);
    else return false;
}

function get_cache($name) {
    $id=shmop_open(get_cache_id($name), "a", 0644, shmop_size(get_cache_id($name)));
    if ($id) $data=unserialize(shmop_read($id, 0, shmop_size($id)));
    else return false;          // failed to load data

    if ($data) return $data;    // array retrieved
    else return false;          // failed to load data
}

function get_cache_id($name) {
    // build list to return a number for a string id
    // maintain this as new caches are created
    $id=array(  'test'  => 1,
                'test2' => 2
                );

    return $id[$name];
}

function get_array_size($a){
    $size = 0;
    while(list($k, $v) = each($a))$size += is_array($v) ? get_array_size($v) : strlen($v);
    return $size;
}

问题出在 get_cache($name) 函数的第一行。由于这是动态的,我需要根据字符串 $name 查找请求的数组的大小,并在 get_cache_id($name) 中使用我的 id 列表引用它。问题是使用 shmop_open 我需要来自 shmop_size 的大小,但是要使用 shmop_size 我首先需要 shmop_open.... 最后我的 apache 错误日志,第 13 行是 get_cache($name) 中 $id 变量的赋值。

[2013 年 3 月 6 日星期三 15:57:57] [错误] [客户端 127.0.0.1] PHP 警告:shmop_size():在 /home/mark/htdocs/phplib/cache 中没有 ID 为 [1] 的共享内存段。 php 在第 13 行 [2013 年 3 月 6 日星期三 15:57:57] [错误] [客户端 127.0.0.1] PHP 注意:反序列化():/home/mark/htdocs/phplib/cache 中 7769 个字节的偏移量 7765 处出错。第 14 行的 php

编辑:工作代码和实现 - 除了下面的答案,这个脚本的 get_array_size 函数是错误的,应该完全省略。相反,在 save_cache 函数中使用 strlen(serialize($data)) 来确定要存储的缓存大小。此外,您还需要删除该 ID 之前存储的缓存。最终脚本应如下所示:

function save_cache($data, $name, $timeout) {
    // delete cache
    $id=shmop_open(get_cache_id($name), "a", 0, 0);
    shmop_delete($id);
    shmop_close($id);

    // get id for name of cache
    $id=shmop_open(get_cache_id($name), "c", 0644, strlen(serialize($data)));

    // return int for data size or boolean false for fail
    if ($id) {
        set_timeout($name, $timeout);
        return shmop_write($id, serialize($data), 0);
    }
    else return false;
}

function get_cache($name) {
    if (!check_timeout($name)) {
        $id=shmop_open(get_cache_id($name), "a", 0, 0);

        if ($id) $data=unserialize(shmop_read($id, 0, shmop_size($id)));
        else return false;          // failed to load data

        if ($data) {                // array retrieved
            shmop_close();
            return $data;
        }
        else return false;          // failed to load data
    }
    else return false;              // data was expired
}

function get_cache_id($name) {
    $id=array(  'test1' => 1
                'test2' => 2
                );

    return $id[$name];
}

function set_timeout($name, $int) {
    $timeout=new DateTime(date('Y-m-d H:i:s'));
    date_add($timeout, date_interval_create_from_date_string("$int seconds"));
    $timeout=date_format($timeout, 'YmdHis');

    $id=shmop_open(100, "a", 0, 0);
    if ($id) $tl=unserialize(shmop_read($id, 0, shmop_size($id)));
    else $tl=array();
    shmop_delete($id);
    shmop_close($id);

    $tl[$name]=$timeout;
    $id=shmop_open(100, "c", 0644, strlen(serialize($tl)));
    shmop_write($id, serialize($tl), 0);
}

function check_timeout($name) {
    $now=new DateTime(date('Y-m-d H:i:s'));
    $now=date_format($now, 'YmdHis');

    $id=shmop_open(100, "a", 0, 0);
    if ($id) $tl=unserialize(shmop_read($id, 0, shmop_size($id)));
    else return true;
    shmop_close($id);

    $timeout=$tl[$name];
    return (intval($now)>intval($timeout));
}

调用此函数以用于 AJAX 自动完成搜索的示例:

header('Content-type: application/json; charset=utf-8');
include '../phplib/conn.php';
include '../phplib/cache.php';

$searchTerm=$_GET['srch'];
$test=array();
$test=get_cache('test');
if (!$test) {
    $odbc=odbc_connection(); // defined in conn.php
    $sql="SELECT * FROM mysearchtable";
    $result=odbc_exec($odbc, $sql) or
        die("<pre>".date('[Y-m-d][H:i:s]').
        " Error: [".odbc_error()."] ".odbc_errormsg()."\n\n $sql");

    $i=0;
    while ($row=odbc_fetch_array($result)) {
        foreach ($row as $key => $value) $row[$key]=trim($value);
        $test[$i]=$row;
        $i++;
    }

    save_cache($test, 'test1', 600); // 10 minutes timeout
    odbc_close($odbc);
}

$result=array();
foreach ($test as $key) {
    if (strpos($key['item'])===0) { // starts the string
    // if (strpos($key['item'])!==false) { // in the string anywhere
        $result[]=array('item' => $key['item'], 'label' => $key['label']);
    }
}

echo json_encode($result);
4

1 回答 1

2

看函数说明中的注释:shmop_open

shmop_open 3rd and 4th parameters should be 0 when opening an already existing segment. If you are getting a cache, the segement should already be created, therefore you don't need to get the segment size in get_cache.

于 2013-03-06T22:24:24.670 回答