没有办法避免所有的碎片,但是仔细选择query_cache_min_res_unit 可以有效的避免内存浪费,query_cache_min_res_unit 过大会产生内存浪费和碎片,query_cache_min_res_unit 过小导致内存经常allocate 也回带来性能影响,所以query_cache_min_res_unit 的设置很关键。
要找一个既能减少碎片,又能避免过多分配内存的情况可以尝试一下的方法,求出平均的缓存大小,(query_cache_size-qcache_free_memory)/qcache_in_queries 这样可以求出平均的缓存结果大小,如果缓存的大小分布的不均匀,那么就很难找到合适的大小,但是我们对于大结果最好不要缓存在mysql服务器端,我们可以采用第三方的应用缓存服务器缓存,这样就能大概估算query_cache_limit的大小。从而能找到一个平衡点。
同时我们可以通过query_free_blocks来测试缓存中碎片的情况,他说明缓存中有多少内存块处于空闲状态,可以试想一下,碎片最严重的情况应该是在两个缓存块之间就有一个空闲的比最小值还小的块,但是同时还有一种可能是前一半是连续存蓄的后一半是空的,如果qcache_free_blocks=qcache_total_blocks/2,那么极有可能是碎片很严重的情况,你可以从qcache_lowmem_prunes参数看出,如果qcache_lowmem_prunes在增加并且还有大量的空闲块,那么就说明碎片确实是很严重。可以使用flush querycache命令移除碎片,她会把所有的存蓄块移动到一端,把空闲块移动到另外一端!reset query cache 会把查询从缓存中移除!