如何使用PHP源码来操作memcached服务
如果管理员不让我们去加载 memcache.dll 文件,我们可以直接通过源码操作
看高手的源代码也很有意思!
特点:
1.不需要开启memcache扩展
2.使用fsockopen()套接字连接memcached
3.同样执行执行CRUD
引用别人写的一个类:
memcached-client.php
<?php //// +---------------------------------------------------------------------------+// | memcached client, PHP |// +---------------------------------------------------------------------------+// | Copyright (c) 2003 Ryan T. Dean <rtdean@cytherianage.net> |// | All rights reserved. |// | |// | Redistribution and use in source and binary forms, with or without |// | modification, are permitted provided that the following conditions |// | are met: |// | |// | 1. Redistributions of source code must retain the above copyright |// | notice, this list of conditions and the following disclaimer. |// | 2. Redistributions in binary form must reproduce the above copyright |// | notice, this list of conditions and the following disclaimer in the |// | documentation and/or other materials provided with the distribution. |// | |// | THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR |// | IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES |// | OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. |// | IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, |// | INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT |// | NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |// | DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |// | THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |// | (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |// | THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |// +---------------------------------------------------------------------------+// | Author: Ryan T. Dean <rtdean> |// | Heavily influenced by the Perl memcached client by Brad Fitzpatrick. |// | Permission granted by Brad Fitzpatrick for relicense of ported Perl |// | client logic under 2-clause BSD license. |// +---------------------------------------------------------------------------+//// $TCAnet$///** * This is the PHP client for memcached - a distributed memory cache daemon. * More information is available at http://www.danga.com/memcached/ * * Usage example: * * require_once 'memcached.php'; * * $mc = new memcached(array( * 'servers' => array('127.0.0.1:10000', * array('192.0.0.1:10010', 2), * '127.0.0.1:10020'), * 'debug' => false, * 'compress_threshold' => 10240, * 'persistant' => true)); * * $mc->add('key', array('some', 'array')); * $mc->replace('key', 'some random string'); * $val = $mc->get('key'); * * @author Ryan T. Dean <rtdean> * @package memcached-client * @version 0.1.2 */// {{{ requirements// }}}// {{{ constants// {{{ flags/** * Flag: indicates data is serialized */define("MEMCACHE_SERIALIZED", 1 * @package memcached-client */class memcached{ // {{{ properties // {{{ public /** * Command statistics * * @var array * @access public */ var $stats; // }}} // {{{ private /** * Cached Sockets that are connected * * @var array * @access private */ var $_cache_sock; /** * Current debug status; 0 - none to 9 - profiling * * @var boolean * @access private */ var $_debug; /** * Dead hosts, assoc array, 'host'=>'unixtime when ok to check again' * * @var array * @access private */ var $_host_dead; /** * Is compression available? * * @var boolean * @access private */ var $_have_zlib; /** * Do we want to use compression? * * @var boolean * @access private */ var $_compress_enable; /** * At how many bytes should we compress? * * @var interger * @access private */ var $_compress_threshold; /** * Are we using persistant links? * * @var boolean * @access private */ var $_persistant; /** * If only using one server; contains ip:port to connect to * * @var string * @access private */ var $_single_sock; /** * Array containing ip:port or array(ip:port, weight) * * @var array * @access private */ var $_servers; /** * Our bit buckets * * @var array * @access private */ var $_buckets; /** * Total # of bit buckets we have * * @var interger * @access private */ var $_bucketcount; /** * # of total servers we have * * @var interger * @access private */ var $_active; // }}} // }}} // {{{ methods // {{{ public functions // {{{ memcached() /** * Memcache initializer * * @param array $args Associative array of settings * * @return mixed * @access public */ function memcached ($args) { $this->set_servers($args['servers']); $this->_debug = $args['debug']; $this->stats = array(); $this->_compress_threshold = $args['compress_threshold']; $this->_persistant = isset($args['persistant']) ? $args['persistant'] : false; $this->_compress_enable = true; $this->_have_zlib = function_exists("gzcompress"); $this->_cache_sock = array(); $this->_host_dead = array(); } // }}} // {{{ add() /** * Adds a key/value to the memcache server if one isn't already set with * that key * * @param string $key Key to set with data * @param mixed $val Value to store * @param interger $exp (optional) Time to expire data at * * @return boolean * @access public */ function add ($key, $val, $exp = 0) { return $this->_set('add', $key, $val, $exp); } // }}} // {{{ decr() /** * Decriment a value stored on the memcache server * * @param string $key Key to decriment * @param interger $amt (optional) Amount to decriment * * @return mixed FALSE on failure, value on success * @access public */ function decr ($key, $amt=1) { return $this->_incrdecr('decr', $key, $amt); } // }}} // {{{ delete() /** * Deletes a key from the server, optionally after $time * * @param string $key Key to delete * @param interger $time (optional) How long to wait before deleting * * @return boolean TRUE on success, FALSE on failure * @access public */ function delete ($key, $time = 0) { if (!$this->_active) return false; $sock = $this->get_sock($key); if (!is_resource($sock)) return false; $key = is_array($key) ? $key[1] : $key; $this->stats['delete']++; $cmd = "delete $key $time\r\n"; if(!fwrite($sock, $cmd, strlen($cmd))) { $this->_dead_sock($sock); return false; } $res = trim(fgets($sock)); if ($this->_debug) printf("MemCache: delete %s (%s)\n", $key, $res); if ($res == "DELETED") return true; return false; } // }}} // {{{ disconnect_all() /** * Disconnects all connected sockets * * @access public */ function disconnect_all () { foreach ($this->_cache_sock as $sock) fclose($sock); $this->_cache_sock = array(); } // }}} // {{{ enable_compress() /** * Enable / Disable compression * * @param boolean $enable TRUE to enable, FALSE to disable * * @access public */ function enable_compress ($enable) { $this->_compress_enable = $enable; } // }}} // {{{ forget_dead_hosts() /** * Forget about all of the dead hosts * * @access public */ function forget_dead_hosts () { $this->_host_dead = array(); } // }}} // {{{ get() /** * Retrieves the value associated with the key from the memcache server * * @param string $key Key to retrieve * * @return mixed * @access public */ function get ($key) { if (!$this->_active) return false; $sock = $this->get_sock($key); if (!is_resource($sock)) return false; $this->stats['get']++; $cmd = "get $key\r\n"; if (!fwrite($sock, $cmd, strlen($cmd))) { $this->_dead_sock($sock); return false; } $val = array(); $this->_load_items($sock, $val); if ($this->_debug) foreach ($val as $k => $v) printf("MemCache: sock %s got %s => %s\r\n", $sock, $k, $v); return $val[$key]; } // }}} // {{{ get_multi() /** * Get multiple keys from the server(s) * * @param array $keys Keys to retrieve * * @return array * @access public * get('key1') keys=array('key1','key2'); */ function get_multi ($keys) { if (!$this->_active) return false; $this->stats['get_multi']++; foreach ($keys as $key) { $sock = $this->get_sock($key); if (!is_resource($sock)) continue; $key = is_array($key) ? $key[1] : $key; if (!isset($sock_keys[$sock])) { $sock_keys[$sock] = array(); $socks[] = $sock; } $sock_keys[$sock][] = $key; } // Send out the requests foreach ($socks as $sock) { $cmd = "get"; foreach ($sock_keys[$sock] as $key) { $cmd .= " ". $key; } $cmd .= "\r\n"; if (fwrite($sock, $cmd, strlen($cmd))) { $gather[] = $sock; } else { $this->_dead_sock($sock); } } // Parse responses $val = array(); foreach ($gather as $sock) { $this->_load_items($sock, $val); } if ($this->_debug) foreach ($val as $k => $v) printf("MemCache: got %s => %s\r\n", $k, $v); return $val; } // }}} // {{{ incr() /** * Increments $key (optionally) by $amt * * @param string $key Key to increment * @param interger $amt (optional) amount to increment * * @return interger New key value? * @access public */ function incr ($key, $amt=1) { return $this->_incrdecr('incr', $key, $amt); } // }}} // {{{ replace() /** * Overwrites an existing value for key; only works if key is already set * * @param string $key Key to set value as * @param mixed $value Value to store * @param interger $exp (optional) Experiation time * * @return boolean * @access public */ function replace ($key, $value, $exp=0) { return $this->_set('replace', $key, $value, $exp); } // }}} // {{{ run_command() /** * Passes through $cmd to the memcache server connected by $sock; returns * output as an array (null array if no output) * * NOTE: due to a possible bug in how PHP reads while using fgets(), each * line may not be terminated by a \r\n. More specifically, my testing * has shown that, on FreeBSD at least, each line is terminated only * with a \n. This is with the PHP flag auto_detect_line_endings set * to falase (the default). * * @param resource $sock Socket to send command on * @param string $cmd Command to run * * @return array Output array * @access public */ function run_command ($sock, $cmd) { if (!is_resource($sock)) return array(); if (!fwrite($sock, $cmd, strlen($cmd))) return array(); while (true) { $res = fgets($sock); $ret[] = $res; if (preg_match('/^END/', $res)) break; if (strlen($res) == 0) break; } return $ret; } // }}} // {{{ set() /** * Unconditionally sets a key to a given value in the memcache. Returns true * if set successfully. * * @param string $key Key to set value as * @param mixed $value Value to set * @param interger $exp (optional) Experiation time * * @return boolean TRUE on success * @access public */ function set ($key, $value, $exp=0) { return $this->_set('set', $key, $value, $exp); } // }}} // {{{ set_compress_threshold() /** * Sets the compression threshold * * @param interger $thresh Threshold to compress if larger than * * @access public */ function set_compress_threshold ($thresh) { $this->_compress_threshold = $thresh; } // }}} // {{{ set_debug() /** * Sets the debug flag * * @param boolean $dbg TRUE for debugging, FALSE otherwise * * @access public * * @see memcahced::memcached */ function set_debug ($dbg) { $this->_debug = $dbg; } // }}} // {{{ set_servers() /** * Sets the server list to distribute key gets and puts between * * @param array $list Array of servers to connect to * * @access public * * @see memcached::memcached() */ function set_servers ($list) { $this->_servers = $list; $this->_active = count($list); $this->_buckets = null; $this->_bucketcount = 0; $this->_single_sock = null; if ($this->_active == 1) $this->_single_sock = $this->_servers[0]; } // }}} // }}} // {{{ private methods // {{{ _close_sock() /** * Close the specified socket * * @param string $sock Socket to close * * @access private */ function _close_sock ($sock) { $host = array_search($sock, $this->_cache_sock); fclose($this->_cache_sock[$host]); unset($this->_cache_sock[$host]); } // }}} // {{{ _connect_sock() /** * Connects $sock to $host, timing out after $timeout * * @param interger $sock Socket to connect * @param string $host Host:IP to connect to * @param float $timeout (optional) Timeout value, defaults to 0.25s * * @return boolean * @access private */ function _connect_sock (&$sock, $host, $timeout = 0.25) { list ($ip, $port) = explode(":", $host); if ($this->_persistant == 1) { $sock = @pfsockopen($ip, $port, $errno, $errstr, $timeout); } else { $sock = @fsockopen($ip, $port, $errno, $errstr, $timeout); } if (!$sock) return false; return true; } // }}} // {{{ _dead_sock() /** * Marks a host as dead until 30-40 seconds in the future * * @param string $sock Socket to mark as dead * * @access private */ function _dead_sock ($sock) { $host = array_search($sock, $this->_cache_sock); list ($ip, $port) = explode(":", $host); $this->_host_dead[$ip] = time() + 30 + intval(rand(0, 10)); $this->_host_dead[$host] = $this->_host_dead[$ip]; unset($this->_cache_sock[$host]); } // }}} // {{{ get_sock() /** * get_sock * * @param string $key Key to retrieve value for; * * @return mixed resource on success, false on failure * @access private */ function get_sock ($key) { if (!$this->_active) return false; if ($this->_single_sock !== null) return $this->sock_to_host($this->_single_sock); $hv = is_array($key) ? intval($key[0]) : $this->_hashfunc($key); if ($this->_buckets === null) { foreach ($this->_servers as $v) { if (is_array($v)) { for ($i=0; $i_buckets = $bu; $this->_bucketcount = count($bu); } $realkey = is_array($key) ? $key[1] : $key; for ($tries = 0; $tries_buckets[$hv % $this->_bucketcount]; $sock = $this->sock_to_host($host); if (is_resource($sock)) return $sock; $hv += $this->_hashfunc($tries . $realkey); } return false; } // }}} // {{{ _hashfunc() /** * Creates a hash interger based on the $key * * @param string $key Key to hash * * @return interger Hash value * @access private */ function _hashfunc ($key) { $hash = 0; for ($i=0; $i<strlen ord return _incrdecr perform increment on string command to key it interger amount adjust new value of private function if>_active) return null; $sock = $this->get_sock($key); if (!is_resource($sock)) return null; $key = is_array($key) ? $key[1] : $key; $this->stats[$cmd]++; if (!fwrite($sock, "$cmd $key $amt\r\n")) return $this->_dead_sock($sock); stream_set_timeout($sock, 1, 0); $line = fgets($sock); if (!preg_match('/^(\d+)/', $line, $match)) return null; return $match[1]; } // }}} // {{{ _load_items() /** * Load items into $ret from $sock * * @param resource $sock Socket to read from * @param array $ret Returned values * * @access private */ function _load_items ($sock, &$ret) { while (1) { $decl = fgets($sock); if ($decl == "END\r\n") { return true; } elseif (preg_match('/^VALUE (\S+) (\d+) (\d+)\r\n$/', $decl, $match)) { list($rkey, $flags, $len) = array($match[1], $match[2], $match[3]); $bneed = $len+2; $offset = 0; while ($bneed > 0) { $data = fread($sock, $bneed); $n = strlen($data); if ($n == 0) break; $offset += $n; $bneed -= $n; $ret[$rkey] .= $data; } if ($offset != $len+2) { // Something is borked! if ($this->_debug) printf("Something is borked! key %s expecting %d got %d length\n", $rkey, $len+2, $offset); unset($ret[$rkey]); $this->_close_sock($sock); return false; } $ret[$rkey] = rtrim($ret[$rkey]); if ($this->_have_zlib && $flags & MEMCACHE_COMPRESSED) $ret[$rkey] = gzuncompress($ret[$rkey]); if ($flags & MEMCACHE_SERIALIZED) $ret[$rkey] = unserialize($ret[$rkey]); } else { if ($this->_debug) print("Error parsing memcached response\n"); return 0; } } } // }}} // {{{ _set() /** * Performs the requested storage operation to the memcache server * * @param string $cmd Command to perform * @param string $key Key to act on * @param mixed $val What we need to store * @param interger $exp When it should expire * * @return boolean * @access private */ function _set ($cmd, $key, $val, $exp) { if (!$this->_active) return false; //get_sock就是去获取到memcached服务器连接 $sock = $this->get_sock($key); if (!is_resource($sock)) return false; $this->stats[$cmd]++; $flags = 0; if (!is_scalar($val)) { $val = serialize($val); $flags |= MEMCACHE_SERIALIZED; if ($this->_debug) printf("client: serializing data as it is not scalar\n"); } $len = strlen($val); if ($this->_have_zlib && $this->_compress_enable && $this->_compress_threshold && $len >= $this->_compress_threshold) { $c_val = gzcompress($val, 9); $c_len = strlen($c_val); if ($c_len _debug) printf("client: compressing data; was %d bytes is now %d bytes\n", $len, $c_len); $val = $c_val; $len = $c_len; $flags |= MEMCACHE_COMPRESSED; } } if (!fwrite($sock, "$cmd $key $flags $exp $len\r\n$val\r\n")) return $this->_dead_sock($sock); $line = trim(fgets($sock)); if ($this->_debug) { if ($flags & MEMCACHE_COMPRESSED) $val = 'compressed data'; printf("MemCache: %s %s => %s (%s)\n", $cmd, $key, $val, $line); } if ($line == "STORED") return true; return false; } // }}} // {{{ sock_to_host() /** * Returns the socket for the host * * @param string $host Host:IP to get socket for * * @return mixed IO Stream or false * @access private */ function sock_to_host ($host) { if (isset($this->_cache_sock[$host])) return $this->_cache_sock[$host]; $now = time(); list ($ip, $port) = explode (":", $host); if (isset($this->_host_dead[$host]) && $this->_host_dead[$host] > $now || isset($this->_host_dead[$ip]) && $this->_host_dead[$ip] > $now) return null; if (!$this->_connect_sock($sock, $host)) return $this->_dead_sock($host); // Do not buffer writes stream_set_write_buffer($sock, 0); $this->_cache_sock[$host] = $sock; return $this->_cache_sock[$host]; } // }}} // }}} // }}}}// }}}?> </strlen></rtdean></rtdean>
代用代码:
<?php require_once 'memcached-client.php'; $mc = new memcached(array( 'servers' => array('127.0.0.1:11211'), //连接的memcacheip和端口 'debug' => false, //是否debug 'compress_threshold' => 10240, /*最大压缩*/ 'persistant' => true)); /*是否是持久连接*/ $mc->set('key1', array('some', 'array')); // $mc->replace('key', 'some random string'); $val = $mc->get('key1'); var_dump($val); //修改 $mc->replace('key1', "北京"); $val = $mc->get('key1'); var_dump($val); //删除 $mc->delete('key1'); $val = $mc->get('key1'); echo "删除后"; var_dump($val);

PHP在現代Web開發中仍然重要,尤其在內容管理和電子商務平台。 1)PHP擁有豐富的生態系統和強大框架支持,如Laravel和Symfony。 2)性能優化可通過OPcache和Nginx實現。 3)PHP8.0引入JIT編譯器,提升性能。 4)雲原生應用通過Docker和Kubernetes部署,提高靈活性和可擴展性。

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

PHP和Python各有優勢,適合不同場景。 1.PHP適用於web開發,提供內置web服務器和豐富函數庫。 2.Python適合數據科學和機器學習,語法簡潔且有強大標準庫。選擇時應根據項目需求決定。

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP成為許多網站首選技術棧的原因包括其易用性、強大社區支持和廣泛應用。 1)易於學習和使用,適合初學者。 2)擁有龐大的開發者社區,資源豐富。 3)廣泛應用於WordPress、Drupal等平台。 4)與Web服務器緊密集成,簡化開發部署。

PHP在現代編程中仍然是一個強大且廣泛使用的工具,尤其在web開發領域。 1)PHP易用且與數據庫集成無縫,是許多開發者的首選。 2)它支持動態內容生成和麵向對象編程,適合快速創建和維護網站。 3)PHP的性能可以通過緩存和優化數據庫查詢來提升,其廣泛的社區和豐富生態系統使其在當今技術棧中仍具重要地位。

在PHP中,弱引用是通過WeakReference類實現的,不會阻止垃圾回收器回收對象。弱引用適用於緩存系統和事件監聽器等場景,需注意其不能保證對象存活,且垃圾回收可能延遲。

\_\_invoke方法允許對象像函數一樣被調用。 1.定義\_\_invoke方法使對象可被調用。 2.使用$obj(...)語法時,PHP會執行\_\_invoke方法。 3.適用於日誌記錄和計算器等場景,提高代碼靈活性和可讀性。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

Atom編輯器mac版下載
最受歡迎的的開源編輯器

ZendStudio 13.5.1 Mac
強大的PHP整合開發環境

SublimeText3漢化版
中文版,非常好用

WebStorm Mac版
好用的JavaScript開發工具

VSCode Windows 64位元 下載
微軟推出的免費、功能強大的一款IDE編輯器