zoukankan      html  css  js  c++  java
  • ecmall分析笔记

    入口文件

    <?php
    
    define('ROOT_PATH', dirname(__FILE__));
    include(ROOT_PATH . '/eccore/ecmall.php');
    
    /* 定义配置信息 */
    ecm_define(ROOT_PATH . '/data/config.inc.php');
    
    /* 启动ECMall */
    ECMall::startup(array(
        'default_app'   =>  'default',
        'default_act'   =>  'index',
        'app_root'      =>  ROOT_PATH . '/app',
        'external_libs' =>  array(
            ROOT_PATH . '/includes/global.lib.php',
            ROOT_PATH . '/includes/libraries/time.lib.php',
            ROOT_PATH . '/includes/ecapp.base.php',
            ROOT_PATH . '/includes/plugin.base.php',
            ROOT_PATH . '/app/frontend.base.php',
            ROOT_PATH . '/includes/subdomain.inc.php',
        ),
    ));
    ?>

    定义原始路径信息启动程序。

    2、

    View Code
    <?php
    
    /**
     *    ECMall框架核心文件,包含最基础的类与函数
     *    Streamlining comes from Sparrow PHP @ Garbin
     *
     *    @author    Garbin
     */
    
    /*---------------------以下是系统常量-----------------------*/
    /* 记录程序启动时间 */
    define('START_TIME', ecm_microtime());
    
    /* 判断请求方式 */
    define('IS_POST', (strtoupper($_SERVER['REQUEST_METHOD']) == 'POST'));
    
    /* 判断请求方式 */
    define('IN_ECM', true);
    
    /* 定义PHP_SELF常量 */
    define('PHP_SELF',  htmlentities(isset($_SERVER['PHP_SELF']) ? $_SERVER['PHP_SELF'] : $_SERVER['SCRIPT_NAME']));
    
    /* 当前ECMall程序版本 */
    define('VERSION', '2.3.0');
    
    /* 当前ECMall程序Release */
    define('RELEASE', '20120918');
    
    /*---------------------以下是PHP在不同版本,不同服务器上的兼容处理-----------------------*/
    
    /* 在部分IIS上会没有REQUEST_URI变量 */
    $query_string = isset($_SERVER['argv'][0]) ? $_SERVER['argv'][0] : $_SERVER['QUERY_STRING'];
    if (!isset($_SERVER['REQUEST_URI']))
    {
        $_SERVER['REQUEST_URI'] = PHP_SELF . '?' . $query_string;
    }
    else
    {
        if (strpos($_SERVER['REQUEST_URI'], '?') === false && $query_string)
        {
            $_SERVER['REQUEST_URI'] .= '?' . $query_string;
        }
    }
    
    /*---------------------以下是系统底层基础类及工具-----------------------*/
    class ECMall
    {
        /* 启动 */
        function startup($config = array())
        {
            /* 加载初始化文件 */
            require(ROOT_PATH . '/eccore/controller/app.base.php');     //基础控制器类
            require(ROOT_PATH . '/eccore/model/model.base.php');   //模型基础类
    
            if (!empty($config['external_libs']))
            {
                foreach ($config['external_libs'] as $lib)
                {
                    require($lib);
                }
            }
            /* 数据过滤 */
            if (!get_magic_quotes_gpc())
            {
                $_GET   = addslashes_deep($_GET);
                $_POST  = addslashes_deep($_POST);
                $_COOKIE= addslashes_deep($_COOKIE);
            }
    
            /* 请求转发 */
            $default_app = $config['default_app'] ? $config['default_app'] : 'default';
            $default_act = $config['default_act'] ? $config['default_act'] : 'index';
    
            $app    = isset($_REQUEST['app']) ? preg_replace('/(\W+)/', '', $_REQUEST['app']) : $default_app;
            $act    = isset($_REQUEST['act']) ? trim($_REQUEST['act']) : $default_act;
            $app_file = $config['app_root'] . "/{$app}.app.php";
            if (!is_file($app_file))
            {
                exit('Missing controller');
            }
    
            require($app_file);
            define('APP', $app);
            define('ACT', $act);
            $app_class_name = ucfirst($app) . 'App';
    
            /* 实例化控制器 */
            $app     = new $app_class_name();
            c($app);
            $app->do_action($act);        //转发至对应的Action
            $app->destruct();
        }
    }
    
    /**
     *    所有类的基础类
     *
     *    @author    Garbin
     *    @usage    none
     */
    class Object
    {
        var $_errors = array();
        var $_errnum = 0;
        function __construct()
        {
            $this->Object();
        }
        function Object()
        {
            #TODO
        }
        /**
         *    触发错误
         *
         *    @author    Garbin
         *    @param     string $errmsg
         *    @return    void
         */
        function _error($msg, $obj = '')
        {
            if(is_array($msg))
            {
                $this->_errors = array_merge($this->_errors, $msg);
                $this->_errnum += count($msg);
            }
            else
            {
                $this->_errors[] = compact('msg', 'obj');
                $this->_errnum++;
            }
        }
    
        /**
         *    检查是否存在错误
         *
         *    @author    Garbin
         *    @return    int
         */
        function has_error()
        {
            return $this->_errnum;
        }
    
        /**
         *    获取错误列表
         *
         *    @author    Garbin
         *    @return    array
         */
        function get_error()
        {
            return $this->_errors;
        }
    }
    
    /**
     *    语言项管理
     *
     *    @author    Garbin
     *    @param    none
     *    @return    void
     */
    class Lang
    {
        /**
         *    获取指定键的语言项
         *
         *    @author    Garbin
         *    @param     none
         *    @return    mixed
         */
        function &get($key = '')
        {
            if (Lang::_valid_key($key) == false)
            {
                return $key;
            }
            $vkey = $key ? strtokey("{$key}", '$GLOBALS[\'__ECLANG__\']') : '$GLOBALS[\'__ECLANG__\']';
            $tmp = eval('if(isset(' . $vkey . '))return ' . $vkey . ';else{ return $key; }');
    
            return $tmp;
        }
    
        /**
         * 验证key的有效性
         *
         * @author Hyber
         * @param string $key
         * @return bool
         */
        function _valid_key($key)
        {
            if (strpos($key, ' ') !== false)
            {
                return false;
            }
            #todo 暂时只判断是否含有空格
            return true;
        }
    
    
        /**
         *    加载指定的语言项至全局语言数据中
         *
         *    @author    Garbin
         *    @param    none
         *    @return    void
         */
        function load($lang_file)
        {
            static $loaded = array();
            $old_lang = $new_lang = array();
            $file_md5 = md5($lang_file);
            if (!isset($loaded[$file_md5]))
            {
                $new_lang = Lang::fetch($lang_file);
                $loaded[$file_md5] = $lang_file;
            }
            else
            {
                return;
            }
            $old_lang =& $GLOBALS['__ECLANG__'];
            if (is_array($old_lang))
            {
                $new_lang = array_merge($old_lang, $new_lang);
            }
    
            $GLOBALS['__ECLANG__'] = $new_lang;
        }
    
        /**
         *    获取一个语言文件的内容
         *
         *    @author    Garbin
         *    @param     string $lang_file
         *    @return    array
         */
        function fetch($lang_file)
        {
            return is_file($lang_file) ? include($lang_file) : array();
        }
    }
    function lang_file($file)
    {
        return ROOT_PATH . '/languages/' . LANG . '/' . $file . '.lang.php';
    }
    
    /**
     *    配置管理器
     *
     *    @author    Garbin
     *    @usage    none
     */
    class Conf
    {
        /**
         *    加载配置项
         *
         *    @author    Garbin
         *    @param     mixed $conf
         *    @return    bool
         */
        function load($conf)
        {
            $old_conf = isset($GLOBALS['ECMALL_CONFIG']) ? $GLOBALS['ECMALL_CONFIG'] : array();
            if (is_string($conf))
            {
                $conf = include($conf);
            }
            if (is_array($old_conf))
            {
                $GLOBALS['ECMALL_CONFIG'] = array_merge($old_conf, $conf);
            }
            else
            {
                $GLOBALS['ECMALL_CONFIG'] = $conf;
            }
        }
        /**
         *    获取配置项
         *
         *    @author    Garbin
         *    @param     string $k
         *    @return    mixed
         */
        function get($key = '')
        {
            $vkey = $key ? strtokey("{$key}", '$GLOBALS[\'ECMALL_CONFIG\']') : '$GLOBALS[\'ECMALL_CONFIG\']';
    
            return eval('if(isset(' . $vkey . '))return ' . $vkey . ';else{ return null; }');
        }
    }
    
    /**
     *    获取视图链接
     *
     *    @author    Garbin
     *    @param     string $engine
     *    @return    object
     */
    function &v($is_new = false, $engine = 'default')
    {
        include_once(ROOT_PATH . '/eccore/view/template.php');
        if ($is_new)
        {
            return new ecsTemplate();
        }
        else
        {
            static $v = null;
            if ($v === null)
            {
                switch ($engine)
                {
                    case 'default':
                        $v = new ecsTemplate();
                    break;
                }
            }
    
            return $v;
        }
    }
    
    /**
     *  获取一个模型
     *
     *  @author Garbin
     *  @param  string $model_name
     *  @param  array  $params
     *  @param  book   $is_new
     *  @return object
     */
    function &m($model_name, $params = array(), $is_new = false)
    {
        static $models = array();
        $model_hash = md5($model_name . var_export($params, true));
        if ($is_new || !isset($models[$model_hash]))
        {
            $model_file = ROOT_PATH . '/includes/models/' . $model_name . '.model.php';
            if (!is_file($model_file))
            {
                /* 不存在该文件,则无法获取模型 */
                return false;
            }
            include_once($model_file);
            $model_name = ucfirst($model_name) . 'Model';
            if ($is_new)
            {
                return new $model_name($params, db());
            }
            $models[$model_hash] = new $model_name($params, db());
        }
    
        return $models[$model_hash];
    }
    
    /**
     * 获取一个业务模型
     *
     * @param string $model_name
     * @param array $params
     * @param bool $is_new
     * @return object
     */
    function &bm($model_name, $params = array(), $is_new = false)
    {
        static $models = array();
        $model_hash = md5($model_name . var_export($params, true));
        if ($is_new || !isset($models[$model_hash]))
        {
            $model_file = ROOT_PATH . '/includes/models/' . $model_name . '.model.php';
            if (!is_file($model_file))
            {
                /* 不存在该文件,则无法获取模型 */
                return false;
            }
            include_once($model_file);
            $model_name = ucfirst($model_name) . 'BModel';
            if ($is_new)
            {
                return new $model_name($params, db());
            }
            $models[$model_hash] = new $model_name($params, db());
        }
    
        return $models[$model_hash];
    }
    
    /**
     *    获取当前控制器实例
     *
     *    @author    Garbin
     *    @return    void
     */
    function c(&$app)
    {
        $GLOBALS['ECMALL_APP'] =& $app;
    }
    
    /**
     *    获取当前控制器
     *
     *    @author    Garbin
     *    @return    Object
     */
    function &cc()
    {
        return $GLOBALS['ECMALL_APP'];
    }
    
    /**
     *    导入一个类
     *
     *    @author    Garbin
     *    @return    void
     */
    function import()
    {
        $c = func_get_args();
        if (empty($c))
        {
            return;
        }
        array_walk($c, create_function('$item, $key', 'include_once(ROOT_PATH . \'/includes/libraries/\' . $item . \'.php\');'));
    }
    
    /**
     *    将default.abc类的字符串转为$default['abc']
     *
     *    @author    Garbin
     *    @param     string $str
     *    @return    string
     */
    function strtokey($str, $owner = '')
    {
        if (!$str)
        {
            return '';
        }
        if ($owner)
        {
            return $owner . '[\'' . str_replace('.', '\'][\'', $str) . '\']';
        }
        else
        {
            $parts = explode('.', $str);
            $owner = '$' . $parts[0];
            unset($parts[0]);
            return strtokey(implode('.', $parts), $owner);
        }
    }
    /**
     *    跟踪调试
     *
     *    @author    Garbin
     *    @param     mixed $var
     *    @return    void
     */
    function trace($var)
    {
        static $i = 0;
        echo $i, '.', var_dump($var), '<br />';
        $i++;
    }
    
    /**
     *  rdump的别名
     *
     *  @author Garbin
     *  @param  any
     *  @return void
     */
    function dump($arr)
    {
        $args = func_get_args();
        call_user_func_array('rdump', $args);
    }
    
    /**
     *  格式化显示出变量
     *
     *  @author Garbin
     *  @param  any
     *  @return void
     */
    function rdump($arr)
    {
        echo '<pre>';
        array_walk(func_get_args(), create_function('&$item, $key', 'print_r($item);'));
        echo '</pre>';
        exit();
    }
    
    /**
     *  格式化并显示出变量类型
     *
     *  @author Garbin
     *  @param  any
     *  @return void
     */
    function vdump($arr)
    {
        echo '<pre>';
        array_walk(func_get_args(), create_function('&$item, $key', 'var_dump($item);'));
        echo '</pre>';
        exit();
    }
    
    /**
     * 创建MySQL数据库对象实例
     *
     * @author  wj
     * @return  object
     */
    function &db()
    {
        include_once(ROOT_PATH . '/eccore/model/mysql.php');
        static $db = null;
        if ($db === null)
        {
            $cfg = parse_url(DB_CONFIG);
    
            if ($cfg['scheme'] == 'mysql')
            {
                if (empty($cfg['pass']))
                {
                    $cfg['pass'] = '';
                }
                else
                {
                    $cfg['pass'] = urldecode($cfg['pass']);
                }
                $cfg ['user'] = urldecode($cfg['user']);
    
                if (empty($cfg['path']))
                {
                    trigger_error('Invalid database name.', E_USER_ERROR);
                }
                else
                {
                    $cfg['path'] = str_replace('/', '', $cfg['path']);
                }
    
                $charset = (CHARSET == 'utf-8') ? 'utf8' : CHARSET;
                $db = new cls_mysql();
                $db->cache_dir = ROOT_PATH. '/temp/query_caches/';
                $db->connect($cfg['host']. ':' .$cfg['port'], $cfg['user'],
                    $cfg['pass'], $cfg['path'], $charset);
            }
            else
            {
                trigger_error('Unkown database type.', E_USER_ERROR);
            }
        }
    
        return $db;
    }
    
    /**
     * 获得当前的域名
     *
     * @return  string
     */
    function get_domain()
    {
        /* 协议 */
        $protocol = (isset($_SERVER['HTTPS']) && (strtolower($_SERVER['HTTPS']) != 'off')) ? 'https://' : 'http://';
    
        /* 域名或IP地址 */
        if (isset($_SERVER['HTTP_X_FORWARDED_HOST']))
        {
            $host = $_SERVER['HTTP_X_FORWARDED_HOST'];
        }
        elseif (isset($_SERVER['HTTP_HOST']))
        {
            $host = $_SERVER['HTTP_HOST'];
        }
        else
        {
            /* 端口 */
            if (isset($_SERVER['SERVER_PORT']))
            {
                $port = ':' . $_SERVER['SERVER_PORT'];
    
                if ((':80' == $port && 'http://' == $protocol) || (':443' == $port && 'https://' == $protocol))
                {
                    $port = '';
                }
            }
            else
            {
                $port = '';
            }
    
            if (isset($_SERVER['SERVER_NAME']))
            {
                $host = $_SERVER['SERVER_NAME'] . $port;
            }
            elseif (isset($_SERVER['SERVER_ADDR']))
            {
                $host = $_SERVER['SERVER_ADDR'] . $port;
            }
        }
    
        return $protocol . $host;
    }
    
    /**
     * 获得网站的URL地址
     *
     * @return  string
     */
    function site_url()
    {
        return get_domain() . substr(PHP_SELF, 0, strrpos(PHP_SELF, '/'));
    }
    
    
    /**
     * 截取UTF-8编码下字符串的函数
     *
     * @param   string      $str        被截取的字符串
     * @param   int         $length     截取的长度
     * @param   bool        $append     是否附加省略号
     *
     * @return  string
     */
    function sub_str($string, $length = 0, $append = true)
    {
    
        if(strlen($string) <= $length) {
            return $string;
        }
    
        $string = str_replace(array('&amp;', '&quot;', '&lt;', '&gt;'), array('&', '"', '<', '>'), $string);
    
        $strcut = '';
    
        if(strtolower(CHARSET) == 'utf-8') {
            $n = $tn = $noc = 0;
            while($n < strlen($string)) {
    
                $t = ord($string[$n]);
                if($t == 9 || $t == 10 || (32 <= $t && $t <= 126)) {
                    $tn = 1; $n++; $noc++;
                } elseif(194 <= $t && $t <= 223) {
                    $tn = 2; $n += 2; $noc += 2;
                } elseif(224 <= $t && $t < 239) {
                    $tn = 3; $n += 3; $noc += 2;
                } elseif(240 <= $t && $t <= 247) {
                    $tn = 4; $n += 4; $noc += 2;
                } elseif(248 <= $t && $t <= 251) {
                    $tn = 5; $n += 5; $noc += 2;
                } elseif($t == 252 || $t == 253) {
                    $tn = 6; $n += 6; $noc += 2;
                } else {
                    $n++;
                }
    
                if($noc >= $length) {
                    break;
                }
    
            }
            if($noc > $length) {
                $n -= $tn;
            }
    
            $strcut = substr($string, 0, $n);
    
        } else {
            for($i = 0; $i < $length; $i++) {
                $strcut .= ord($string[$i]) > 127 ? $string[$i].$string[++$i] : $string[$i];
            }
        }
    
        $strcut = str_replace(array('&', '"', '<', '>'), array('&amp;', '&quot;', '&lt;', '&gt;'), $strcut);
    
        if ($append && $string != $strcut)
        {
            $strcut .= '...';
        }
    
        return $strcut;
    
    }
    
    /**
     * 获得用户的真实IP地址
     *
     * @return  string
     */
    function real_ip()
    {
        static $realip = NULL;
    
        if ($realip !== NULL)
        {
            return $realip;
        }
    
        if (isset($_SERVER))
        {
            if (isset($_SERVER['HTTP_X_FORWARDED_FOR']))
            {
                $arr = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
    
                /* 取X-Forwarded-For中第一个非unknown的有效IP字符串 */
                foreach ($arr AS $ip)
                {
                    $ip = trim($ip);
    
                    if ($ip != 'unknown')
                    {
                        $realip = $ip;
    
                        break;
                    }
                }
            }
            elseif (isset($_SERVER['HTTP_CLIENT_IP']))
            {
                $realip = $_SERVER['HTTP_CLIENT_IP'];
            }
            else
            {
                if (isset($_SERVER['REMOTE_ADDR']))
                {
                    $realip = $_SERVER['REMOTE_ADDR'];
                }
                else
                {
                    $realip = '0.0.0.0';
                }
            }
        }
        else
        {
            if (getenv('HTTP_X_FORWARDED_FOR'))
            {
                $realip = getenv('HTTP_X_FORWARDED_FOR');
            }
            elseif (getenv('HTTP_CLIENT_IP'))
            {
                $realip = getenv('HTTP_CLIENT_IP');
            }
            else
            {
                $realip = getenv('REMOTE_ADDR');
            }
        }
    
        preg_match("/[\d\.]{7,15}/", $realip, $onlineip);
        $realip = !empty($onlineip[0]) ? $onlineip[0] : '0.0.0.0';
    
        return $realip;
    }
    
    /**
     * 验证输入的邮件地址是否合法
     *
     * @param   string      $email      需要验证的邮件地址
     *
     * @return bool
     */
    function is_email($user_email)
    {
        $chars = "/^([a-z0-9+_]|\\-|\\.)+@(([a-z0-9_]|\\-)+\\.)+[a-z]{2,5}\$/i";
        if (strpos($user_email, '@') !== false && strpos($user_email, '.') !== false)
        {
            if (preg_match($chars, $user_email))
            {
                return true;
            }
            else
            {
                return false;
            }
        }
        else
        {
            return false;
        }
    }
    
    /**
     * 检查是否为一个合法的时间格式
     *
     * @param   string  $time
     * @return  void
     */
    function is_time($time)
    {
        $pattern = '/[\d]{4}-[\d]{1,2}-[\d]{1,2}\s[\d]{1,2}:[\d]{1,2}:[\d]{1,2}/';
    
        return preg_match($pattern, $time);
    }
    
    /**
     * 获得服务器上的 GD 版本
     *
     * @return      int         可能的值为0,1,2
     */
    function gd_version()
    {
        import('image.lib');
    
        return imageProcessor::gd_version();
    }
    
    /**
     * 递归方式的对变量中的特殊字符进行转义
     *
     * @access  public
     * @param   mix     $value
     *
     * @return  mix
     */
    function addslashes_deep($value)
    {
        if (empty($value))
        {
            return $value;
        }
        else
        {
            return is_array($value) ? array_map('addslashes_deep', $value) : addslashes($value);
        }
    }
    
    /**
     * 将对象成员变量或者数组的特殊字符进行转义
     *
     * @access   public
     * @param    mix        $obj      对象或者数组
     * @author   Xuan Yan
     *
     * @return   mix                  对象或者数组
     */
    function addslashes_deep_obj($obj)
    {
        if (is_object($obj) == true)
        {
            foreach ($obj AS $key => $val)
            {
                if ( ($val) == true)
                {
                    $obj->$key = addslashes_deep_obj($val);
                }
                else
                {
                    $obj->$key = addslashes_deep($val);
                }
            }
        }
        else
        {
            $obj = addslashes_deep($obj);
        }
    
        return $obj;
    }
    
    /**
     * 递归方式的对变量中的特殊字符去除转义
     *
     * @access  public
     * @param   mix     $value
     *
     * @return  mix
     */
    function stripslashes_deep($value)
    {
        if (empty($value))
        {
            return $value;
        }
        else
        {
            return is_array($value) ? array_map('stripslashes_deep', $value) : stripslashes($value);
        }
    }
    /**
     *  将一个字串中含有全角的数字字符、字母、空格或'%+-()'字符转换为相应半角字符
     *
     * @access  public
     * @param   string       $str         待转换字串
     *
     * @return  string       $str         处理后字串
     */
    function make_semiangle($str)
    {
        $arr = array('0' => '0', '1' => '1', '2' => '2', '3' => '3', '4' => '4',
                     '5' => '5', '6' => '6', '7' => '7', '8' => '8', '9' => '9',
                     'A' => 'A', 'B' => 'B', 'C' => 'C', 'D' => 'D', 'E' => 'E',
                     'F' => 'F', 'G' => 'G', 'H' => 'H', 'I' => 'I', 'J' => 'J',
                     'K' => 'K', 'L' => 'L', 'M' => 'M', 'N' => 'N', 'O' => 'O',
                     'P' => 'P', 'Q' => 'Q', 'R' => 'R', 'S' => 'S', 'T' => 'T',
                     'U' => 'U', 'V' => 'V', 'W' => 'W', 'X' => 'X', 'Y' => 'Y',
                     'Z' => 'Z', 'a' => 'a', 'b' => 'b', 'c' => 'c', 'd' => 'd',
                     'e' => 'e', 'f' => 'f', 'g' => 'g', 'h' => 'h', 'i' => 'i',
                     'j' => 'j', 'k' => 'k', 'l' => 'l', 'm' => 'm', 'n' => 'n',
                     'o' => 'o', 'p' => 'p', 'q' => 'q', 'r' => 'r', 's' => 's',
                     't' => 't', 'u' => 'u', 'v' => 'v', 'w' => 'w', 'x' => 'x',
                     'y' => 'y', 'z' => 'z',
                     '(' => '(', ')' => ')', '[' => '[', ']' => ']', '【' => '[',
                     '】' => ']', '〖' => '[', '〗' => ']', '「' => '[', '」' => ']',
                     '『' => '[', '』' => ']', '{' => '{', '}' => '}', '《' => '<',
                     '》' => '>',
                     '%' => '%', '+' => '+', '—' => '-', '-' => '-', '~' => '-',
                     ':' => ':', '。' => '.', '、' => ',', ',' => '.', '、' => '.',
                     ';' => ',', '?' => '?', '!' => '!', '…' => '-', '‖' => '|',
                     '"' => '"', ''' => '`', '`' => '`', '|' => '|', '〃' => '"',
                     ' ' => ' ');
    
        return strtr($str, $arr);
    }
    
    /**
     * 格式化费用:可以输入数字或百分比的地方
     *
     * @param   string      $fee    输入的费用
     */
    function format_fee($fee)
    {
        $fee = make_semiangle($fee);
        if (strpos($fee, '%') === false)
        {
            return floatval($fee);
        }
        else
        {
            return floatval($fee) . '%';
        }
    }
    
    /**
     * 根据总金额和费率计算费用
     *
     * @param     float    $amount    总金额
     * @param     string    $rate    费率(可以是固定费率,也可以是百分比)
     * @param     string    $type    类型:s 保价费 p 支付手续费 i 发票税费
     * @return     float    费用
     */
    function compute_fee($amount, $rate, $type)
    {
        $amount = floatval($amount);
        if (strpos($rate, '%') === false)
        {
            return round(floatval($rate), 2);
        }
        else
        {
            $rate = floatval($rate) / 100;
            if ($type == 's')
            {
                return round($amount * $rate, 2);
            }
            elseif($type == 'p')
            {
                return round($amount * $rate / (1 - $rate), 2);
            }
            else
            {
                return round($amount * $rate, 2);
            }
        }
    }
    
    /**
     * 获取服务器的ip
     *
     * @access      public
     *
     * @return string
     **/
    function real_server_ip()
    {
        static $serverip = NULL;
    
        if ($serverip !== NULL)
        {
            return $serverip;
        }
    
        if (isset($_SERVER))
        {
            if (isset($_SERVER['SERVER_ADDR']))
            {
                $serverip = $_SERVER['SERVER_ADDR'];
            }
            else
            {
                $serverip = '0.0.0.0';
            }
        }
        else
        {
            $serverip = getenv('SERVER_ADDR');
        }
    
        return $serverip;
    }
    /**
     * 获得用户操作系统的换行符
     *
     * @access  public
     * @return  string
     */
    function get_crlf()
    {
    /* LF (Line Feed, 0x0A, \N) 和 CR(Carriage Return, 0x0D, \R) */
        if (stristr($_SERVER['HTTP_USER_AGENT'], 'Win'))
        {
            $the_crlf = "\r\n";
        }
        elseif (stristr($_SERVER['HTTP_USER_AGENT'], 'Mac'))
        {
            $the_crlf = "\r"; // for old MAC OS
        }
        else
        {
            $the_crlf = "\n";
        }
    
        return $the_crlf;
    }
    
    /**
     * 编码转换函数
     *
     * @author  wj
     * @param string $source_lang       待转换编码
     * @param string $target_lang         转换后编码
     * @param string $source_string      需要转换编码的字串
     * @return string
     */
    function ecm_iconv($source_lang, $target_lang, $source_string = '')
    {
        static $chs = NULL;
    
        /* 如果字符串为空或者字符串不需要转换,直接返回 */
        if ($source_lang == $target_lang || $source_string == '' || preg_match("/[\x80-\xFF]+/", $source_string) == 0)
        {
            return $source_string;
        }
    
        if ($chs === NULL)
        {
            import('iconv.lib');
            $chs = new Chinese(ROOT_PATH . '/');
        }
    
        return strtolower($target_lang) == 'utf-8' ? addslashes(stripslashes($chs->Convert($source_lang, $target_lang, $source_string))) : $chs->Convert($source_lang, $target_lang, $source_string);
    }
    
    function ecm_geoip($ip)
    {
        static $fp = NULL, $offset = array(), $index = NULL;
    
        $ip    = gethostbyname($ip);
        $ipdot = explode('.', $ip);
        $ip    = pack('N', ip2long($ip));
    
        $ipdot[0] = (int)$ipdot[0];
        $ipdot[1] = (int)$ipdot[1];
        if ($ipdot[0] == 10 || $ipdot[0] == 127 || ($ipdot[0] == 192 && $ipdot[1] == 168) || ($ipdot[0] == 172 && ($ipdot[1] >= 16 && $ipdot[1] <= 31)))
        {
            return 'LAN';
        }
    
        if ($fp === NULL)
        {
            $fp     = fopen(ROOT_PATH . 'includes/codetable/ipdata.dat', 'rb');
            if ($fp === false)
            {
                return 'Invalid IP data file';
            }
            $offset = unpack('Nlen', fread($fp, 4));
            if ($offset['len'] < 4)
            {
                return 'Invalid IP data file';
            }
            $index  = fread($fp, $offset['len'] - 4);
        }
    
        $length = $offset['len'] - 1028;
        $start  = unpack('Vlen', $index[$ipdot[0] * 4] . $index[$ipdot[0] * 4 + 1] . $index[$ipdot[0] * 4 + 2] . $index[$ipdot[0] * 4 + 3]);
        for ($start = $start['len'] * 8 + 1024; $start < $length; $start += 8)
        {
            if ($index{$start} . $index{$start + 1} . $index{$start + 2} . $index{$start + 3} >= $ip)
            {
                $index_offset = unpack('Vlen', $index{$start + 4} . $index{$start + 5} . $index{$start + 6} . "\x0");
                $index_length = unpack('Clen', $index{$start + 7});
                break;
            }
        }
    
        fseek($fp, $offset['len'] + $index_offset['len'] - 1024);
        $area = fread($fp, $index_length['len']);
    
        fclose($fp);
        $fp = NULL;
    
        return $area;
    }
    
    function ecm_json_encode($value)
    {
        if (CHARSET == 'utf-8' && function_exists('json_encode'))
        {
            return json_encode($value);
        }
    
        $props = '';
        if (is_object($value))
        {
            foreach (get_object_vars($value) as $name => $propValue)
            {
                if (isset($propValue))
                {
                    $props .= $props ? ','.ecm_json_encode($name)  : ecm_json_encode($name);
                    $props .= ':' . ecm_json_encode($propValue);
                }
            }
            return '{' . $props . '}';
        }
        elseif (is_array($value))
        {
            $keys = array_keys($value);
            if (!empty($value) && !empty($value) && ($keys[0] != '0' || $keys != range(0, count($value)-1)))
            {
                foreach ($value as $key => $val)
                {
                    $key = (string) $key;
                    $props .= $props ? ','.ecm_json_encode($key)  : ecm_json_encode($key);
                    $props .= ':' . ecm_json_encode($val);
                }
                return '{' . $props . '}';
            }
            else
            {
                $length = count($value);
                for ($i = 0; $i < $length; $i++)
                {
                    $props .= ($props != '') ? ','.ecm_json_encode($value[$i])  : ecm_json_encode($value[$i]);
                }
                return '[' . $props . ']';
            }
        }
        elseif (is_string($value))
        {
            //$value = stripslashes($value);
            $replace  = array('\\' => '\\\\', "\n" => '\n', "\t" => '\t', '/' => '\/',
                            "\r" => '\r', "\b" => '\b', "\f" => '\f',
                            '"' => '\"', chr(0x08) => '\b', chr(0x0C) => '\f'
                            );
            $value  = strtr($value, $replace);
            if (CHARSET == 'big5' && $value{strlen($value)-1} == '\\')
            {
                $value  = substr($value,0,strlen($value)-1);
            }
            return '"' . $value . '"';
        }
        elseif (is_numeric($value))
        {
            return $value;
        }
        elseif (is_bool($value))
        {
            return $value ? 'true' : 'false';
        }
        elseif (empty($value))
        {
            return '""';
        }
        else
        {
            return $value;
        }
    }
    
    function ecm_json_decode($value, $type = 0)
    {
        if (CHARSET == 'utf-8' && function_exists('json_decode'))
        {
            return empty($type) ? json_decode($value) : get_object_vars_deep(json_decode($value));
        }
    
        if (!class_exists('JSON'))
        {
            import('json.lib');
        }
        $json = new JSON();
        return $json->decode($value, $type);
    }
    
    /**
     * 返回由对象属性组成的关联数组
     *
     * @access   pubilc
     * @param    obj    $obj
     *
     * @return   array
     */
    function get_object_vars_deep($obj)
    {
        if(is_object($obj))
        {
            $obj = get_object_vars($obj);
        }
        if(is_array($obj))
        {
            foreach ($obj as $key => $value)
            {
                $obj[$key] = get_object_vars_deep($value);
            }
        }
        return $obj;
    }
    
    function file_ext($filename)
    {
        return trim(substr(strrchr($filename, '.'), 1, 10));
    }
    
    /**
     * 创建像这样的查询: "IN('a','b')";
     *
     * @access   public
     * @param    mix      $item_list      列表数组或字符串,如果为字符串时,字符串只接受数字串
     * @param    string   $field_name     字段名称
     * @author   wj
     *
     * @return   void
     */
    function db_create_in($item_list, $field_name = '')
    {
        if (empty($item_list))
        {
            return $field_name . " IN ('') ";
        }
        else
        {
            if (!is_array($item_list))
            {
                $item_list = explode(',', $item_list);
                foreach ($item_list as $k=>$v)
                {
                    $item_list[$k] = intval($v);
                }
            }
    
            $item_list = array_unique($item_list);
            $item_list_tmp = '';
            foreach ($item_list AS $item)
            {
                if ($item !== '')
                {
                    $item_list_tmp .= $item_list_tmp ? ",'$item'" : "'$item'";
                }
            }
            if (empty($item_list_tmp))
            {
                return $field_name . " IN ('') ";
            }
            else
            {
                return $field_name . ' IN (' . $item_list_tmp . ') ';
            }
        }
    }
    
    /**
     * 创建目录(如果该目录的上级目录不存在,会先创建上级目录)
     * 依赖于 ROOT_PATH 常量,且只能创建 ROOT_PATH 目录下的目录
     * 目录分隔符必须是 / 不能是 \
     *
     * @param   string  $absolute_path  绝对路径
     * @param   int     $mode           目录权限
     * @return  bool
     */
    function ecm_mkdir($absolute_path, $mode = 0777)
    {
        if (is_dir($absolute_path))
        {
            return true;
        }
    
        $root_path      = ROOT_PATH;
        $relative_path  = str_replace($root_path, '', $absolute_path);
        $each_path      = explode('/', $relative_path);
        $cur_path       = $root_path; // 当前循环处理的路径
        foreach ($each_path as $path)
        {
            if ($path)
            {
                $cur_path = $cur_path . '/' . $path;
                if (!is_dir($cur_path))
                {
                    if (@mkdir($cur_path, $mode))
                    {
                        fclose(fopen($cur_path . '/index.htm', 'w'));
                    }
                    else
                    {
                        return false;
                    }
                }
            }
        }
    
        return true;
    }
    
    /**
     * 删除目录,不支持目录中带 ..
     *
     * @param string $dir
     *
     * @return boolen
     */
    function ecm_rmdir($dir)
    {
        $dir = str_replace(array('..', "\n", "\r"), array('', '', ''), $dir);
        $ret_val = false;
        if (is_dir($dir))
        {
            $d = @dir($dir);
            if($d)
            {
                while (false !== ($entry = $d->read()))
                {
                   if($entry!='.' && $entry!='..')
                   {
                       $entry = $dir.'/'.$entry;
                       if(is_dir($entry))
                       {
                           ecm_rmdir($entry);
                       }
                       else
                       {
                           @unlink($entry);
                       }
                   }
                }
                $d->close();
                $ret_val = rmdir($dir);
             }
        }
        else
        {
            $ret_val = unlink($dir);
        }
    
        return $ret_val;
    }
    
    function price_format($price, $price_format = NULL)
    {
        if (empty($price)) $price = '0.00';
        $price = number_format($price, 2);
    
        if ($price_format === NULL)
        {
            $price_format = Conf::get('price_format');
        }
    
        return sprintf($price_format, $price);
    }
    
    /**
     *  设置COOKIE
     *
     *  @access public
     *  @param  string $key     要设置的COOKIE键名
     *  @param  string $value   键名对应的值
     *  @param  int    $expire  过期时间
     *  @return void
     */
    function ecm_setcookie($key, $value, $expire = 0, $cookie_path=COOKIE_PATH, $cookie_domain=COOKIE_DOMAIN)
    {
        setcookie($key, $value, $expire, $cookie_path, $cookie_domain);
    }
    
    /**
     *  获取COOKIE的值
     *
     *  @access public
     *  @param  string $key    为空时将返回所有COOKIE
     *  @return mixed
     */
    function ecm_getcookie($key = '')
    {
        return isset($_COOKIE[$key]) ? $_COOKIE[$key] : 0;
    }
    
    
    /**
     * 对数组转码
     *
     * @param   string  $func
     * @param   array   $params
     *
     * @return  mixed
     */
    function ecm_iconv_deep($source_lang, $target_lang, $value)
    {
        if (empty($value))
        {
            return $value;
        }
        else
        {
            if (is_array($value))
            {
                foreach ($value as $k=>$v)
                {
                    $value[$k] = ecm_iconv_deep($source_lang, $target_lang, $v);
                }
                return $value;
            }
            elseif (is_string($value))
            {
                return ecm_iconv($source_lang, $target_lang, $value);
            }
            else
            {
                return $value;
            }
        }
    }
    
    /**
     *  fopen封装函数
     *
     *  @author wj
     *  @param string $url
     *  @param int    $limit
     *  @param string $post
     *  @param string $cookie
     *  @param boolen $bysocket
     *  @param string $ip
     *  @param int    $timeout
     *  @param boolen $block
     *  @return responseText
     */
    function ecm_fopen($url, $limit = 500000, $post = '', $cookie = '', $bysocket = false, $ip = '', $timeout = 15, $block = true)
    {
        $return = '';
        $matches = parse_url($url);
        $host = $matches['host'];
        $path = $matches['path'] ? $matches['path'].($matches['query'] ? '?'.$matches['query'] : '') : '/';
        $port = !empty($matches['port']) ? $matches['port'] : 80;
    
        if($post)
        {
            $out = "POST $path HTTP/1.0\r\n";
            $out .= "Accept: */*\r\n";
            //$out .= "Referer: $boardurl\r\n";
            $out .= "Accept-Language: zh-cn\r\n";
            $out .= "Content-Type: application/x-www-form-urlencoded\r\n";
            $out .= "User-Agent: $_SERVER[HTTP_USER_AGENT]\r\n";
            $out .= "Host: $host\r\n";
            $out .= 'Content-Length: '.strlen($post)."\r\n";
            $out .= "Connection: Close\r\n";
            $out .= "Cache-Control: no-cache\r\n";
            $out .= "Cookie: $cookie\r\n\r\n";
            $out .= $post;
        }
        else
        {
            $out = "GET $path HTTP/1.0\r\n";
            $out .= "Accept: */*\r\n";
            //$out .= "Referer: $boardurl\r\n";
            $out .= "Accept-Language: zh-cn\r\n";
            $out .= "User-Agent: $_SERVER[HTTP_USER_AGENT]\r\n";
            $out .= "Host: $host\r\n";
            $out .= "Connection: Close\r\n";
            $out .= "Cookie: $cookie\r\n\r\n";
        }
        $fp = @fsockopen(($ip ? $ip : $host), $port, $errno, $errstr, $timeout);
        if(!$fp)
        {
            return '';
        }
        else
        {
            stream_set_blocking($fp, $block);
            stream_set_timeout($fp, $timeout);
            @fwrite($fp, $out);
            $status = stream_get_meta_data($fp);
            if(!$status['timed_out'])
            {
                while (!feof($fp))
                {
                    if(($header = @fgets($fp)) && ($header == "\r\n" ||  $header == "\n"))
                    {
                        break;
                    }
                }
    
                $stop = false;
                while(!feof($fp) && !$stop)
                {
                    $data = fread($fp, ($limit == 0 || $limit > 8192 ? 8192 : $limit));
                    $return .= $data;
                    if($limit)
                    {
                        $limit -= strlen($data);
                        $stop = $limit <= 0;
                    }
                }
            }
            @fclose($fp);
            return $return;
        }
    }
    
    /**
     * 危险 HTML代码过滤器
     *
     * @param   string  $html   需要过滤的html代码
     *
     * @return  string
     */
    function html_filter($html)
    {
        $filter = array(
            "/\s/",
            "/<(\/?)(script|i?frame|style|html|body|title|link|\?|\%)([^>]*?)>/isU",//object|meta|
            "/(<[^>]*)on[a-zA-Z]\s*=([^>]*>)/isU",
            );
    
        $replace = array(
            " ",
            "&lt;\\1\\2\\3&gt;",
            "\\1\\2",
            );
    
        $str = preg_replace($filter,$replace,$html);
        return $str;
    }
    
    /**
     * 清理系统所有编译文件,缓存文件、模板结构数据
     *
     * @author  wj
     * @param   void
     *
     * @return  void
     */
    function clean_cache()
    {
        /*清理缓存*/
        $cache_dirs = array(
            ROOT_PATH . '/temp/caches',
            ROOT_PATH . '/temp/compiled/mall/admin',
            ROOT_PATH . '/temp/compiled/mall/',
            ROOT_PATH . '/temp/compiled/store/admin',
            ROOT_PATH . '/temp/compiled/store',
            ROOT_PATH . '/temp/js',
            ROOT_PATH . '/temp/query_caches',
            ROOT_PATH . '/temp/tag_caches',
            ROOT_PATH . '/temp/style',
        );
    
        foreach ($cache_dirs as $dir)
        {
            $d = dir($dir);
            if ($d)
            {
                while (false !== ($entry = $d->read()))
                {
                    if($entry!='.' && $entry!='..' && $entry != '.svn' && $entry != 'admin' && $entry != 'index.html')
                    {
                       ecm_rmdir($dir . '/'. $entry);
                    }
                }
                $d->close();
            }
        }
    
        /*主分类缓存数据*/
        if (is_file(ROOT_PATH . '/temp/query_caches/cache_category.php'))
        {
            unlink(ROOT_PATH . '/temp/query_caches/cache_category.php');
        }
    
        /*清除一个周前图片缓存并回收多余目录*/
        $expiry_time = strtotime('-1 week');
        $path = ROOT_PATH . '/temp/thumb';
        $d = dir($path);
        if ($d)
        {
            while(false !== ($entry = $d->read()))
            {
                if ($entry!='.' && $entry!= '..' && $entry != '.svn' && is_dir(($dir = ($path . '/' . $entry))))
                {
                    $sd = dir($dir);
                    if ($sd)
                    {
                        $left_dir_count = 0;
                        while(false !== ($entry = $sd->read()))
                        {
                            if ($entry!='.' && $entry!= '..' && is_dir(($subdir = ($dir . '/' . $entry))))
                            {
                                $fsd = dir($subdir);
                                $left_file_count = 0;
                                while (false !== ($entry= $fsd->read()))
                                {
                                    if ($entry!='.' && $entry!='..' && $entry != 'index.htm' && is_file(($file =$subdir . '/' . $entry)))
                                    {
                                        if (filemtime($file) < $expiry_time)
                                        {
                                            unlink($file);
                                        }
                                        else
                                        {
                                            $left_file_count ++;
                                        }
                                    }
                                }
                                $fsd->close();
                                if ($left_file_count == 0)
                                {
                                    //清除空目录
                                    ecm_rmdir($subdir);
                                }
                                else
                                {
                                    $left_dir_count ++;
                                }
                            }
                        }
                        $sd->close();
                        if ($left_dir_count == 0) ecm_rmdir($dir);
                    }
                }
            }
            $d->close();
        }
    
    }
    
    /**
     * 如果系统不存在file_put_contents函数则声明该函数
     *
     * @author  wj
     * @param   string  $file
     * @param   mix     $data
     * @return  int
     */
    if (!function_exists('file_put_contents'))
    {
        define('FILE_APPEND', 'FILE_APPEND');
        if (!defined('LOCK_EX'))
        {
            define('LOCK_EX', 'LOCK_EX');
        }
    
        function file_put_contents($file, $data, $flags = '')
        {
            $contents = (is_array($data)) ? implode('', $data) : $data;
    
            $mode = ($flags == 'FILE_APPEND') ? 'ab+' : 'wb';
    
            if (($fp = @fopen($file, $mode)) === false)
            {
                return false;
            }
            else
            {
                $bytes = fwrite($fp, $contents);
                fclose($fp);
    
                return $bytes;
            }
        }
    }
    
    /**
     * 去除字符串右侧可能出现的乱码
     *
     * @author  wj
     * @param   string      $str        字符串
     *
     *
     * @return  string
     */
    function trim_right($str)
    {
        $len = strlen($str);
        /* 为空或单个字符直接返回 */
        if ($len == 0 || ord($str{$len-1}) < 127)
        {
            return $str;
        }
        /* 有前导字符的直接把前导字符去掉 */
        if (ord($str{$len-1}) >= 192)
        {
           return substr($str, 0, $len-1);
        }
        /* 有非独立的字符,先把非独立字符去掉,再验证非独立的字符是不是一个完整的字,不是连原来前导字符也截取掉 */
        $r_len = strlen(rtrim($str, "\x80..\xBF"));
        if ($r_len == 0 || ord($str{$r_len-1}) < 127)
        {
            return sub_str($str, 0, $r_len);
        }
    
        $as_num = ord(~$str{$r_len -1});
        if ($as_num > (1<<(6 + $r_len - $len)))
        {
            return $str;
        }
        else
        {
            return substr($str, 0, $r_len-1);
        }
    }
    
    /**
     * 通过该函数运行函数可以抑制错误
     *
     * @author  weberliu
     * @param   string      $fun        要屏蔽错误的函数名
     * @return  mix         函数执行结果
     */
    function _at($fun)
    {
        $arg = func_get_args();
        unset($arg[0]);
        $ret_val = @call_user_func_array($fun, $arg);
    
        return $ret_val;
    }
    
    /**
     * 调用外部函数
     *
     * @author  weberliu
     * @param   string  $func
     * @param   array   $params
     *
     * @return  mixed
     */
    function outer_call($func, $params=null)
    {
        restore_error_handler();
    
        $res = call_user_func_array($func, $params);
    
        set_error_handler('exception_handler');
    
        return $res;
    }
    
    function reset_error_handler()
    {
        set_error_handler('exception_handler');
    }
    
    /**
     * 返回是否是通过浏览器访问的页面
     *
     * @author wj
     * @param  void
     * @return boolen
     */
    function is_from_browser()
    {
        static $ret_val = null;
        if ($ret_val === null)
        {
            $ret_val = false;
            $ua = isset($_SERVER['HTTP_USER_AGENT']) ? strtolower($_SERVER['HTTP_USER_AGENT']) : '';
            if ($ua)
            {
                if ((strpos($ua, 'mozilla') !== false) && ((strpos($ua, 'msie') !== false) || (strpos($ua, 'gecko') !== false)))
                {
                    $ret_val = true;
                }
                elseif (strpos($ua, 'opera'))
                {
                    $ret_val = true;
                }
            }
        }
        return $ret_val;
    }
    
    /**
     *    从文件或数组中定义常量
     *
     *    @author    Garbin
     *    @param     mixed $source
     *    @return    void
     */
    function ecm_define($source)
    {
        if (is_string($source))
        {
            /* 导入数组 */
            $source = include($source);
        }
        if (!is_array($source))
        {
            /* 不是数组,无法定义 */
            return false;
        }
        foreach ($source as $key => $value)
        {
            if (is_string($value) || is_numeric($value) || is_bool($value) || is_null($value))
            {
                /* 如果是可被定义的,则定义 */
                define(strtoupper($key), $value);
            }
        }
    }
    
    /**
     *    获取当前时间的微秒数
     *
     *    @author    Garbin
     *    @return    float
     */
    function ecm_microtime()
    {
        if (PHP_VERSION >= 5.0)
        {
            return microtime(true);
        }
        else
        {
            list($usec, $sec) = explode(" ", microtime());
    
            return ((float)$usec + (float)$sec);
        }
    }
    
    ?>

      这个文件分为几个部分

      1、定义系统常量和server常量

      2、定义ecmall类,加载从入口文件中配置的类库和控制器以及模型类库

       3、请求转发,如果定义了默认的路由,以此转发,如果没有定义默认控制器为 default 和index

      4、从参数app和action中获取控制器和动作

      5、更具配置参数中的路径和 $_REQUEST中获取的controller name 加载控制器,如果没有找到抛出异常

      6、加载控制器         define('APP', $app);        define('ACT', $act); 定义两个常量

      7、转发到相关的action do_action()

    另外需要说明的就是在 ecmall.php中定义了系统中所有的基类

      8、使用v()方法初始化模板引擎对象_init_view()

      9、assign方法把变量带如模板

      10、display()输出。

      视图层的细节和model类的细节等下再来研究

      

    is_file

  • 相关阅读:
    最短路详解
    树状数组基础
    并查集初步
    python shell的交互模式和文本编辑模式
    基于python的selenium自动化测试环境搭建
    Nagios基本搭建
    mysql错误代码对照表较完整
    搭建phpMyAdmin
    Cent OS 7 搭建MySQL
    详解封装源码包成RPM包
  • 原文地址:https://www.cnblogs.com/linksgo2011/p/2939181.html
Copyright © 2011-2022 走看看