类的模式方法都是 public 的,以 两个下划线开头,在某些情况下自动被调用,有点类似 hook 或者 事件触发。
魔术方法的传参方式不能使用引用。
__construct() :
构造函数,实例化对象时调用,可以接收传入的参数
__destruct():
析构函数,对象被注销、脚本结束、对象的所有引用被删除时调用(垃圾回收),该函数不能接收参数,析构函数可以在代码中被显示调用,但是不建议这样做,应该由系统在垃圾回收时自动调用。
__call(string methodName, array arguments):
试图调用一个不存在的方法时被调用,这在 PHP 框架的路由设计中很有用,比如 传了一个错误的路径解析出一个不存在的 action,或者是某种有意设计成的路劲规则 ,可以用该方法灵活处理:按路劲的信息直接指向某个文件或跳转到错误页面。
static __callStatic(string methodName, array arguments):
试图调用一个不存在的静态方法时调用,这个方法应该声明为静态方法。
__get(string key):
试图通过对象访问一个不能访问的属性(不存在的属性或保护、私有属性)时被调用,应该返回一个值。
例如 $obj->key ; //key 属性不存在或不是 public 是就会触发 __get(),并获得返回值(或者你也可以直接抛出错误,或做其他处理)
如果使用 empty() 来检测一个对象通过 __get() 方法返回的属性值,将得到 true, 而 isset()() 的检测将得到 false.
__set(string key):
试图通过对象为一个不能访问的属性赋值时被调用,该方法的返回值被忽略。
__isset(string key) :
当使用 isset() 或 empty() 检测一个无法在外部访问的对象属性时被触发,应当返回 bool 值。
__unset(string key) :
对一个无法在外部访问的对象属性使用 unset() 注销时被触发
__sleep() :
对一个对象进行使用 serialize() 序列化时被调用,返回一个以属性名称为元素的数组告诉脚本应该保存对象的哪些属性值。
__wakeup():
对一个对象系列化字符串( 通过 serialize($obj)得到的字符串 )使用 unserialize() 反序列化时被调用。
protected $link;
private $server, $username, $password, $db;
public function __construct($server, $username, $password, $db) {
$this->server = $server;
$this->username = $username;
$this->password = $password;
$this->db = $db;
$this->connect();
}
private function connect() {
$this->link = mysql_connect()($this->server, $this->username, $this->password);
mysql_select_db($this->db, $this->link);
}
public function __sleep() {//系列化时只需保存这些属性的值
return array('server', 'username', 'password', 'db');//必须返回一个数组
}
public function __wakeup() { //反系列化得到对象后立即连接数据库
$this->connect();
}
}
__toString():
试图将对象作为一个字符串使用时被调用,返回一个字符串,类似 js 对象的 toString()
__invoke():
将一个对象作为函数调用时被调用。如
$object($a,$b) 将调用 $object->__invoke($a,$b)
__set_state() :
对一个对象使用 var_export() 时被调用,其返回值将被打印
__clone():
对一个对象使用 clone 操作时被调用,无返回值,该方法可用于 在返回 clone 得到的对象之前修改其属性使得 clone 得到的对象与 被操 作的对象属性值可以不同,但并不能通过返回 null 或 false 来阻止 对象克隆操作, clone 操作得到的对象不是通过该方法的返回值返回 的。
注意事项:
__get、__set 方法可常用于灵活处理 对象私有属性、保护属性的访问。
由于 PHP 对象对于的方法使用 isset() 或 empty() 判定时,不会认为 方法是一个可访问的属性(写多了 javascript 要注意了,PHP 里面 属性就是属性方法就是方法不可混淆)
在使用 __get 时 ,你可能会在该方法中对属性使用 isset($this->key) 判断,尤其是要特别处理 私有属性的时候,这时要注意 __set() 有没有定义以及如何定义以免出现误判。
如果一个对象的方法在调用时不需要传参,或者传参是固定的,可以通过 __get() 方法来将方法属性化,在 __get() 里面自动调用该方法并将值返回。
比如在类内定义 __get 方法如下:
public function __get($key){
if(property_exists($this,$key)){
return $this->$key; //私有、保护属性允许访问
}else if(method_exists($this,'get'.$key)){
return $this->{'get'.$key}();
//或者....
// $methodName = 'get'.$key;
//return $this-> $methodName();
}else{
thrown new Exception('class '.__class__.' do not has property '.$key);
}
}
//方法属性化访问:
$obj->getModelName();
$obj->ModelName; //属性化
unserialize() 方法:当试图反序列化一个对象时,该函数需要知道对象的 类,如果这个系列化字符串从其他方式得来,脚本环境中没有定义对象的类,就需要将类的文件引入,unserialize() 第二个参数是可选的 callback 型参数,用于引入类所在的文件。
function importClass($calssName){
include('xxxx.php'); //包含该类的文件
}
unserialize($objstr,$callbackName);
类的自动加载: __autoload()
__autoload() 是PHP执行环境中约定的一个函数而非某个类的方法,如果一个类在使用之前没有加载到当前文件,会自动调用 __autoload() 函数来加载该类,通常这些类的加载规则都是约定的,比如这些类包含在以类名命名的文件内,该方法可以实现类的按需加载,避免脚本执行前加载不必要的类从而降低资源占用、提交性能。
注意:__autoload() 内的错误不能被 try-catch 捕获。
require_once(PATH.'/calsses/'.$class_name.'.php');
}
$obj1 = new mycalss1();
注册 __autoload() 自动调用的函数:
spl 代码库在 PHP5.0 之后默认自动启用
spl_autoload_register([callback]); //不将具体实现的加载代码写在 __autoload() 内,可使用该函数注册回调函数。
如果使用类的方法作为回调函数需要传入一个数组:
spl_autoload_register(array('class_name'|$obj,'method_name'));
例如:
spl_autoload_register(array($this,'autoloadClass'));
spl_autoload_register(array('YiiBase','autoload'));// YII 框架的自动加载类的实现, YiiBase 类实现了一个autoload 方法。
spl_autoload_register() 可以注册多个加载函数,成功加载类文件之前将逐个尝试所有注册的加载函数。这在不同的类使用不同逻辑来导入类文件的时候很有用。
spl_autoload_unregister(); //取消某个注册的加载函数,参数与 spl_autoload_register() 相同.
spl_autoload_functions();// 以数组形式返回所有注册的 __autoload() 函数
spl_autoload(class_name[,file_extentions]); // __autoload() 函数的默认实现。 spl_autoload_register() 被调用时如果没有传入 函数名,则默认使用该函数,该函数的执行规则是: 类名转为小写作为文件名,传入的 file_extentions(多个扩展名以逗号隔开,默认为 .inc 和 .php)为扩展名,根据得到的文件名尝试在 php.ini 内设置的 include paths 中搜索。
spl_autoload_call(class_name);//手动调用所有注册的 __autoload() 函数来主动加载类文件
spl_autoload_extentions([file_extentions]); //注册或返回 spl_autoload() 中可以使用的文件扩展名,扩展名可以是 .a.b 这样的形式,例如:
spl_autoload_extentions(".class.php");
spl_autoload_register(); //使用spl_autoload() 来尝试自动加载类文件
//这样 spl_autoload('myclassName'); 会尝试加载 文件 "myclassName.class.php" .
php模拟登录qq邮箱(curl命令详解),感兴趣的朋友可以参考下。
<?php
header("Content-type:text/html;charset=utf-8");
$cookie_file = dirname(__FILE__)."/cookie_".md5(basename(__FILE__)).".txt"; // 设置Cookie文件保存路径及文件名
function vlogin($url,$data){ // 模拟登录获取Cookie函数
$curl = curl_init(); // 启动一个CURL会话
curl_setopt($curl, CURLOPT_URL, $url); // 要访问的地址
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0); // 对认证证书来源的检查
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1); // 从证书中检查SSL加密算法是否存在
curl_setopt($curl, CURLOPT_USERAGENT, $_SERVER['HTTP_USER_AGENT']); // 模拟用户使用的浏览器
curl_setopt($curl, CURLOPT_FOLLOWLOCATION, 1); // 使用自动跳转
curl_setopt($curl, CURLOPT_AUTOREFERER, 1); // 自动设置Referer
curl_setopt($curl, CURLOPT_POST, 1); // 发送一个常规的Post请求
curl_setopt($curl, CURLOPT_POSTFIELDS, $data); // Post提交的数据包
curl_setopt($curl, CURLOPT_COOKIEJAR, $GLOBALS['cookie_file']); // 存放Cookie信息的文件名称
curl_setopt($curl, CURLOPT_COOKIEFILE, $GLOBALS['cookie_file']); // 读取上面所储存的Cookie信息
curl_setopt($curl, CURLOPT_TIMEOUT, 30); // 设置超时限制防止死循环
curl_setopt($curl, CURLOPT_HEADER, 0); // 显示返回的Header区域内容
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 获取的信息以文件流的形式返回
$tmpInfo = curl_exec($curl); // 执行操作
if (curl_errno($curl)) {
echo 'Errno'.curl_error($curl);
}
curl_close($curl); // 关闭CURL会话
return $tmpInfo; // 返回数据
}
function vget($url){ // 模拟获取内容函数
$curl = curl_init(); // 启动一个CURL会话
curl_setopt($curl, CURLOPT_URL, $url); // 要访问的地址
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0); // 对认证证书来源的检查
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1); // 从证书中检查SSL加密算法是否存在
curl_setopt($curl, CURLOPT_USERAGENT, $_SERVER['HTTP_USER_AGENT']); // 模拟用户使用的浏览器
curl_setopt($curl, CURLOPT_FOLLOWLOCATION, 1); // 使用自动跳转
curl_setopt($curl, CURLOPT_AUTOREFERER, 1); // 自动设置Referer
curl_setopt($curl, CURLOPT_HTTPGET, 1); // 发送一个常规的Post请求
curl_setopt($curl, CURLOPT_COOKIEFILE, $GLOBALS['cookie_file']); // 读取上面所储存的Cookie信息
curl_setopt($curl, CURLOPT_TIMEOUT, 30); // 设置超时限制防止死循环
curl_setopt($curl, CURLOPT_HEADER, 0); // 显示返回的Header区域内容
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 获取的信息以文件流的形式返回
$tmpInfo = curl_exec($curl); // 执行操作
if (curl_errno($curl)) {
echo 'Errno'.curl_error($curl);
}
curl_close($curl); // 关闭CURL会话
return $tmpInfo; // 返回数据
}
function vpost($url,$data){ // 模拟提交数据函数
$curl = curl_init(); // 启动一个CURL会话
curl_setopt($curl, CURLOPT_URL, $url); // 要访问的地址
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0); // 对认证证书来源的检查
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1); // 从证书中检查SSL加密算法是否存在
curl_setopt($curl, CURLOPT_USERAGENT, $_SERVER['HTTP_USER_AGENT']); // 模拟用户使用的浏览器
curl_setopt($curl, CURLOPT_FOLLOWLOCATION, 1); // 使用自动跳转
curl_setopt($curl, CURLOPT_AUTOREFERER, 1); // 自动设置Referer
curl_setopt($curl, CURLOPT_POST, 1); // 发送一个常规的Post请求
curl_setopt($curl, CURLOPT_POSTFIELDS, $data); // Post提交的数据包
curl_setopt($curl, CURLOPT_COOKIEFILE, $GLOBALS['cookie_file']); // 读取上面所储存的Cookie信息
curl_setopt($curl, CURLOPT_TIMEOUT, 30); // 设置超时限制防止死循环
curl_setopt($curl, CURLOPT_HEADER, 0); // 显示返回的Header区域内容
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 获取的信息以文件流的形式返回
$tmpInfo = curl_exec($curl); // 执行操作
if (curl_errno($curl)) {
echo 'Errno'.curl_error($curl);
}
curl_close($curl); // 关键CURL会话
return $tmpInfo; // 返回数据
}
function delcookie($cookie_file){ // 删除Cookie函数
@unlink($cookie_file); // 执行删除
}
function readcookies( $file)
{
$result = null;
$fp = fopen( $file, "r" );
if($fp)
{
while ( !feof( $fp ) )
{
$buffer = fgets( $fp, 4096 );
$result = $buffer;
//$tmp = @split( "/t", $buffer );
//$result[@trim( $tmp[5] )] = @trim( $tmp[6] );
}
fclose($fp);
}
return $result;
}
$url = 'http://w.mail.qq.com/cgi-bin/loginpage?f=xhtml';
if(!file_exists($cookie_file)) { // 检测Cookie是否存在
$str = vget($url); // 获取提交后台
preg_match("/action=\"([^\"]*?)\"/isU",$str,$hash); // 提取登录随机值
print_r($hash[1]);
vlogin($hash[1],'&f=xhtml&uin=你的qq号&aliastype=@qq.com&pwd=qq号密码&mss=1'); // 登录获取Cookie
}
else
{
vget("http://w30.mail.qq.com/cgi-bin/today?sid=ggQq2H-cUHdDdHs0z6rT6vN8,4,z-yTNgDwU&first=1");
echo '生成了cookie';
}
?>
一个缓存接口demo,可以兼容redis和memcache,redis的高级特性没有添加进去,大家可以自行添加。
<?php
/**
* 工厂方法模式
* -------------
* @author zhangqian
* @version v1.0
*/
//缓存接口
interface cache {
public function init($conf);
public function setVal($key , $val);
public function getVal($key);
public function delVal($key);
public function autoIncreament($key);
}
//mem
class mymemCache implements cache {
//mymem连接
public function init($conf)
{
echo '初始化mymem';
}
public function setVal($key , $val)
{
echo 'mem设置值';
}
public function getVal($key)
{
echo 'mem获取值';
}
public function delVal($key)
{
echo 'mem删除值';
}
public function autoIncreament($key)
{
echo 'mem自增';
}
}
//redis
class redisCache implements cache {
//redis连接
public function init($arr)
{
echo '初始化redis';
}
public function setVal($key , $val)
{
echo 'redis设置值';
}
public function getVal($key)
{
echo 'redis获取值';
}
public function delVal($key)
{
echo 'redis删除值';
}
public function autoIncreament($key)
{
echo 'redis自增';
}
}
class cacheFactory
{
private static $obj;
private static $type;
private static $conf;
private static $allowtype = array('mymem','redis');
private static function getConfig()
{
//include_once('config.php');加载配置文件 获取缓存的类型 及缓存的配置参数
global $_SC;
self::$type = $_SC['cachetype'];//做空值的判断
self::$conf = $_SC['cacheconf'];//做空值的判断
}
//外部调用创建缓存对象
public static function CreateOperation(){
self::getConfig();
try
{
$error = '未知的缓存类型';
if(in_array(self::$type , self::$allowtype))
{
$type = self::$type.'Cache';
self::$obj = new $type;
self::$obj->init(self::$conf);//连接
}
else
throw new Exception($error);
}
catch (Exception $e) {
echo 'Caught exception: ', $e->getMessage(), "\n";
exit;
}
return self::$obj;
}
}
$_SC = array();
$_SC['cachetype'] = 'redis';//mymem
$_SC['cacheconf'] = array();
$cache = cacheFactory::CreateOperation();
$cache->setVal('a',1);
echo '<br />';
$a = $cache->getVal('a');
echo '<br />';
$cache->delVal('a');
?>