Memcache는 PHP 개발에 흔히 사용되는 캐시 방식으로, 동시성이 높은 시스템의 필수 구성 요소입니다.
실제 개발에서 Memcache에는 만족스럽지 못한 문제가 있습니다. 즉, Memcache는 키에 대한 그룹 작업을 지원할 수 없습니다.
그룹 작업은 도메인 작업이라고도 할 수 있습니다. 예를 들어 기사 시스템은 프런트 엔드 부분에서 Memcache를 사용하여 목록 페이지 데이터와 기사 세부 사항 페이지 데이터를 캐시합니다. 그런 다음 기사가 백그라운드에 게시되면 목록 페이지가 최신 목록으로 업데이트되어야 하며, 여기에는 많은 목록 페이지가 포함될 수 있습니다. 물론 기사 상세페이지의 경우에는 업데이트할 필요가 없습니다.
자, 지금은 프로그램이 목록 페이지 데이터를 자동으로 업데이트할 수 있도록 원본 캐시를 삭제해야 합니다. 그러나 Memcache의 플러시 기능을 사용하면 문제가 있습니다. 즉, 목록 페이지와 기사 페이지의 데이터를 포함한 모든 데이터가 지워지는 문제가 있습니다. 대규모 동시성 조건에서는 모든 캐시가 삭제되고 캐시가 다시 빌드될 때 문제가 발생합니다. 매우 높은 부하를 발생시킵니다.
또한 프로그램 구성, 속도 향상을 위해 캐시된 데이터베이스 테이블 구조 등 삭제하고 싶지 않은 일부 캐시 변수가 손실되는 경우가 있을 수 있습니다.
따라서 목록 페이지를 하나의 그룹으로, 기사 페이지 데이터를 다른 그룹으로, 프로그램 구성을 다른 그룹으로 설정할 수 있는 캐싱 메커니즘이 필요합니다. 목록 페이지를 다시 작성해야 하는 경우 다른 그룹의 데이터에 영향을 주지 않고 목록 페이지 그룹의 모든 데이터만 삭제하면 됩니다.
여러 솔루션을 테스트한 결과 다음 솔루션이 가장 이상적이고 빠릅니다. 먼저 코드를 살펴보고 원리에 대해 이야기해 보겠습니다.
<?php class MyCache { private $mmc = null; private $group = null; function __construct($group){ if(!class_exists('mmcache')){ $this->mmc = false; return; } $this->mmc = new memcache(); $this->mmc->addServer('192.168.1.5', 11211); $this->mmc->addServer('192.168.1.6', 11211); $this->group = $group; $this->version = $this->mmc->get('version_'.$group); if(empty($this->version)){ $this->version=1; } } function set($key, $var, $expire=3600){ if(!$this->mmc)return; $this->check_version($this->group,$this->version,$key); return $this->mmc->set($this->group.'_'.$this->version.'_'.$key, $var, $expire); } function get($key){ if(!$this->mmc)return; $this->check_version($this->group,$this->version,$key); return $this->mmc->get($this->group.'_'.$this->version.'_'.$key); } function incr($key, $value=1){ if(!$this->mmc)return; return $this->mmc->increment($this->group.'_'.$this->version.'_'.$key, $value); } function decr($key, $value=1){ if(!$this->mmc)return; return $this->mmc->decrement($this->group.'_'.$this->version.'_'.$key, $value); } function delete($key){ if(!$this->mmc)return; return $this->mmc->delete($this->group.'_'.$this->version.'_'.$key); } function flush(){ if(!$this->mmc)return; ++$this->version; $this->mmc->set('version_'.$this->group, $this->version); } function check_version($goup,$version,$key){ if($version>1){ $version_old=$version-1; return $this->mmc->delete($goup.'_'.$version_old.'_'.$key); } } } ?>
위 클래스는 Memcache 서비스 연결, 값 설정 및 가져오기, 값 증가 및 감소, 키 삭제 및 모두 삭제(플러시)를 포함하여 비교적 완료되었습니다. 여기에는 일반 Memcache 작업 기능과 전체 삭제(플러시) 작업에 대한 확장이 포함됩니다.
코드에서 볼 수 있듯이 지원 그룹의 플러시 기능은 버전 키를 통해 구현됩니다. 즉, 그룹의 변수가 저장될 때마다 버전 값이 키에 추가됩니다. 변수의 버전 값은 숫자(1부터 시작)이며, 버전 값은 키를 저장하고 검색할 때 사용됩니다.
개발자가 현재 그룹의 데이터를 플러시하려는 경우 플러시 작업은 단순히 일부 버전 값(+1)을 변경한 다음 다음에 키에 액세스할 때 원래 값이 아닙니다. ——버전이 변경되었기 때문에, 즉 키 이름이 변경되었습니다. 이러한 방식으로 원래 값은 효율성 오버헤드 없이 Memcache에 의해 자동으로 재활용됩니다. 또한, 프로그램은 버전의 저장 및 검색만 추가할 뿐이며 데이터의 양이 극히 적고 기본적으로 시스템 효율성에 영향을 미치지 않습니다.
위 클래스를 통해 Memcache 캐시에서 그룹 연산을 수행할 수 있으며, 이 PHP 클래스도 Memcache의 인터페이스 기능에 직접 접근할 수 있도록 소켓을 추가하는 등 확장이 가능하므로 별도의 작업이 필요하지 않습니다. 플러시의 오작동을 방지하는 데 더 효과적인 PHP 환경 클래스에 Memcache 확장을 설치하고 apc와 같은 캐싱 메커니즘을 추가한 후에는 Memcache 인터페이스에 대한 소켓 액세스가 확장보다 훨씬 느리지 않습니다.
또한 MyCache 클래스에는 추가 기능이 있습니다. 즉, Memcache 서비스가 실패하면 MyCache 클래스는 직접 오류를 발생시키지 않고 단순히 null 값을 반환합니다.
다음과 같은 Mycache 클래스의 사용:
// 引入定义 include('MyCache.php'); // 实例化 $mc = new MyCache('abc'); // 要有域 // 设置值 $mc->set('word', 'hello world', 900); // 取得值 echo $mc->get('word'); // 删除值 $mc->delete('word'); echo $mc->get('word'); $mc->set('counter', 1, 290000); echo $mc->get('counter'); // 增加值 $mc->incr('counter'); $mc->incr('counter'); echo $mc->get('counter'); // 减少值 $mc->decr('counter'); echo $mc->get('counter'); // 按组删 $mc->flush();
위는 PHP에서 운영하는 MEMCACHE 클래스(도메인 작업)의 코드 인스턴스의 내용입니다. support group, more 관련 내용은 PHP 중국어 홈페이지(www.php.cn)를 참고해주세요!