php实现压缩多个CSS与JS文件的方法


Posted in PHP onNovember 11, 2014

本文实例讲述了php实现压缩多个CSS与JS文件的方法。分享给大家供大家参考。具体实现方法如下:

1. 压缩css

<?php    

header('Content-type: text/css');    

ob_start("compress");    

function compress($buffer) {    

    /* remove comments */    

    $buffer = preg_replace('!/\*[^*]*\*+([^/][^*]*\*+)*/!', '', $buffer);    

    /* remove tabs, spaces, newlines, etc. */    

    $buffer = str_replace(array("\r\n", "\r", "\n", "\t", '  ', '    ', '    '), '', $buffer);    

    return $buffer;    

}      

    

/* your css files */    

include('galleria.css');    

include('articles.css');    

    

ob_end_flush();

使用方法如下:
<link href="compress.php" rel="stylesheet" type="text/css" /><span id="tester">test</span>

2. 压缩js,利用jsmin类:

本实例源自:http://code.google.com/p/minify/

header('Content-type: text/javascript');    

require 'jsmin.php';    

echo JSMin::minify(file_get_contents('common.js') . file_get_contents('common2.js'));

其中jsmin.php文件如下:

<?php

/**

 * jsmin.php - PHP implementation of Douglas Crockford's JSMin.

 *

 * This is pretty much a direct port of jsmin.c to PHP with just a few

 * PHP-specific performance tweaks. Also, whereas jsmin.c reads from stdin and

 * outputs to stdout, this library accepts a string as input and returns another

 * string as output.

 *

 * PHP 5 or higher is required.

 *

 * Permission is hereby granted to use this version of the library under the

 * same terms as jsmin.c, which has the following license:

 *

 * --

 * Copyright (c) 2002 Douglas Crockford  (www.crockford.com)

 *

 * Permission is hereby granted, free of charge, to any person obtaining a copy of

 * this software and associated documentation files (the "Software"), to deal in

 * the Software without restriction, including without limitation the rights to

 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies

 * of the Software, and to permit persons to whom the Software is furnished to do

 * so, subject to the following conditions:

 *

 * The above copyright notice and this permission notice shall be included in all

 * copies or substantial portions of the Software.

 *

 * The Software shall be used for Good, not Evil.

 *

 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR

 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,

 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE

 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER

 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,

 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE

 * SOFTWARE.

 * --

 *

 * @package JSMin

 * @author Ryan Grove <ryan@wonko.com>

 * @copyright 2002 Douglas Crockford <douglas@crockford.com> (jsmin.c)

 * @copyright 2008 Ryan Grove <ryan@wonko.com> (PHP port)

 * @copyright 2012 Adam Goforth <aag@adamgoforth.com> (Updates)

 * @license http://opensource.org/licenses/mit-license.php MIT License

 * @version 1.1.2 (2012-05-01)

 * @link https://github.com/rgrove/jsmin-php

 */

class JSMin {

  const ORD_LF            = 10;

  const ORD_SPACE         = 32;

  const ACTION_KEEP_A     = 1;

  const ACTION_DELETE_A   = 2;

  const ACTION_DELETE_A_B = 3;

  protected $a           = '';

  protected $b           = '';

  protected $input       = '';

  protected $inputIndex  = 0;

  protected $inputLength = 0;

  protected $lookAhead   = null;

  protected $output      = '';

  // -- Public Static Methods --------------------------------------------------

  /**

   * Minify Javascript

   *

   * @uses __construct()

   * @uses min()

   * @param string $js Javascript to be minified

   * @return string

   */

  public static function minify($js) {

    $jsmin = new JSMin($js);

    return $jsmin->min();

  }

  // -- Public Instance Methods ------------------------------------------------

  /**

   * Constructor

   *

   * @param string $input Javascript to be minified

   */

  public function __construct($input) {

    $this->input       = str_replace("\r\n", "\n", $input);

    $this->inputLength = strlen($this->input);

  }

  // -- Protected Instance Methods ---------------------------------------------

  /**

   * Action -- do something! What to do is determined by the $command argument.

   *

   * action treats a string as a single character. Wow!

   * action recognizes a regular expression if it is preceded by ( or , or =.

   *

   * @uses next()

   * @uses get()

   * @throws JSMinException If parser errors are found:

   *         - Unterminated string literal

   *         - Unterminated regular expression set in regex literal

   *         - Unterminated regular expression literal

   * @param int $command One of class constants:

   *      ACTION_KEEP_A      Output A. Copy B to A. Get the next B.

   *      ACTION_DELETE_A    Copy B to A. Get the next B. (Delete A).

   *      ACTION_DELETE_A_B  Get the next B. (Delete B).

  */

  protected function action($command) {

    switch($command) {

      case self::ACTION_KEEP_A:

        $this->output .= $this->a;

      case self::ACTION_DELETE_A:

        $this->a = $this->b;

        if ($this->a === "'" || $this->a === '"') {

          for (;;) {

            $this->output .= $this->a;

            $this->a       = $this->get();

            if ($this->a === $this->b) {

              break;

            }

            if (ord($this->a) <= self::ORD_LF) {

              throw new JSMinException('Unterminated string literal.');

            }

            if ($this->a === '\\') {

              $this->output .= $this->a;

              $this->a       = $this->get();

            }

          }

        }

      case self::ACTION_DELETE_A_B:

        $this->b = $this->next();

        if ($this->b === '/' && (

            $this->a === '(' || $this->a === ',' || $this->a === '=' ||

            $this->a === ':' || $this->a === '[' || $this->a === '!' ||

            $this->a === '&' || $this->a === '|' || $this->a === '?' ||

            $this->a === '{' || $this->a === '}' || $this->a === ';' ||

            $this->a === "\n" )) {

          $this->output .= $this->a . $this->b;

          for (;;) {

            $this->a = $this->get();

            if ($this->a === '[') {

              /*

                inside a regex [...] set, which MAY contain a '/' itself. Example: mootools Form.Validator near line 460:

                  return Form.Validator.getValidator('IsEmpty').test(element) || (/^(?:[a-z0-9!#$%&'*+/=?^_`{|}~-]\.?){0,63}[a-z0-9!#$%&'*+/=?^_`{|}~-]@(?:(?:[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?\.)*[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?|\[(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\])$/i).test(element.get('value'));

              */

              for (;;) {

                $this->output .= $this->a;

                $this->a = $this->get();

                if ($this->a === ']') {

                    break;

                } elseif ($this->a === '\\') {

                  $this->output .= $this->a;

                  $this->a       = $this->get();

                } elseif (ord($this->a) <= self::ORD_LF) {

                  throw new JSMinException('Unterminated regular expression set in regex literal.');

                }

              }

            } elseif ($this->a === '/') {

              break;

            } elseif ($this->a === '\\') {

              $this->output .= $this->a;

              $this->a       = $this->get();

            } elseif (ord($this->a) <= self::ORD_LF) {

              throw new JSMinException('Unterminated regular expression literal.');

            }

            $this->output .= $this->a;

          }

          $this->b = $this->next();

        }

    }

  }

  /**

   * Get next char. Convert ctrl char to space.

   *

   * @return string|null

   */

  protected function get() {

    $c = $this->lookAhead;

    $this->lookAhead = null;

    if ($c === null) {

      if ($this->inputIndex < $this->inputLength) {

        $c = substr($this->input, $this->inputIndex, 1);

        $this->inputIndex += 1;

      } else {

        $c = null;

      }

    }

    if ($c === "\r") {

      return "\n";

    }

    if ($c === null || $c === "\n" || ord($c) >= self::ORD_SPACE) {

      return $c;

    }

    return ' ';

  }

  /**

   * Is $c a letter, digit, underscore, dollar sign, or non-ASCII character.

   *

   * @return bool

   */

  protected function isAlphaNum($c) {

    return ord($c) > 126 || $c === '\\' || preg_match('/^[\w\$]$/', $c) === 1;

  }

  /**

   * Perform minification, return result

   *

   * @uses action()

   * @uses isAlphaNum()

   * @uses get()

   * @uses peek()

   * @return string

   */

  protected function min() {

    if (0 == strncmp($this->peek(), "\xef", 1)) {

        $this->get();

        $this->get();

        $this->get();

    } 

    $this->a = "\n";

    $this->action(self::ACTION_DELETE_A_B);

    while ($this->a !== null) {

      switch ($this->a) {

        case ' ':

          if ($this->isAlphaNum($this->b)) {

            $this->action(self::ACTION_KEEP_A);

          } else {

            $this->action(self::ACTION_DELETE_A);

          }

          break;

        case "\n":

          switch ($this->b) {

            case '{':

            case '[':

            case '(':

            case '+':

            case '-':

            case '!':

            case '~':

              $this->action(self::ACTION_KEEP_A);

              break;

            case ' ':

              $this->action(self::ACTION_DELETE_A_B);

              break;

            default:

              if ($this->isAlphaNum($this->b)) {

                $this->action(self::ACTION_KEEP_A);

              }

              else {

                $this->action(self::ACTION_DELETE_A);

              }

          }

          break;

        default:

          switch ($this->b) {

            case ' ':

              if ($this->isAlphaNum($this->a)) {

                $this->action(self::ACTION_KEEP_A);

                break;

              }

              $this->action(self::ACTION_DELETE_A_B);

              break;

            case "\n":

              switch ($this->a) {

                case '}':

                case ']':

                case ')':

                case '+':

                case '-':

                case '"':

                case "'":

                  $this->action(self::ACTION_KEEP_A);

                  break;

                default:

                  if ($this->isAlphaNum($this->a)) {

                    $this->action(self::ACTION_KEEP_A);

                  }

                  else {

                    $this->action(self::ACTION_DELETE_A_B);

                  }

              }

              break;

            default:

              $this->action(self::ACTION_KEEP_A);

              break;

          }

      }

    }

    return $this->output;

  }

  /**

   * Get the next character, skipping over comments. peek() is used to see

   *  if a '/' is followed by a '/' or '*'.

   *

   * @uses get()

   * @uses peek()

   * @throws JSMinException On unterminated comment.

   * @return string

   */

  protected function next() {

    $c = $this->get();

    if ($c === '/') {

      switch($this->peek()) {

        case '/':

          for (;;) {

            $c = $this->get();

            if (ord($c) <= self::ORD_LF) {

              return $c;

            }

          }

        case '*':

          $this->get();

          for (;;) {

            switch($this->get()) {

              case '*':

                if ($this->peek() === '/') {

                  $this->get();

                  return ' ';

                }

                break;

              case null:

                throw new JSMinException('Unterminated comment.');

            }

          }

        default:

          return $c;

      }

    }

    return $c;

  }

  /**

   * Get next char. If is ctrl character, translate to a space or newline.

   *

   * @uses get()

   * @return string|null

   */

  protected function peek() {

    $this->lookAhead = $this->get();

    return $this->lookAhead;

  }

}

// -- Exceptions ---------------------------------------------------------------

class JSMinException extends Exception {}

?>

希望本文所述对大家的php程序设计有所帮助。

PHP 相关文章推荐
php基础知识:类与对象(3) 构造函数和析构函数
Dec 13 PHP
php+mysql写的简单留言本实例代码
Jul 25 PHP
PHP运行时强制显示出错信息的代码
Apr 20 PHP
解析php file_exists无效的解决办法
Jun 26 PHP
浅析PHP Socket技术
Aug 02 PHP
php判断数组中是否存在指定键(key)的方法
Mar 17 PHP
php实现的Curl封装类Curl.class.php用法实例分析
Sep 25 PHP
PHP实现的自定义数组排序函数与排序类示例
Nov 18 PHP
yii2-GridView在开发中常用的功能及技巧总结
Jan 07 PHP
thinkphp5 URL和路由的功能详解与实例
Dec 26 PHP
laravel5.6 框架邮件队列database驱动简单demo示例
Jan 26 PHP
PHP设计模式(九)外观模式Facade实例详解【结构型】
May 02 PHP
详谈PHP文件目录基础操作
Nov 11 #PHP
浅谈PHP解析URL函数parse_url和parse_str
Nov 11 #PHP
php 魔术方法详解
Nov 11 #PHP
php多个文件及图片上传实例详解
Nov 10 #PHP
PHP文件上传判断file是否己选择上传文件的方法
Nov 10 #PHP
php数组操作之键名比较与差集、交集赋值的方法
Nov 10 #PHP
php json转换成数组形式代码分享
Nov 10 #PHP
You might like
提问的智慧(2)
2006/10/09 PHP
php 多个submit提交表单 处理方法
2009/07/07 PHP
解析PHP中ob_start()函数的用法
2013/06/24 PHP
PHP 中使用explode()函数切割字符串为数组的示例
2017/05/06 PHP
php插件Xajax使用方法详解
2017/08/31 PHP
PHP操作Redis常用技巧总结
2018/04/24 PHP
JS获取IUSR_机器名和IWAM_机器名帐号的密码
2006/12/06 Javascript
用JAVASCRIPT如何给&amp;lt;textarea&amp;gt;&amp;lt;/textarea&amp;gt;赋值
2007/04/20 Javascript
修改jQuery.Autocomplete插件 支持中文输入法 避免TAB、ENTER键失效、导致表单提交
2009/10/11 Javascript
点击弹出层外区域关闭弹出层jquery特效示例
2013/08/25 Javascript
js实现幻灯片效果(基于jquery插件)
2013/11/05 Javascript
jquery.ui.draggable中文文档(原文翻译)
2013/11/15 Javascript
PHP和NodeJs开发的应用如何共用Session
2015/04/16 NodeJs
nodejs简单实现中英文翻译
2015/05/04 NodeJs
jQuery插件pagewalkthrough实现引导页效果
2015/07/05 Javascript
javascript:void(0)是什么意思及href=#与href=javascriptvoid(0)的区别
2015/11/13 Javascript
分享js粘帖屏幕截图到web页面插件screenshot-paste
2020/08/21 Javascript
探讨:JavaScript ECAMScript5 新特性之get/set访问器
2016/05/05 Javascript
纯JS实现表单验证实例
2016/12/24 Javascript
JS实现图片预加载之无序预加载功能代码
2017/05/12 Javascript
vue滚动轴插件better-scroll使用详解
2017/10/17 Javascript
javascript数据类型中的一些小知识点(推荐)
2019/04/18 Javascript
jQuery操作事件完整实例分析
2020/01/10 jQuery
python进阶教程之动态类型详解
2014/08/30 Python
Python HTTP客户端自定义Cookie实现实例
2017/04/28 Python
python3通过subprocess模块调用脚本并和脚本交互的操作
2020/12/05 Python
class类在python中获取金融数据的实例方法
2020/12/10 Python
详解CSS透明opacity和IE各版本透明度滤镜filter的最准确用法
2016/12/20 HTML / CSS
法国奢华女性时尚配饰网上商店:Monnier Frères
2016/08/27 全球购物
一份软件工程师的面试试题
2016/02/01 面试题
中专毕业个人的自荐信格式
2013/09/21 职场文书
家居设计专业个人自荐信范文
2013/11/26 职场文书
平安工地建设方案
2014/05/06 职场文书
安全环保演讲稿
2014/08/28 职场文书
2014第二批党的群众路线教育实践活动对照检查材料思想汇报
2014/09/18 职场文书
2015年小学开学寄语
2015/02/27 职场文书