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 相关文章推荐
聊天室php&amp;mysql(五)
Oct 09 PHP
PHP 面向对象 PHP5 中的常量
May 05 PHP
让PHP COOKIE立即生效,不用刷新就可以使用
Mar 09 PHP
php错误提示failed to open stream: HTTP request failed!的完美解决方法
Jun 06 PHP
win7计划任务定时执行PHP脚本设置图解
May 09 PHP
PHP的文件操作与算法实现的面试题示例
Aug 10 PHP
注意!PHP 7中不要做的10件事
Sep 18 PHP
PHP排序算法之冒泡排序(Bubble Sort)实现方法详解
Apr 20 PHP
phpstorm 配置xdebug的示例代码
Mar 31 PHP
laravel-admin 后台表格筛选设置默认的查询日期方法
Oct 03 PHP
关于laravel 日志写入失败问题汇总
Oct 17 PHP
基于PHP实现短信验证码发送次数限制
Jul 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
use jscript Create a SQL Server database
2007/06/16 Javascript
JQuery在光标位置插入内容的实现代码
2010/06/18 Javascript
Javascript 倒计时源代码.(时.分.秒) 详细注释版
2011/05/09 Javascript
jQuery弹出(alert)select选择的值
2013/04/21 Javascript
js带缩略图的图片轮播效果代码分享
2015/09/14 Javascript
总结javascript中的六种迭代器
2016/08/16 Javascript
Bootstrap Table表格一直加载(load)不了数据的快速解决方法
2016/09/17 Javascript
Vue2.0利用 v-model 实现组件props双向绑定的优美解决方案
2017/03/13 Javascript
webpack+vue-cli项目中引入外部非模块格式js的方法
2018/09/28 Javascript
解决JQuery的ajax函数执行失败alert函数弹框一闪而过问题
2019/04/10 jQuery
Vue实现简单计算器案例
2020/02/25 Javascript
微信小程序纯文本实现@功能
2020/04/08 Javascript
vue+axios全局添加请求头和参数操作
2020/07/24 Javascript
Python实现删除文件中含“指定内容”的行示例
2017/06/09 Python
解决python写入带有中文的字符到文件错误的问题
2019/01/31 Python
Python学习笔记之函数的定义和作用域实例详解
2019/08/13 Python
Python统计文本词汇出现次数的实例代码
2020/02/27 Python
python pandas.DataFrame.loc函数使用详解
2020/03/26 Python
python如何运行js语句
2020/09/09 Python
pycharm2020.1.2永久破解激活教程,实测有效
2020/10/29 Python
10个python爬虫入门实例(小结)
2020/11/01 Python
意大利在线药房:Saninforma
2021/02/11 全球购物
C++如何引用一个已经定义过的全局变量
2014/08/25 面试题
小学生获奖感言范文
2014/02/02 职场文书
保密工作实施方案
2014/02/24 职场文书
亲子活动总结
2014/04/26 职场文书
洗手间标语
2014/06/23 职场文书
贵阳市党的群众路线教育实践活动党(工)委领导班子整改方案
2014/10/26 职场文书
个人求职自荐信范文
2015/03/06 职场文书
2015年教师节新闻稿
2015/07/17 职场文书
驾驶员管理制度范本
2015/08/06 职场文书
同学联谊会邀请函
2019/06/24 职场文书
暑假开始了,你的暑假学习计划写好了吗?
2019/07/04 职场文书
Python线程池与GIL全局锁实现抽奖小案例
2022/04/13 Python
MySQL 表锁定 LOCK和UNLOCK TABLES的 SQL语法
2022/04/18 MySQL
python区块链实现简版工作量证明
2022/05/25 Python