php实现压缩合并js的方法【附demo源码下载】


Posted in PHP onSeptember 22, 2016

本文实例讲述了php实现压缩合并js的方法。分享给大家供大家参考,具体如下:

test.php文件如下:

require_once('jsmin.php');
$files = glob("js/*.js");
$js = "";
foreach($files as $file) {
  $js .= JSMin::minify(file_get_contents($file));
}
file_put_contents("combined.js", $js);
echo "success";

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 相关文章推荐
玩转虚拟域名◎+ .
Oct 09 PHP
PHP 应用程序的安全 -- 不能违反的四条安全规则
Nov 26 PHP
探讨各种PHP字符串函数的总结分析
Jun 05 PHP
解析PHPExcel使用的常用说明以及把PHPExcel整合进CI框架的介绍
Jun 24 PHP
php调用MySQL存储过程的方法集合(推荐)
Jul 03 PHP
PHP SPL使用方法和他的威力
Nov 12 PHP
thinkphp实现发送邮件密码找回功能实例
Dec 01 PHP
ThinkPHP添加更新标签的方法
Dec 05 PHP
yii2 页面底部加载css和js的技巧
Apr 21 PHP
Zend Framework入门教程之Zend_Mail用法示例
Dec 08 PHP
PHP性能测试工具xhprof安装与使用方法详解
Apr 29 PHP
laravel 判断查询数据库返回值的例子
Oct 11 PHP
php简单压缩css样式示例
Sep 22 #PHP
php 伪造HTTP_REFERER页面URL来源的三种方法
Sep 22 #PHP
PHP mysqli_free_result()与mysqli_fetch_array()函数详解
Sep 21 #PHP
Yii2.0 Basic代码中路由链接被转义的处理方法
Sep 21 #PHP
php5.2的curl-bug 服务器被php进程卡死问题排查
Sep 19 #PHP
php支付宝在线支付接口开发教程
Sep 19 #PHP
iOS10推送通知开发教程
Sep 19 #PHP
You might like
JavaScript类库D
2010/10/24 Javascript
解析DHTML,JavaScript,DOM,BOM以及WEB标准的描述
2013/06/19 Javascript
对frameset、frame、iframe的js操作示例代码
2013/08/16 Javascript
JS控制输入框内字符串长度
2014/05/21 Javascript
javascript验证身份证号
2015/03/03 Javascript
jQuery-1.9.1源码分析系列(十)事件系统之事件包装
2015/11/20 Javascript
AngularJS使用ngOption实现下拉列表的实例代码
2016/01/23 Javascript
Angular 页面跳转时传参问题
2016/08/01 Javascript
基于jQuery和Bootstrap框架实现仿知乎前端动态列表效果
2016/11/09 Javascript
Vue 拦截器对token过期处理方法
2018/01/23 Javascript
浅谈FastClick 填坑及源码解析
2018/03/02 Javascript
Vue封装一个简单轻量的上传文件组件的示例
2018/03/21 Javascript
小程序实现列表多个批量倒计时
2021/01/29 Javascript
vue 地图可视化 maptalks 篇实例代码详解
2019/05/21 Javascript
微信小程序实现页面监听自定义组件的触发事件
2020/11/01 Javascript
详解Python中的循环语句的用法
2015/04/09 Python
深入理解Python中装饰器的用法
2016/06/28 Python
浅谈DataFrame和SparkSql取值误区
2018/06/09 Python
win7下python3.6安装配置方法图文教程
2018/07/31 Python
Django使用Jinja2模板引擎的示例代码
2019/08/09 Python
Flask框架学习笔记之使用Flask实现表单开发详解
2019/08/12 Python
用Python批量把文件复制到另一个文件夹的实现方法
2019/08/16 Python
python3.7将代码打包成exe程序并添加图标的方法
2019/10/11 Python
Django更新models数据库结构步骤
2020/04/01 Python
详解CSS3中常用的样式【基本文本和字体样式】
2020/10/20 HTML / CSS
当x.equals(y)等于true时,x.hashCode()与y.hashCode()可以不相等,这句话对不对
2015/05/02 面试题
妇科医生自荐信
2013/11/05 职场文书
中秋节礼品促销方案
2014/02/02 职场文书
妇女儿童发展规划实施方案
2014/03/16 职场文书
刊首寄语大全
2014/04/11 职场文书
二年级评语大全
2014/04/23 职场文书
监督检查工作方案
2014/05/28 职场文书
《这片土地是神圣的》教学反思
2016/02/16 职场文书
2016年第16个全民国防教育日宣传活动总结
2016/04/05 职场文书
python使用XPath解析数据爬取起点小说网数据
2021/04/22 Python
Golang解析JSON对象
2022/04/30 Golang