首页 > 编程技术 > php

php中autoload的实现例子介绍

发布时间:2016-11-25 15:36

autoload在php中其实是一个魔术方法了,我们可以指定类目录及规则可以自动加载类文件从而可以省去我们使用include来加载文件了,下面一直来看看关于autoload方法一些例子.

我们在写web应用程序时通常对每个类都建立一个 PHP 源文件。为了使用这些源文件,我们就需要在每个脚本开头写大量的的包含语句(include,require)。在 PHP 5 中,不再需要这样了。我们可__autoload()函数和spl_autoload_register函数实现实现自己的加载源文件的机制,它们会在试图使用尚未被定义的类时自动调用。通过调用这些函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类。本文的主要目标是讲述如何在扩展中用C语言实现自动加载源文件的机制,但是在这之前我们先熟悉一下在PHP脚本中实现自动加载的方法。

在脚本中实现自动加载

在 PHP 5 中我们可以定义一个 __autoload() 函数,它会在试图使用尚未被定义的类时自动调用,这样我们就可以定义一些自己的加载规则了。

<?php
function __autoload($class_name) {
    require_once $class_name . '.php';
}

$obj  = new MyClass1();
$obj2 = new MyClass2();
?>

使用spl_autoload_register我们可以一次注册多个加载函数,PHP会在试图使用尚未被定义的类时按注册顺序调用。

<?php
function autoload_services($class_name)
{
    $file = 'services/' . $class_name. '.php';
    if (file_exists($file))
    {
        require_once($file);
    }
}
function autoload_vos($class_name)
{
    $file = 'vos/' . $class_name. '.php';
    if (file_exists($file))
    {
        require_once($file);
    }
}
spl_autoload_register('autoload_services');
spl_autoload_register('autoload_vos');
?>

在php扩展中实现自动加载

最近在写一个php扩展,其中一个功能就是实现类的自动加载,其实也是通过在内核中调用spl_autoload_register函数来实现。使用zend API调用spl_autoload_register函数还是相对简单的,下面我们主要讲一下如何在内核中实现inclue/require/include_once/require_once等指令的功能。其实inclue/require/include_once/require_once等指令主要是读入文件编译并执行,下面的方法就是完成了这些操作,代码中有详细的注释。

/*
*  loader_import首先将PHP源文件编译成op_array,然后依次执行op_array中的opcode
*/
int loader_import(char *path, int len TSRMLS_DC) {
    zend_file_handle file_handle;
    zend_op_array   *op_array;
    char realpath[MAXPATHLEN];

    if (!VCWD_REALPATH(path, realpath)) {
        return 0;
    }

    file_handle.filename = path;
    file_handle.free_filename = 0;
    file_handle.type = ZEND_HANDLE_FILENAME;
    file_handle.opened_path = NULL;
    file_handle.handle.fp = NULL;
   
    //调用zend API编译源文件
    op_array = zend_compile_file(&file_handle, ZEND_INCLUDE TSRMLS_CC);

    if (op_array && file_handle.handle.stream.handle) {
        int dummy = 1;

        if (!file_handle.opened_path) {
            file_handle.opened_path = path;
        }
       
        //将源文件注册到执行期间的全局变量(EG)的include_files列表中,这样就标记了源文件已经包含过了
        zend_hash_add(&EG(included_files), file_handle.opened_path, strlen(file_handle.opened_path)+1, (void *)&dummy,
                sizeof(int), NULL);
    }
    zend_destroy_file_handle(&file_handle TSRMLS_CC);

    //开始执行op_array
    if (op_array) {
        zval *result = NULL;
        //保存原来的执行环境,包括active_op_array,opline_ptr等
        zval ** __old_return_value_pp   =  EG(return_value_ptr_ptr);
        zend_op ** __old_opline_ptr     = EG(opline_ptr);
        zend_op_array * __old_op_array  = EG(active_op_array);
        //保存环境完成后,初始化本次执行环境,替换op_array
        EG(return_value_ptr_ptr) = &result;
        EG(active_op_array)      = op_array;

#if ((PHP_MAJOR_VERSION == 5) && (PHP_MINOR_VERSION > 2)) || (PHP_MAJOR_VERSION > 5)
        if (!EG(active_symbol_table)) {
            zend_rebuild_symbol_table(TSRMLS_C);
        }
#endif
        //调用zend API执行源文件的op_array
        zend_execute(op_array TSRMLS_CC);
        //op_array执行完成后销毁,要不然就要内存泄露了,哈哈
        destroy_op_array(op_array TSRMLS_CC);
        efree(op_array);
        //通过检查执行期间的全局变量(EG)的exception是否被标记来确定是否有异常
        if (!EG(exception)) {
            if (EG(return_value_ptr_ptr) && *EG(return_value_ptr_ptr)) {
                zval_ptr_dtor(EG(return_value_ptr_ptr));
            }
        }
        //ok,执行到这里说明源文件的op_array已经执行完成了,我们要恢复原来的执行环境了
        EG(return_value_ptr_ptr) = __old_return_value_pp;
        EG(opline_ptr)           = __old_opline_ptr;
        EG(active_op_array)      = __old_op_array;

        return 1;
    }
    return 0;
}

php中DOMDocument对于xml操作我们只要是英文是没有问题了,但如果是中文字体就会有乱码问题了,下面我们就此问题给各位介绍一些解决办法吧.

PHP的DOM内部是utf8机制的。在loadHTML时,是通过检查字符中meta的charset来设置编码的。如果没有charset,就当iso8859进行处理了。而这种情况下进行saveXML时,输出来的却是utf8,所以就看到乱码了。

这么说是不是还不太理解,举个例子:

$xml = new DOMDocument();
@$xml->loadHTML('<div>我就是测试看看 - http://www.111cn.net</div>');
 
$dom = new DOMXPath($xml);
echo $dom->query('//div')->item(0)->saveXML();

打开网页执行,你会发现输出乱码了。那如何解决这个问题呢?有两种方式。

第一种:在loadHTML的时候指定编码,下面这段代码引用自php.net官方文档中的回复

$doc = new DOMDocument();
$doc->loadHTML('<?xml encoding="UTF-8">' . $html);
 
// dirty fix
foreach ($doc->childNodes as $item)
    if ($item->nodeType == XML_PI_NODE)
        $doc->removeChild($item); // remove hack
$doc->encoding = 'UTF-8'; // insert proper

第二种方法,通过iconv对输出的字符重新转换,代码如下:

echo iconv("UTF-8", "GB18030//TRANSLIT", $dom->saveXML($n) );

我们经常会碰到一些投票程序了,今天小编整理了一个简单的红蓝投票功能的实现程序了,有需要了解这种投票的朋友进入来看看吧.

本文是一篇综合知识应用类文章,需要您具备PHP、jQuery、MySQL以及html和css方面的基本知识。本文在《PHP+MySql+jQuery实现的“顶”和“踩”投票功能》一文基础上做了适当改进,共用了数据表,您可以先点击了解这篇文章。

HTML

我们需要在页面中展示红蓝双方的观点,以及对应的投票数和比例,以及用于投票交互的手型图片,本例以#red和#blue分别表示红蓝双方。.redhand和.bluehand用来做手型投票按钮,.redbar和.bluebar展示红蓝双方比例调,#red_num和#blue_num展示双方投票数。
 
<div class="vote">
    <div class="votetitle">您对Helloweba提供的文章的看法?</div>
    <div class="votetxt">非常实用<span>完全看不懂</span></div>
    <div class="red" id="red">
        <div class="redhand"></div>
        <div class="redbar" id="red_bar">
            <span></span>
            <p id="red_num"></p>
        </div>
    </div>
    <div class="blue" id="blue">
        <div class="bluehand"></div>
        <div class="bluebar" id="blue_bar">
            <span></span>
            <p id="blue_num"></p>
        </div>
    </div>
</div>

CSS

使用CSS将页面美化,加载背景图片,确定相对位置等等,你可以直接复制以下代码,在自己的项目中稍作修改即可。
 
.vote{width:288px; height:220px; margin:60px auto 20px auto;position:relative}
.votetitle{width:100%;height:62px; background:url(icon.png) no-repeat 0 30px; font-size:15px}
.red{position:absolute; left:0; top:90px; height:80px;}
.blue{position:absolute; right:0; top:90px; height:80px;}
.votetxt{line-height:24px}
.votetxt span{float:right}
.redhand{position:absolute; left:0;width:36px; height:36px; background:url(icon.png) no-repeat -1px -38px;cursor:pointer}
.bluehand{position:absolute; right:0;width:36px; height:36px; background:url(icon.png) no-repeat -41px -38px;cursor:pointer}
.grayhand{width:34px; height:34px; background:url(icon.png) no-repeat -83px -38px;cursor:pointer}
.redbar{position:absolute; left:42px; margin-top:8px;}
.bluebar{position:absolute; right:42px; margin-top:8px; }
.redbar span{display:block; height:6px; background:red; width:100%;border-radius:4px;}
.bluebar span{display:block; height:6px; background:#09f; width:100%;border-radius:4px; position:absolute; right:0}
.redbar p{line-height:20px; color:red;}
.bluebar p{line-height:20px; color:#09f; text-align:right; margin-top:6px}

jQuery

当点击手型按钮时,利用jQuery的$.getJSON()向后台php发送Ajax请求,如果请求成功,将会得到后台返回的json数据,jQuery再将json数据进行处理。以下函数:getdata(url,sid),传递了两个参数,url是请求的后台php地址,sid表示当前投票主题ID,我们在该函数中,返回的json数据有红蓝双方的投票数,以及双方比例,根据比例计算比例条的宽度,异步交互展示投票效果。
 
function getdata(url,sid){
    $.getJSON(url,{id:sid},function(data){
        if(data.success==1){
            var w = 208; //定义比例条的总宽度
            //红方投票数
            $("#red_num").html(data.red);
            $("#red").css("width",data.red_percent*100+"%");
            var red_bar_w = w*data.red_percent-10;
            //红方比例条宽度
            $("#red_bar").css("width",red_bar_w);
            //蓝方投票数
            $("#blue_num").html(data.blue);
            $("#blue").css("width",data.blue_percent*100+"%");
            var blue_bar_w = w*data.blue_percent;
            //蓝方比例条宽度
            $("#blue_bar").css("width",blue_bar_w);
        }else{
            alert(data.msg);
        }
    });
}

当页面初次加载时,即调用getdata(),然后点击给红方投票或给蓝方投票同样调用getdata(),只是传递的参数不一样。注意本例中的参数sid我们设置为1,是根据数据表中的id设定的,开发者可以根据实际项目读取准确的id。
 
$(function(){
    //获取初始数据
    getdata("vote.php",1);
    //红方投票
    $(".redhand").click(function(){
        getdata("vote.php?action=red",1);
    });
    //蓝方投票
    $(".bluehand").click(function(){
        getdata("vote.php?action=blue",1);
    });
});

PHP

前端请求了后台的vote.php,vote.php将根据接收的参数,连接数据库,调用相关函数。
 
include_once("connect.php");
 
$action = $_GET['action'];
$id = intval($_GET['id']);
$ip = get_client_ip();//获取ip
 
if($action=='red'){//红方投票
    vote(1,$id,$ip);
}elseif($action=='blue'){//蓝方投票
    vote(0,$id,$ip);
}else{//默认返回初始数据
    echo jsons($id);
}

函数vote($type,$id,$ip)用来做出投票动作,$type表示投票方,$id表示投票主题的id,$ip表示用户当前ip。首先根据用户当前IP,查询投票记录表votes_ip中是否已经存在当前ip记录,如果存在,则说明用户已投票,否则更新红方或蓝方的投票数,并将当前用户投票记录写入到votes_ip表中以防重复投票。
 
function vote($type,$id,$ip){
    $ip_sql=mysql_query("select ip from votes_ip where vid='$id' and ip='$ip'");
    $count=mysql_num_rows($ip_sql);
    if($count==0){//还没有投票
        if($type==1){//红方
            $sql = "update votes set likes=likes+1 where id=".$id;
        }else{//蓝方
            $sql = "update votes set unlikes=unlikes+1 where id=".$id;
        }
        mysql_query($sql);
        
        $sql_in = "insert into votes_ip (vid,ip) values ('$id','$ip')";
        mysql_query($sql_in);
        if(mysql_insert_id()>0){
            echo jsons($id);
        }else{
            $arr['success'] = 0;
            $arr['msg'] = '操作失败,请重试';
            echo json_encode($arr);
        }
    }else{
        $arr['success'] = 0;
        $arr['msg'] = '已经投票过了';
        echo json_encode($arr);
    }
}
函数jsons($id)通过查询当前id的投票数,计算比例并返回json数据格式供前端调用。
 
function jsons($id){
    $query = mysql_query("select * from votes where id=".$id);
    $row = mysql_fetch_array($query);
    $red = $row['likes'];
    $blue = $row['unlikes'];
    $arr['success']=1;
    $arr['red'] = $red;
    $arr['blue'] = $blue;
    $red_percent = round($red/($red+$blue),3);
    $arr['red_percent'] = $red_percent;
    $arr['blue_percent'] = 1-$red_percent;
    
    return json_encode($arr);
}

文中还涉及到获取用户真实IP的函数:get_client_ip(),点击这里可以看相关代码:

//获取用户真实IP
function getIp() {
    if (getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"), "unknown"))
        $ip = getenv("HTTP_CLIENT_IP");
    else
        if (getenv("HTTP_X_FORWARDED_FOR") && strcasecmp(getenv("HTTP_X_FORWARDED_FOR"), "unknown"))
            $ip = getenv("HTTP_X_FORWARDED_FOR");
        else
            if (getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"), "unknown"))
                $ip = getenv("REMOTE_ADDR");
            else
                if (isset ($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'], "unknown"))
                    $ip = $_SERVER['REMOTE_ADDR'];
                else
                    $ip = "unknown";
    return ($ip);
}


最后,贴上Mysql数据表,votes表用来记录红蓝双方的投票总数,votes_ip表则用来存放用户的投票IP记录。

CREATE TABLE IF NOT EXISTS `votes` (
  `id` int(10) NOT NULL AUTO_INCREMENT,
  `likes` int(10) NOT NULL DEFAULT '0',
  `unlikes` int(10) NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=MyISAM  DEFAULT CHARSET=utf8;
 
 
INSERT INTO `votes` (`id`, `likes`, `unlikes`) VALUES
(1, 30, 10);
 
CREATE TABLE IF NOT EXISTS `votes_ip` (
  `id` int(10) NOT NULL,
  `vid` int(10) NOT NULL,
  `ip` varchar(40) NOT NULL
) ENGINE=MyISAM DEFAULT CHARSET=utf8;


再次提醒,下载的demo如果运行不了,请先检查数据库连接配置是否正确

EXTRACT函数功能比较强大了可以直接把接受到的字符变成变量了,初次知道此函数是一个同事使用这种方法来接受数据了,下面我们来看关于EXTRACT的用法.

PHP extract函数的作用是:函数从数组中把变量导入到当前的符号表中。对于数组中的每个元素,键名用于变量名,键值用于变量值。用extract函数一个技巧是可以把字段变成变量,方便调用。


$info = $this->db->get_one(array('id'=>$specialid, 'disabled'=>0)); 

//将当前专题id所对应的字段作为变量导入到当前的符号表 

extract($info);

本例$info是从数据库查出的结果,以数组的形式作为变量,键名是字段名,值是字段值,使用extract函数就把键名作为变量名,字段值作为变量值了。

比如array(‘name’=>’zhangsan’,’sex’=>’man’,’id’=>3232);

那么extract($array)后,就可以直接使用$name,$sex,$id了。

再看

<?php
$size = "old size"; //注意最后size变量的值。
$a = array(
"color" => "red",
"size" => "XXL",
"price" => "53");
extract($a);
echo "color = $color<br />";
echo "size = $size<br />";
echo "price = $price<br />";
?>


结果为:

color = red
size = XXL
price = 53

通过上例发现$size的值为XXL,而不是之前的”old size”,说明默认情况下当数组中的key与已有的变量冲突时,将覆盖原来的变量。


例子

form.html

<form action="action.php" method="post">
<input type="text" name="username">
<input type="password" name="password">
<input type="submit">

在action.php中只要使用extract()函数将$_POST全局数据解开:
action.php

<?php
extract($_POST);
//相当于$username = $_POST['username'];
//$password = $_POST['password'];
?>

是不是很方便呢?

extract的作用分析

addslashes -- 使用反斜线引用字符串     

extract(addslashes($_POST)); --处理POST表单
    
把客户端<FORM METHOD="POST"...>表单中的变量名取出来。     

extract(addslashes($_GET)); --处理GET表单     


把客户端<FORM METHOD="GET"...>表单中的变量名取出来。  

我们知道php提供了6个迭代器接口了,那么这6个接口怎么样呢?有没有朋友都了解?如果各位朋友不知道的可以和小编一起来看看.

PHP预定义了6个接口介绍如下:

Traversable 遍历接口(检测一个类是否可以使用�0�2foreach�0�2进行遍历的接口)
Iterator 迭代器接口(可在内部迭代自己的外部迭代器或类的接口)
IteratorAggregate 聚合式迭代器接口(创建外部迭代器的接口)
OuterIterator 迭代器嵌套接口(将一个或多个迭代器包裹在另一个迭代器中)
RecursiveIterator 递归迭代访问接口(提供递归访问功能)
SeekableIterator 可索引迭代访问接口(实现查找功能)

1.Traversable遍历接口

呵呵!其实它不是一个在PHP中可以使用的接口,内部类才可使用,它有一个用途就是检测一个类是否可以遍历。

if($class instanceof Traversable) {
    //foreach
}

 2.Iterator迭代器接口


接口摘要:
Iterator extends Traversable 

    //返回当前索引游标指向的元素 
    abstract public mixed current(void) 
    //返回当前索引游标指向的元素的键名 
    abstract public scalar key(void) 
    //移动当前索引游标指向下一元素 
    abstract public void next(void) 
    //重置索引游标的指向第一个元素 
    abstract public void rewind(void) 
    //判断当前索引游标指向的是否是一个元素,常常在调用 rewind()或 next()使用 
    abstract public boolean valid(void) 
}

 

以上可以让一个类实现一个基本的迭代功能,如下可以看到迭代的调用顺序:


class  myIterator  implements  Iterator  {
    private  $position  =  0 ;
    private  $array  = array(
        "firstelement" ,
        "secondelement" ,
        "lastelement" ,
    );
 
    public function  __construct () {
        $this -> position  =  0 ;
    }
 
    function  rewind () {
        var_dump ( __METHOD__ );
        $this -> position  =  0 ;
    }
 
    function  current () {
        var_dump ( __METHOD__ );
        return  $this -> array [ $this -> position ];
    }
 
    function  key () {
        var_dump ( __METHOD__ );
        return  $this -> position ;
    }
 
    function  next () {
        var_dump ( __METHOD__ );
        ++ $this -> position ;
    }
 
    function  valid () {
        var_dump ( __METHOD__ );
        return isset( $this -> array [ $this -> position ]);
    }
}
 
$it  = new  myIterator ;
 
foreach( $it  as  $key  =>  $value ) {
    var_dump ( $key ,  $value );
    echo  "\n" ;
}

3.IteratorAggregate聚合式迭代器接口


接口摘要:


IteratorAggregate  extends Traversable  {
 
//获取外部迭代器
abstract public Traversable getIterator  ( void )
}

getIterator是一个Iterator或Traversable接口的类的一个实例。如下获取外部迭代器实现迭代访问。
class  myData  implements  IteratorAggregate  {
    public  $property1  =  "Public property one" ;
    public  $property2  =  "Public property two" ;
    public  $property3  =  "Public property three" ;
 
    public function  __construct () {
        $this -> property4  =  "last property" ;
    }
 
   
    public function  getIterator () {
        return new  ArrayIterator ( $this );
    }
}
 
$obj  = new  myData ;
 
foreach( $obj  as  $key  =>  $value ) {
    var_dump ( $key ,  $value );
    echo  "\n" ;
}

4.ArrayAccess数组式访问接口


接口摘要:
ArrayAccess  {
    /* 方法 */
    abstract public boolean offsetExists  ( mixed  $offset  ) //检查偏移位置是否存在
    abstract public mixed offsetGet  ( mixed  $offset  ) //获取一个偏移位置的值
    abstract public void offsetSet  ( mixed  $offset  , mixed  $value  ) //设置一个偏移位置的值
    abstract public void offsetUnset  ( mixed  $offset  ) //复位一个偏移位置的值
}

如下可像访问数组一样访问对象:
class  obj  implements  arrayaccess  {
    private  $container  = array();
    public function  __construct () {
        $this -> container  = array(
            "one"    =>  1 ,
            "two"    =>  2 ,
            "three"  =>  3 ,
        );
    }
    public function  offsetSet ( $offset ,  $value ) {
        if ( is_null ( $offset )) {
            $this -> container [] =  $value ;
        } else {
            $this -> container [ $offset ] =  $value ;
        }
    }
    public function  offsetExists ( $offset ) {
        return isset( $this -> container [ $offset ]);
    }
    public function  offsetUnset ( $offset ) {
        unset( $this -> container [ $offset ]);
    }
    public function  offsetGet ( $offset ) {
        return isset( $this -> container [ $offset ]) ?  $this -> container [ $offset ] :  null ;
    }
}
 
$obj  = new  obj ;
 
var_dump (isset( $obj [ "two" ]));
var_dump ( $obj [ "two" ]);
unset( $obj [ "two" ]);
var_dump (isset( $obj [ "two" ]));
$obj [ "two" ] =  "A value" ;
var_dump ( $obj [ "two" ]);
$obj [] =  'Append 1' ;
$obj [] =  'Append 2' ;
$obj [] =  'Append 3' ;
print_r ( $obj );

5.Serializable序列化接口
接口摘要:
Serializable  {
 
    /* 方法 */
    abstract public string serialize  ( void ) //对象的字符串表示
    abstract public mixed unserialize  ( string $serialized  ) // 构造对象
}

实现该接口的类不再支持__sleep()和__wakeup()。使用很简单,只要序列化对象时serialize方法会被调用,当反序列化时,unserialize方法被调用。

class  obj  implements  Serializable  {
    private  $data ;
    public function  __construct () {
        $this -> data  =  "My private data" ;
    }
    public function  serialize () {
        return  serialize ( $this -> data );
    }
    public function  unserialize ( $data ) {
        $this -> data  =  unserialize ( $data );
    }
    public function  getData () {
        return  $this -> data ;
    }
}
 
$obj  = new  obj ;
$ser  =  serialize ( $obj );
print_r($ser);
$newobj  =  unserialize ( $ser );
print_r($newobj);

6.Closure


接口摘要:
Closure  {
    /* 方法 */
    __construct  ( void ) //用于禁止实例化的构造函数
    public static Closure bind  ( Closure  $closure  , object $newthis  [, mixed  $newscope  = 'static'  ] ) //复制一个闭包,绑定指定的$this对象和类作用域。
    public Closure bindTo  ( object $newthis  [, mixed  $newscope  = 'static'  ] ) //复制当前闭包对象,绑定指定的$this对象和类作用域。
}

class  A  {
    private static  $sfoo  =  1 ;
    private  $ifoo  =  2 ;
}
 $cl1  = static function() {
    return  A :: $sfoo ;
};
 $cl2  = function() {
    return  $this -> ifoo ;
};
 
 $bcl1  =  Closure :: bind ( $cl1 ,  null ,  'A' );
 $bcl2  =  Closure :: bind ( $cl2 , new  A (),  'A' );
echo  $bcl1 (),  "\n" ;
echo  $bcl2 (),  "\n" ;

标签:[!--infotagslink--]

您可能感兴趣的文章: