Heim >Backend-Entwicklung >PHP-Tutorial >Konsistentes Hashing – PHP

Konsistentes Hashing – PHP

WBOY
WBOYOriginal
2016-07-29 09:11:20855Durchsuche

/**
 * Flexihash – Eine einfache konsistente Hashing-Implementierung für PHP.
 * 
 * Die MIT-Lizenz
 * 
 * Copyright (c) 2008 Paul Annesley
 * 
 * Erlaubnis ist Hiermit wird jeder Person, die eine Kopie
 * dieser Software und der zugehörigen Dokumentationsdateien (die „Software“) erhält, kostenlos gestattet, mit der Software ohne Einschränkung zu handeln
 *, einschließlich und ohne Einschränkung der Rechte
 * Kopien der Software zu verwenden, zu kopieren, zu ändern, zusammenzuführen, zu veröffentlichen, zu verteilen, unterzulizenzieren und/oder zu verkaufen
 * und Personen, denen die Software zur Verfügung gestellt wird, dies zu gestatten, vorbehaltlich die folgenden Bedingungen:
 * 
 * Der obige Urheberrechtshinweis und dieser Genehmigungshinweis müssen in
 * allen Kopien oder wesentlichen Teilen der Software enthalten sein.
 * 
 * DIE SOFTWARE WIRD ZUR VERFÜGUNG GESTELLT " WIE BESEHEN“, OHNE JEGLICHE AUSDRÜCKLICHE ODER STILLSCHWEIGENDE GEWÄHRLEISTUNG, EINSCHLIESSLICH, ABER NICHT BESCHRÄNKT AUF DIE GEWÄHRLEISTUNG DER MARKTGÄNGIGKEIT,
 * EIGNUNG FÜR EINEN BESTIMMTEN ZWECK UND NICHTVERLETZUNG. IN KEINEM FALL SIND DIE
 * AUTOREN ODER COPYRIGHT-INHABER FÜR JEGLICHE ANSPRÜCHE, SCHÄDEN ODER ANDERE
 * HAFTUNG, SEI ES AUS EINER VERTRAGLICHEN HANDLUNG, unerlaubter Handlung ODER ANDERWEITIG, ENTSTEHEND AUS
 * AUS ODER IN VERBINDUNG MIT DER SOFTWARE ODER DER NUTZUNG ODER ANDEREN HANDELN IN
 * DER SOFTWARE.
 * 
 * @author Paul Annesley
 * @link http://paul.annesley.cc/
 * @copyright Paul Annesley, 2008
 * @comment by MyZ (http://blog.csdn.net/mayongzhan)
 */
/**
 * Eine einfache konsistente Hashing-Implementierung mit steckbaren Hash-Algorithmen.
 *
 * @author Paul Annesley
 * @package Flexihash
 * @licence http://www.opensource.org/licenses /mit-license.php
 */
class Flexihash
{
/**
* Die Anzahl der Positionen, auf die jedes Ziel gehasht werden soll.
*
* @var int
* @comment Die Anzahl der virtuellen Knoten, um das Problem der ungleichmäßigen Knotenverteilung zu lösen
*/
private $_replicas = 64;
/**
* Der Hash-Algorithmus, gekapselt in einer Flexihash_Hasher-Implementierung.
* @var object Flexihash_Hasher
* @comment 使用的hash方法 : md5,crc32
*/
private $_hasher;
/**
* Interner Zähler für die aktuelle Anzahl an Zielen.
* @var int
* @comment Knotenzähler
*/
private $_targetCount = 0;
/ **
* Interne Zuordnung von Positionen (Hash-Ausgaben) zu Zielen
* @var array { position => target, ... }
* @comment Position entsprechender Knoten, wird verwendet, um Ziele basierend auf der Position in der Suche zu bestimmen Besuchter Knoten
*/
private $_positionToTarget = array();
/**
* Interne Zuordnung von Zielen zu Listen von Positionen, zu denen das Ziel gehasht wird.
* @var array { target => [ position, position, ... ], ... }
* @comment The entsprechende Position des Knotens, wird zum Löschen des Knotens
verwendet*/
private $_targetToPositions = array();
/**
* Ob die interne Karte von Positionen zu Zielen bereits sortiert ist.
* @var boolean
* @comment Ob die interne Karte von Positionen zu Zielen bereits sortiert ist.
* /
private $_positionToTargetSorted = false;
/**
* Konstruktor
* @param object $hasher Flexihash_Hasher
* @param int $replicas Anzahl der Positionen, an die jedes Ziel gehasht werden soll.
* @comment Konstruktor, bestimmen Sie die zu verwendende Hash-Methode und die Anforderungen Anzahl virtueller Knoten Je mehr virtuelle Knoten vorhanden sind, desto gleichmäßiger ist die Verteilung, aber desto langsamer wird der verteilte Betrieb des Programms sein
*/
public function __construct(Flexihash_Hasher $hasher = null, $replicas = null)
{
$this-> ;_hasher = $hasher ? $hasher : new Flexihash_Crc32Hasher();
if (!empty($replicas)) $this->_replicas = $replicas;
}
/**
* Ein Ziel hinzufügen.
* @param string $target
* @chainable
* @comment Einen Knoten hinzufügen und die Knoten entsprechend der Anzahl der virtuellen Knoten auf mehrere virtuelle Standorte verteilen
*/
öffentliche Funktion addTarget($target)
{
if (isset($this->_targetToPositions[$target]))
{
throw new Flexihash_
Exception
("Target '$ target' existiert bereits.");}$this->_targetToPositions[$target] = array();
// hash das Ziel in mehrere Positionen
für ($i = 0 ; $i < $this->_replicas; $i++)
{
$position = $this->_hasher->hash($target . $i);
$this-> ;_positionToTarget[$position] = $target; // lookup
$this->_targetToPositions[$target] []= $position; // Zielentfernung
}
$this->_positionToTargetSorted = false;
$this->_targetCount++;
return $this;
}
/**
* Fügen Sie eine Liste von Zielen hinzu.
* @param array $targets
* @chainable
*/
öffentliche Funktion addTargets($targets)
{
foreach ($targets as $target)
{
$this->addTarget($target);
}
$this zurückgeben;
}
/**
* 대상을 제거합니다.
* @param string $target
* @chainable
*/
공용 함수 RemoveTarget($target)
{
if (!isset($this->_targetToPositions[$target]))
{
새 Flexihash 발생_예외("'$target' 대상이 존재하지 않습니다.");
}
foreach ($this->_targetToPositions[$target] as $position)
{
unset($this->_positionToTarget[$position]);
}
unset($this->_targetToPositions[$target]);
$this->_targetCount --;
return $this;
}
/**
* 모든 잠재적 대상 목록
* @return array
*/
공용 함수 getAllTargets()
{
return array_keys($this->_targetToPositions) ;
}
/**
* 주어진 리소스에 대한 대상을 조회합니다.
* @param string $resource
* @return string
*/
공개 함수 조회($resource)
{
$targets = $this->lookupList($resource, 1);
if (empty($targets)) throw new Flexihash_예외('대상이 존재하지 않음');
return $targets[0];
}
/**
* 우선순위에 따라 리소스에 대한 대상 목록을 가져옵니다.
* 최대 $requestedCount 대상이 반환되며 총 개수가 적으면 그보다 적습니다.
*
* @param string $ resources
* @param int $requestedCount 반환할 목록의 길이
* @return array 대상 목록
* @comment 현재 리소스에 해당하는 노드를 찾습니다.
* 노드가 있으면 반환합니다. 비어 있으면 노드가 하나만 있으면
*                                                                                            >                               리소스의 위치를 ​​첫 번째 순서 위치로 결정(링 형성)
* 찾은 노드 반환
*/
공용 함수 lookupList($resource, $requestedCount)
{
if (!$requestedCount)
throw new Flexihash_
Exception('잘못된 개수 요청됨'); // 대상 없음
if (empty($this->_positionToTarget))
return array();
// 단일 대상 최적화
if ($this->_targetCount == 1)
return array_unique(array_values($this->_positionToTarget));
// 리소스를 위치로 해시
$resourcePosition = $this->_hasher->hash($resource );
$results = array();
$collect = false;
$this->_sortPositionTargets();
//resourcePosition
foreach($this- >_positionToTarget as $key => $value)
{
// 리소스 위치 전달 후 대상 수집 시작
if (!$collect && $key > $resourcePosition)
{
$collect = true;
}
// 모든 대상의 첫 번째 인스턴스만 수집
if ($collect && !in_array($value, $results))
{
$results []= $value;
}
// 결과가 충분할 때 반환하거나 소진된 목록
if (count($results) == $requestedCount || count($results) == $this->_targetCount)
{
return $results;
}
}
// 루프 시작 - resourcesPosition
foreach($this->_positionToTarget as $key => $value)
아래의 값 검색 {
if (!in_array($value, $results))
{
$results []= $value;
}
// 결과가 충분하면 반환하거나 목록이 소진되면 반환
if (count($results) == $requestedCount || count($results) == $this->_targetCount)
{
return $results;
}
}
// 두 "부분"을 모두 반복한 후 결과를 반환
return $results;
}
공용 함수 __toString()
{
return sprintf(
'%s{targets:[ %s]}',
get_class($this),
implode(',', $this->getAllTargets())
);
}
// --- ------------------------
// 비공개 메소드
/**
* 위치별로 내부 매핑(위치에서 대상까지)을 정렬합니다
*/
비공개 함수 _sortPositionTargets()
{
// 아직 키(위치)로 정렬하지 않은 경우
if (!$this->_positionToTargetSorted)
{
ksort($this->_positionToTarget, SORT_REGULAR);
$this->_positionToTargetSorted = true;
}
}
}
/**
 * 주어진 값을 정렬 가능한 고정 크기 주소 공간으로 해시합니다.
 *
 * @author Paul Annesley
 * @package Flexihash
 * @licence http://www.opensource.org/ 라이센스/mit-license.php
 */
인터페이스 Flexihash_Hasher
{
/**
* 주어진 문자열을 32비트 주소 공간으로 해시합니다.
*
* 출력은 32비트 이상의 원시 데이터일 수 있습니다(예:
* 32비트 값을 나타내는 16진수 문자).
*
* 데이터는 0xFFFFFFFF 가능한 값을 가져야 하며
* SORT_REGULAR를 사용하는 PHP 정렬 함수로 정렬할 수 있어야 합니다.
*
* @param string
* @return 혼합 다음을 사용하여 정렬 가능한 형식 0xFFFFFFFF 가능한 값
*/
공개 함수 해시($string);
}
/**
 * Verwendet CRC32, um einen Wert in einen signierten 32-Bit-Int-Adressraum zu hashen.
 * Unter 32-Bit-PHP läuft dies (sicher) in negative Ints über.
 *
 * @author Paul Annesley
 * @package Flexihash
 * @licence http://www.opensource.org/licenses/mit-license.php
 */
Klasse Flexihash_Crc32Hasher
implementiert Flexihash_Hasher
{
/* (nicht-phpdoc)
* @see Flexihash_Hasher::hash()
*/
public function hash($string)
{
return crc32($string);
}
}
/**
 * Verwendet CRC32, um einen Wert in einen 32-Bit-Binärzeichenfolgen-Datenadressraum zu hashen.
 *
 * @author Paul Annesley
 * @package Flexihash
 * @licence http://www. opensource.org/licenses/mit-license.php
 */
class Flexihash_Md5Hasher
implementiert Flexihash_Hasher
{
/* (nicht-phpdoc)
* @see Flexihash_Hasher::hash()
*/
public function hash($string)
{
return substr(md5($string), 0, 8); // 8 Hexits = 32bit
// 4 Bytes binärer MD5-Daten könnten ebenfalls verwendet werden, aber
// Leistung scheint zu sein gleich.
}
}
/**
 * Eine von Flexihash ausgelöste Ausnahme.
 *
 * @author Paul Annesley
 * @package Flexihash
 * @licence http://www.opensource.org /licenses/mit-license.php
 */
class Flexihash_Exception erweitert Exception
{
}

Das Obige hat einen konsistenten Hash-PHP eingeführt, einschließlich Ausnahmeinhalten. Ich hoffe, dass es für Freunde hilfreich sein wird, die sich für PHP-Tutorials interessieren.

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Vorheriger Artikel:PHP的学习--可变变量Nächster Artikel:PHP的学习--可变函数