﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>BlogJava-java路，自己走-文章分类-java.theory</title><link>http://www.blogjava.net/syniii/category/47375.html</link><description /><language>zh-cn</language><lastBuildDate>Thu, 07 Apr 2011 04:50:40 GMT</lastBuildDate><pubDate>Thu, 07 Apr 2011 04:50:40 GMT</pubDate><ttl>60</ttl><item><title>[转]Consistent Hashing - memcached分布式原理</title><link>http://www.blogjava.net/syniii/articles/340783.html</link><dc:creator>杨罗罗</dc:creator><author>杨罗罗</author><pubDate>Wed, 15 Dec 2010 05:35:00 GMT</pubDate><guid>http://www.blogjava.net/syniii/articles/340783.html</guid><wfw:comment>http://www.blogjava.net/syniii/comments/340783.html</wfw:comment><comments>http://www.blogjava.net/syniii/articles/340783.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/syniii/comments/commentRss/340783.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/syniii/services/trackbacks/340783.html</trackback:ping><description><![CDATA[<span style="widows: 2; text-transform: none; text-indent: 0px; border-collapse: separate; font: medium Simsun; white-space: normal; orphans: 2; letter-spacing: normal; color: rgb(0,0,0); word-spacing: 0px; -webkit-border-horizontal-spacing: 0px; -webkit-border-vertical-spacing: 0px; -webkit-text-decorations-in-effect: none; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px" class="Apple-style-span"><span style="text-align: left; line-height: 21px; font-family: verdana, sans-serif; font-size: 14px" class="Apple-style-span">
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">下面这篇文章写的非常好，结合memcached的 特点利用Consistent hasning 算法，可以打造一个非常完备的分布式缓存服务器。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="contents"></p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing#content_2_0">memcached的分布式</a>
    <ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list2">
        <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_1">memcached的分布式是什么意思？</a></li>
    </ul>
    </li>
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_2">Cache::Memcached的分布式方法</a>
    <ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list2">
        <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_3">根据余数计算分散</a></li>
        <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_4">根据余数计算分散的缺点</a></li>
    </ul>
    </li>
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_5">Consistent Hashing</a>
    <ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list2">
        <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_6">Consistent Hashing的简单说明</a></li>
        <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_7">支持Consistent Hashing的函数库</a></li>
    </ul>
    </li>
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.paitoubing.cn/blog/memcached_consistent_hashing/#content_2_8">总结</a></li>
</ul>
<h2 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">memcached的分布式</h2>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">正如第1次中介绍的那样， memcached虽然称为&#8220;分布式&#8221;缓存服务器，但服务器端并没有&#8220;分布式&#8221;功能。 服务器端仅包括 第2次、 第3次 前坂介绍的内存存储功能，其实现非常简单。 至于memcached的分布式，则是完全由客户端程序库实现的。 这种分布式是memcached的最大特点。</p>
<h3 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">memcached的分布式是什么意思？</h3>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">这里多次使用了&#8220;分布式&#8221;这个词，但并未做详细解释。 现在开始简单地介绍一下其原理，各个客户端的实现基本相同。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">下面假设memcached服务器有node1～node3三台， 应用程序要保存键名为&#8220;tokyo&#8221;&#8220;kanagawa&#8221;&#8220;chiba&#8221;&#8220;saitama&#8221;&#8220;gunma&#8221; 的数据。</p>
<p style="text-align: left; padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="img_margin"><img style="border-right-width: 0px; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="memcached-0004-01.png" alt="memcached-0004-01.png" src="http://tech.idv2.com/wp-content/uploads/2008/07/memcached-0004-01.png" width="441" height="405" /></p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">图1 分布式简介：准备</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">首先向memcached中添加&#8220;tokyo&#8221;。将&#8220;tokyo&#8221;传给客户端程序库后， 客户端实现的算法就会根据&#8220;键&#8221;来决定保存数据的memcached服务器。 服务器选定后，即命令它保存&#8220;tokyo&#8221;及其值。</p>
<p style="text-align: left; padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="img_margin"><img style="border-right-width: 0px; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="memcached-0004-02.png" alt="memcached-0004-02.png" src="http://tech.idv2.com/wp-content/uploads/2008/07/memcached-0004-02.png" width="445" height="454" /></p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">图2 分布式简介：添加时</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">同样，&#8220;kanagawa&#8221;&#8220;chiba&#8221;&#8220;saitama&#8221;&#8220;gunma&#8221;都是先选择服务器再保存。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">接下来获取保存的数据。获取时也要将要获取的键&#8220;tokyo&#8221;传递给函数库。 函数库通过与数据保存时相同的算法，根据&#8220;键&#8221;选择服务器。 使用的算法相同，就能选中与保存时相同的服务器，然后发送get命令。 只要数据没有因为某些原因被删除，就能获得保存的值。</p>
<p style="text-align: left; padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="img_margin"><img style="border-right-width: 0px; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="memcached-0004-03.png" alt="memcached-0004-03.png" src="http://tech.idv2.com/wp-content/uploads/2008/07/memcached-0004-03.png" width="449" height="457" /></p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">图3 分布式简介：获取时</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">这样，将不同的键保存到不同的服务器上，就实现了memcached的分布式。 memcached服务器增多后，键就会分散，即使一台memcached服务器发生故障 无法连接，也不会影响其他的缓存，系统依然能继续运行。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">接下来介绍<a style="color: rgb(51,102,153); text-decoration: none" href="http://tech.idv2.com/2008/07/10/memcached-001/">第1次</a><span class="Apple-converted-space">&nbsp;</span>中提到的Perl客户端函数库Cache::Memcached实现的分布式方法。</p>
<h2 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">Cache::Memcached的分布式方法</h2>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">Perl的memcached客户端函数库Cache::Memcached是 memcached的作者Brad Fitzpatrick的作品，可以说是原装的函数库了。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://search.cpan.org/dist/Cache-Memcached/">Cache::Memcached - search.cpan.org</a></li>
</ul>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">该函数库实现了分布式功能，是memcached标准的分布式方法。</p>
<h3 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">根据余数计算分散</h3>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">Cache::Memcached的分布式方法简单来说，就是&#8220;根据服务器台数的余数进行分散&#8221;。 求得键的整数哈希值，再除以服务器台数，根据其余数来选择服务器。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">下面将Cache::Memcached简化成以下的Perl脚本来进行说明。</p>
<div class="hl-surround">
<div class="hl-main"><span style="color: navy">use strict</span><span style="color: gray">;<br />
</span><span style="color: navy">use warnings</span><span style="color: gray">;<br />
</span><span style="color: navy">use String::CRC32</span><span style="color: gray">;<br />
<br />
</span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@nodes</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">node1</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">node2</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">node3</span><span style="color: rgb(139,0,0)">'</span><span style="color: olive">)</span><span style="color: gray">;<br />
</span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@keys</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">tokyo</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">'</span><span style="color: red">kanagawa</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">'</span><span style="color: red">chiba</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">'</span><span style="color: red">saitama</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">'</span><span style="color: red">gunma</span><span style="color: rgb(139,0,0)">'</span><span style="color: olive">)</span><span style="color: gray">;<br />
<br />
</span><span style="color: green">foreach</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$key</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: rgb(0,0,139)">@keys</span><span style="color: olive">)</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">{</span><span style="color: gray"><br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$crc</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">crc32</span><span style="color: olive">(</span><span style="color: rgb(0,0,139)">$key</span><span style="color: olive">)</span><span style="color: gray">;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(255,165,0)"># CRC値</span><span style="color: gray"><br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$mod</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$crc</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>%<span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$#nodes</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>+<span class="Apple-converted-space">&nbsp;</span></span><span style="color: maroon">1</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">)</span><span style="color: gray">;<br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$server</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$nodes</span><span style="color: olive">[</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$mod</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">]</span><span style="color: gray">;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(255,165,0)"># 根据余数选择服务器</span><span style="color: gray"><br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">printf</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">"</span><span style="color: red">%s =&amp;gt; %s</span><span style="color: navy">\n</span><span style="color: rgb(139,0,0)">"</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$key</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$server</span><span style="color: gray">;<br />
</span><span style="color: olive">}</span></div>
</div>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">Cache::Memcached在求哈希值时使用了CRC。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://search.cpan.org/dist/String-CRC32/">String::CRC32 - search.cpan.org</a></li>
</ul>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">首先求得字符串的CRC值，根据该值除以服务器节点数目得到的余数决定服务器。 上面的代码执行后输入以下结果：</p>
<pre>tokyo       =&gt; node2<br />
kanagawa =&gt; node3<br />
chiba       =&gt; node2<br />
saitama   =&gt; node1<br />
gunma     =&gt; node1</pre>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">根据该结果，&#8220;tokyo&#8221;分散到node2，&#8220;kanagawa&#8221;分散到node3等。 多说一句，当选择的服务器无法连接时，Cache::Memcached会将连接次数 添加到键之后，再次计算哈希值并尝试连接。这个动作称为rehash。 不希望rehash时可以在生成Cache::Memcached对象时指定&#8220;rehash =&gt; 0&#8221;选项。</p>
<h3 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">根据余数计算分散的缺点</h3>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">余数计算的方法简单，数据的分散性也相当优秀，但也有其缺点。 那就是当添加或移除服务器时，缓存重组的代价相当巨大。 添加服务器后，余数就会产生巨变，这样就无法获取与保存时相同的服务器， 从而影响缓存的命中率。用Perl写段代码来验证其代价。</p>
<div class="hl-surround">
<div class="hl-main"><span style="color: navy">use strict</span><span style="color: gray">;<br />
</span><span style="color: navy">use warnings</span><span style="color: gray">;<br />
</span><span style="color: navy">use String::CRC32</span><span style="color: gray">;<br />
<br />
</span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@nodes</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@ARGV</span><span style="color: gray">;<br />
</span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@keys</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">a</span><span style="color: rgb(139,0,0)">'</span><span style="color: gray">..</span><span style="color: rgb(139,0,0)">'</span><span style="color: red">z</span><span style="color: rgb(139,0,0)">'</span><span style="color: olive">)</span><span style="color: gray">;<br />
</span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">%nodes</span><span style="color: gray">;<br />
<br />
</span><span style="color: green">foreach</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$key</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">@keys</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">)</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">{</span><span style="color: gray"><br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$hash</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">crc32</span><span style="color: olive">(</span><span style="color: rgb(0,0,139)">$key</span><span style="color: olive">)</span><span style="color: gray">;<br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$mod</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$hash</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>%<span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$#nodes</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>+<span class="Apple-converted-space">&nbsp;</span></span><span style="color: maroon">1</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">)</span><span style="color: gray">;<br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$server</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>=<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$nodes</span><span style="color: olive">[</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$mod</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">]</span><span style="color: gray">;<br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">push</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span>@</span><span style="color: olive">{</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$nodes</span><span style="color: olive">{</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$server</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">}</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">}</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$key</span><span style="color: gray">;<br />
</span><span style="color: olive">}</span><span style="color: gray"><br />
<br />
</span><span style="color: green">foreach</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">my</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$node</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">(</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">sort</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">keys</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">%nodes</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">)</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">{</span><span style="color: gray"><br />
&nbsp;&nbsp;<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">printf</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">"</span><span style="color: red">%s: %s</span><span style="color: navy">\n</span><span style="color: rgb(139,0,0)">"</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$node</span><span style="color: gray">,<span class="Apple-converted-space">&nbsp;</span></span><span style="color: green">join</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(139,0,0)">"</span><span style="color: red">,</span><span style="color: rgb(139,0,0)">"</span><span style="color: gray">, @</span><span style="color: olive">{</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: rgb(0,0,139)">$nodes</span><span style="color: olive">{</span><span style="color: rgb(0,0,139)">$node</span><span style="color: olive">}</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: olive">}</span><span style="color: gray">;<br />
</span><span style="color: olive">}</span></div>
</div>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">这段Perl脚本演示了将&#8220;a&#8221;到&#8220;z&#8221;的键保存到memcached并访问的情况。 将其保存为mod.pl并执行。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">首先，当服务器只有三台时：</p>
<div class="hl-surround">
<div class="hl-main"><span style="color: gray">$<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">mod</span><span style="color: gray">.</span><span style="color: blue">pl</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node1</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node2</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">nod3</span><span style="color: gray"><br />
</span><span style="color: blue">node1</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">a</span><span style="color: gray">,</span><span style="color: blue">c</span><span style="color: gray">,</span><span style="color: blue">d</span><span style="color: gray">,</span><span style="color: blue">e</span><span style="color: gray">,</span><span style="color: blue">h</span><span style="color: gray">,</span><span style="color: blue">j</span><span style="color: gray">,</span><span style="color: blue">n</span><span style="color: gray">,</span><span style="color: blue">u</span><span style="color: gray">,</span><span style="color: blue">w</span><span style="color: gray">,</span><span style="color: blue">x</span><span style="color: gray"><br />
</span><span style="color: blue">node2</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">g</span><span style="color: gray">,</span><span style="color: blue">i</span><span style="color: gray">,</span><span style="color: blue">k</span><span style="color: gray">,</span><span style="color: blue">l</span><span style="color: gray">,</span><span style="color: blue">p</span><span style="color: gray">,</span><span style="color: blue">r</span><span style="color: gray">,</span><span style="color: blue">s</span><span style="color: gray">,</span><span style="color: green">y</span><span style="color: gray"><br />
</span><span style="color: blue">node3</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">b</span><span style="color: gray">,</span><span style="color: blue">f</span><span style="color: gray">,</span><span style="color: blue">m</span><span style="color: gray">,</span><span style="color: blue">o</span><span style="color: gray">,</span><span style="color: rgb(139,0,0)">q,</span><span style="color: red">t</span><span style="color: rgb(139,0,0)">,</span><span style="color: blue">v</span><span style="color: gray">,</span><span style="color: blue">z</span></div>
</div>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">结果如上，node1保存a、c、d、e&#8230;&#8230;，node2保存g、i、k&#8230;&#8230;， 每台服务器都保存了8个到10个数据。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">接下来增加一台memcached服务器。</p>
<div class="hl-surround">
<div class="hl-main"><span style="color: gray">$<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">mod</span><span style="color: gray">.</span><span style="color: blue">pl</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node1</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node2</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node3</span><span style="color: gray"><span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">node4</span><span style="color: gray"><br />
</span><span style="color: blue">node1</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">d</span><span style="color: gray">,</span><span style="color: blue">f</span><span style="color: gray">,</span><span style="color: blue">m</span><span style="color: gray">,</span><span style="color: blue">o</span><span style="color: gray">,</span><span style="color: blue">t</span><span style="color: gray">,</span><span style="color: blue">v</span><span style="color: gray"><br />
</span><span style="color: blue">node2</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">b</span><span style="color: gray">,</span><span style="color: blue">i</span><span style="color: gray">,</span><span style="color: blue">k</span><span style="color: gray">,</span><span style="color: blue">p</span><span style="color: gray">,</span><span style="color: blue">r</span><span style="color: gray">,</span><span style="color: green">y</span><span style="color: gray"><br />
</span><span style="color: blue">node3</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">e</span><span style="color: gray">,</span><span style="color: blue">g</span><span style="color: gray">,</span><span style="color: blue">l</span><span style="color: gray">,</span><span style="color: blue">n</span><span style="color: gray">,</span><span style="color: blue">u</span><span style="color: gray">,</span><span style="color: blue">w</span><span style="color: gray"><br />
</span><span style="color: blue">node4</span><span style="color: gray">:<span class="Apple-converted-space">&nbsp;</span></span><span style="color: blue">a</span><span style="color: gray">,</span><span style="color: blue">c</span><span style="color: gray">,</span><span style="color: blue">h</span><span style="color: gray">,</span><span style="color: blue">j</span><span style="color: gray">,</span><span style="color: rgb(139,0,0)">q,</span><span style="color: red">s</span><span style="color: rgb(139,0,0)">,</span><span style="color: blue">x</span><span style="color: gray">,</span><span style="color: blue">z</span></div>
</div>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">添加了node4。可见，只有d、i、k、p、r、y命中了。像这样，添加节点后 键分散到的服务器会发生巨大变化。26个键中只有六个在访问原来的服务器， 其他的全都移到了其他服务器。命中率降低到23%。在Web应用程序中使用memcached时， 在添加memcached服务器的瞬间缓存效率会大幅度下降，负载会集中到数据库服务器上， 有可能会发生无法提供正常服务的情况。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">mixi的Web应用程序运用中也有这个问题，导致无法添加memcached服务器。 但由于使用了新的分布式方法，现在可以轻而易举地添加memcached服务器了。 这种分布式方法称为 Consistent Hashing。</p>
<h2 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">Consistent Hashing</h2>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">关于Consistent Hashing的思想，mixi株式会社的开发blog等许多地方都介绍过， 这里只简单地说明一下。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://alpha.mixi.co.jp/blog/?p=158">mixi Engineers&#8217; Blog - スマートな分散で快適キャッシュライフ</a></li>
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.hyuki.com/yukiwiki/wiki.cgi?ConsistentHashing">ConsistentHashing - コンシステント ハッシュ法</a></li>
</ul>
<h3 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">Consistent Hashing的简单说明</h3>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">Consistent Hashing如下所示：首先求出memcached服务器（节点）的哈希值， 并将其配置到0～2<sup>32</sup>的圆（continuum）上。 然后用同样的方法求出存储数据的键的哈希值，并映射到圆上。 然后从数据映射到的位置开始顺时针查找，将数据保存到找到的第一个服务器上。 如果超过2<sup>32</sup>仍然找不到服务器，就会保存到第一台memcached服务器上。</p>
<p style="text-align: left; padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="img_margin"><img style="border-right-width: 0px; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="memcached-0004-04.png" alt="memcached-0004-04.png" src="http://tech.idv2.com/wp-content/uploads/2008/07/memcached-0004-04.png" width="521" height="409" /></p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">图4 Consistent Hashing：基本原理</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">从上图的状态中添加一台memcached服务器。余数分布式算法由于保存键的服务器会发生巨大变化 而影响缓存的命中率，但Consistent Hashing中，只有在continuum上增加服务器的地点逆时针方向的 第一台服务器上的键会受到影响。</p>
<p style="text-align: left; padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px" class="img_margin"><img style="border-right-width: 0px; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="memcached-0004-05.png" alt="memcached-0004-05.png" src="http://tech.idv2.com/wp-content/uploads/2008/07/memcached-0004-05.png" width="518" height="439" /></p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">图5 Consistent Hashing：添加服务器</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">因此，Consistent Hashing最大限度地抑制了键的重新分布。 而且，有的Consistent Hashing的实现方法还采用了虚拟节点的思想。 使用一般的hash函数的话，服务器的映射地点的分布非常不均匀。 因此，使用虚拟节点的思想，为每个物理节点（服务器） 在continuum上分配100～200个点。这样就能抑制分布不均匀， 最大限度地减小服务器增减时的缓存重新分布。</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">通过下文中介绍的使用Consistent Hashing算法的memcached客户端函数库进行测试的结果是， 由服务器台数（n）和增加的服务器台数（m）计算增加服务器后的命中率计算公式如下：</p>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">(1 - n/(n+m)) * 100</p>
<h3 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">支持Consistent Hashing的函数库</h3>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">本连载中多次介绍的Cache::Memcached虽然不支持Consistent Hashing， 但已有几个客户端函数库支持了这种新的分布式算法。 第一个支持Consistent Hashing和虚拟节点的memcached客户端函数库是 名为libketama的PHP库，由last.fm开发。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://www.lastfm.jp/user/RJ/journal/2007/04/10/rz_libketama_-_a_consistent_hashing_algo_for_memcache_clients">libketama - a consistent hashing algo for memcache clients &#8211; RJ ブログ - Users at Last.fm</a></li>
</ul>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">至于Perl客户端，连载的第1次 中介绍过的Cache::Memcached::Fast和Cache::Memcached::libmemcached支持 Consistent Hashing。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://search.cpan.org/dist/Cache-Memcached-Fast/">Cache::Memcached::Fast - search.cpan.org</a></li>
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://search.cpan.org/dist/Cache-Memcached-libmemcached/">Cache::Memcached::libmemcached - search.cpan.org</a></li>
</ul>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">两者的接口都与Cache::Memcached几乎相同，如果正在使用Cache::Memcached， 那么就可以方便地替换过来。Cache::Memcached::Fast重新实现了libketama， 使用Consistent Hashing创建对象时可以指定ketama_points选项。</p>
<pre>my $memcached = Cache::Memcached::Fast-&gt;new({<br />
servers =&gt; ["192.168.0.1:11211","192.168.0.2:11211"],<br />
ketama_points =&gt; 150<br />
});</pre>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">另外，Cache::Memcached::libmemcached 是一个使用了Brain Aker开发的C函数库libmemcached的Perl模块。 libmemcached本身支持几种分布式算法，也支持Consistent Hashing， 其Perl绑定也支持Consistent Hashing。</p>
<ul style="padding-bottom: 0px; list-style-type: none; margin: 5px 0px 5px 16px; padding-left: 16px; padding-right: 0px; padding-top: 0px" class="list1">
    <li style="padding-bottom: 2px; list-style-type: disc; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 2px"><a style="color: rgb(51,102,153); text-decoration: none" href="http://tangent.org/552/libmemcached.html">Tangent Software: libmemcached</a></li>
</ul>
<h2 style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; padding-top: 0px">总结</h2>
<p style="padding-bottom: 0px; margin: 1em 0px 0.5em; padding-left: 0px; padding-right: 0px; padding-top: 0px">本次介绍了memcached的分布式算法，主要有memcached的分布式是由客户端函数库实现， 以及高效率地分散数据的Consistent Hashing算法。下次将介绍mixi在memcached应用方面的一些经验， 和相关的兼容应用程序。</p>
</span></span>
 <img src ="http://www.blogjava.net/syniii/aggbug/340783.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/syniii/" target="_blank">杨罗罗</a> 2010-12-15 13:35 <a href="http://www.blogjava.net/syniii/articles/340783.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>