php归档格式:phar文件详解(创建、使用、解包还原提取)

转载一篇,突然遇到一个冷知识,phar的东东,貌似和jar、war是一种鬼。

重点使用一下下面这个东东,就能解包出来东东了。

  1. $phar = new Phar('lib/yunke.phar', 0);  
  2. $phar->setAlias ( "yun.phar"); 

https://blog.csdn.net/u011474028/article/details/54973571

一个php应用程序往往是由多个文件构成的,如果能把他们集中为一个文件来分发和运行是很方便的,这样的列子有很多,比如在window操作系统上面的安装程序、一个jquery库等等,为了做到这点php采用了phar文档文件格式,这个概念源自java的jar,但是在设计时主要针对 PHP 的 Web 环境,与 JAR 归档不同的是Phar 归档可由 PHP 本身处理,因此不需要使用额外的工具来创建或使用,使用php脚本就能创建或提取它。phar是一个合成词,由PHP 和 Archive构成,可以看出它是php归档文件的意思。

关于phar的官网文档请见http://php.net/manual/en/book.phar.php,本文档可以看做和官网文档互为补充

phar归档文件有三种格式:tar归档、zip归档、phar归档,前两种执行需要php安装Phar 扩展支持,用的也比较少,这里主要讲phar归档格式。

phar格式归档文件可以直接执行,它的产生依赖于Phar扩展,由自己编写的php脚本产生。

Phar 扩展对 PHP 来说并不是一个新鲜的概念,在php5.3已经内建于php中,它最初使用 PHP 编写并被命名为 PHP_Archive,然后在 2005 年被添加到 PEAR 库。由于在实际中,解决这一问题的纯 PHP 解决方案非常缓慢,因此 2007 年重新编写为纯 C 语言扩展,同时添加了使用 SPL 的 ArrayAccess 对象遍历 Phar 归档的支持。自那时起,人们做了大量工作来改善 Phar 归档的性能。

Phar 扩展依赖于php流包装器,关于此可访问笔者的另外一篇帖子:

http://blog.csdn.net/u011474028/article/details/52814049

很多php应用都是以phar格式分发并运行的,著名的有依赖管理:composer、单元测试:phpunit,下面我们来看一看如何创建、运行、提取还原。

phar文件的创建:

首先在php.ini中修改phar.readonly这个选项,去掉前面的分号,并改值为off,由于安全原因该选项默认是on,如果在php.ini中是禁用的(值为0或off),那么在用户脚本中可以开启或关闭,如果在php.ini中是开启的,那么用户脚本是无法关闭的,所以这里设置为off来展示示例。

我们来建立一个项目,在服务器根目录中建立项目文件夹为project,目录内的结构如下:

[plain] view plain copy

  1. file  
  2.     -yunek.js  
  3.     -yunke.css  
  4. lib  
  5.     -lib_a.php  
  6. template  
  7.     -msg.html  
  8. index.php  
  9. Lib.php  

其中file文件夹有两个内容为空的js和css文件,仅仅演示phar可以包含多种文件格式

lib_a.php内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/10 
  6.  * Time: 9:23 
  7.  */  
  8. function show(){  
  9. echo "l am show()";  
  10. }  

msg.html内容如下:

[php] view plain copy

  1. <!DOCTYPE html>  
  2. <html lang="en">  
  3. <head>  
  4.     <meta charset="UTF-8">  
  5.     <title>phar</title>  
  6. </head>  
  7. <body>  
  8. <?=$str; ?>  
  9. </body>  
  10. </html>  

index.php内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/10 
  6.  * Time: 9:17 
  7.  */  
  8. require "lib/lib_a.php";  
  9. show();  
  10. $str = isset($_GET["str"]) ? $_GET["str"] : "hello world";  
  11. include "template/msg.html";  

Lib.php内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/10 
  6.  * Time: 9:20 
  7.  */  
  8. function yunke()  
  9. {  
  10. echo "l am yunke()";  
  11. }  

项目文件准备好了,开始创建,现在在project文件夹同级目录建立一个yunkeBuild.php,用于产生phar格式文件,内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/10 
  6.  * Time: 9:36 
  7.  */  
  8. //产生一个yunke.phar文件  
  9. $phar = new Phar('yunke.phar', 0, 'yunke.phar');  
  10. // 添加project里面的所有文件到yunke.phar归档文件  
  11. $phar->buildFromDirectory(dirname(__FILE__) . '/project');  
  12. //设置执行时的入口文件,第一个用于命令行,第二个用于浏览器访问,这里都设置为index.php  
  13. $phar->setDefaultStub('index.php', 'index.php');  

然后在浏览器中访问这个yunkeBuild.php文件,将产生一个yunke.phar文件,此时服务器根目录结构如下:

[plain] view plain copy

  1. project  
  2. yunkeBuild.php  
  3. yunke.phar  

这就是产生一个phar归档文件最简单的过程了,更多内容请看官网,这里需要注意的是如果项目不具备单一执行入口则不宜使用phar归档文件

phar归档文件的使用:

我们在服务器根目录建立一个index.php文件来演示如何使用上面创建的phar文件,内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/8 
  6.  * Time: 9:33 
  7.  */  
  8. require "yunke.phar";  
  9. require "phar://yunke.phar/Lib.php";  
  10. yunke();  

如果index.php文件中只有第一行,那么和不使用归档文件时,添加如下代码完全相同:

[php] view plain copy

  1. require "project/index.php";  

如果没有第二行,那么第三行的yunke()将提示未定义,所以可见require一个phar文件时并不是导入了里面所有的文件,而只是导入了入口执行文件而已,但在实际项目中往往在这个入口文件里导入其他需要使用的文件,在本例中入口执行文件为project/index.php

phar文件的提取还原:

我们有时候会好奇phar里面包含的文件源码,这个时候就需要将phar文件还原,如果只是看一看的话可以使用一些ide工具,比如phpstorm 10就能直接打开它,如果需要修改那么就需要提取操作了,为了演示,我们下载一个composer.phar放在服务器目录,在根目录建立一个get.php文件,内容如下:

[php] view plain copy

  1. <?php  
  2. /** 
  3.  * Created by yunke. 
  4.  * User: yunke 
  5.  * Date: 2017/2/9 
  6.  * Time: 19:02 
  7.  */  
  8. $phar = new Phar('composer.phar');  
  9. $phar->extractTo('composer'); //提取一份原项目文件  
  10. $phar->convertToData(Phar::ZIP); //另外再提取一份,和上行二选一即可  

用浏览器访问这个文件,即可提取出来,以上列子展示了两种提取方式:第二行将建立一个composer目录,并将提取出来的内容放入,第三行将产生一个composer.zip文件,解压即可得到提取还原的项目文件。

补充:

1、在部署phar文件到生产服务器时需要调整服务器的配置,避免当访问时浏览器直接下载phar文件

2、可以为归档设置别名,别名保存在归档文件中永久保存,它可以用一个简短的名字引用归档,而不管归档文件在文件系统中存储在那里,设置别名:

[php] view plain copy

  1. $phar = new Phar('lib/yunke.phar', 0);  
  2. $phar->setAlias ( "yun.phar");  

设置别名后可以如下使用:

[php] view plain copy

  1. <?php  
  2. require "lib/yunke.phar";  
  3. require "phar://yun.phar/Lib.php";  //使用别名访问归档文件  
  4. require "phar://lib/yunke.phar/Lib.php"; //当然仍然可以使用这样的方式去引用  

如果在制作phar文件时没有指定别名,也可以在存根文件里面使用Phar::mapPhar('yunke.phar');指定

3、归档文件中有一个存根文件,其实就是一段php执行代码,在制作归档时可以设置,直接执行归档文件时,其实就是执行它,所以它是启动文件;在脚本中包含归档文件时就像包含普通php文件一样包含它并运行,但直接以phar://的方式包含归档中某一个文件时不会执行存根代码, 往往在存根文件里面require包含要运行的其他文件,对存根文件的限制仅为以__HALT_COMPILER();结束,默认的存根设计是为在没有phar扩展时能够运行,它提取phar文件内容到一个临时目录再执行,不过从php5.3开始该扩展默认内置启用了

4、制作的phar文件不能被改动,因此配置文件之类的文件需要另外放置在归档文件外面

5、mapPhar函数:这个函数只应该在stub存根代码中调用,在没有设置归档别名的时候可以用来设置别名,打开一个引用映射到phar流

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏黑泽君的专栏

安装最新版本的Oracle公司的虚拟机软件 VirtualBox + 安装虚拟机 Windows XP 系统 + 安装 Oracle 11g 软件 + 出现 ERROR: ORA-12541: TNS

  VirtualBox的下载链接:https://www.virtualbox.org/wiki/Downloads

2661
来自专栏Java进阶架构师

01: springboot快速入门之三分钟构建可运行项目

Spring Boot英文中是“引导”的意思,是用来简化Spring应用的搭建到开发的过程。可以这么说。SpringBoot是一个服务于框架的框架,服务范围是简...

753
来自专栏FreeBuf

Kali Shellter 5.1:动态ShellCode注入工具 绕过安全软件

或许你还在为无法绕过反病毒软件获得Meterpreter shell而懊恼,试试Shellter 5.1吧。 首先需要获取一个正常的Windows exe可执行...

3096
来自专栏技术之路

翻译qmake文档(一) qmake指南和概述

英文文档连接: http://qt-project.org/doc/qt-5/qmake-manual.html http://qt-project.org/d...

2667
来自专栏SpringBoot 核心技术

第四十一章: 基于SpringBoot & RabbitMQ完成DirectExchange分布式消息消费

2425
来自专栏数据之美

rsync error: protocol incompatibility / mismatch

1、问题 今日在维护集群环境的时候,遇到了一个小问题,rsync 向集群中的机器传输文件的时候报错: protocol version mismatch -- ...

3206
来自专栏ImportSource

使用Spring Boot开发一个Spring Mobile程序

1. 概述 Spring Mobile是一个基于Spring Web MVC框架扩展的一个针对不同移动终端的应用开发框架。通过它我们在适配不同终端方面,就不用费...

5347
来自专栏Fish

安装IDEA和运行SCALA程序

下载与配置IDEA 从官网下载 里面有Ultimate(最终版)和Community(社区版),对于普通的开发者来说,社区版就够了,然后因为我本来配置了JDK...

4209
来自专栏互联网杂技

SpringBoot (一) :入门篇

Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而...

1253
来自专栏LeoXu的博客

解决问题“Parsing Data for android-23 failed Unsupported major.minor version 51.0”

我是在 Eclipse 里面把 Android 工程中 project.properties 配置文件的 target 配置从 android-15 改成了 a...

1152

扫码关注云+社区

领取腾讯云代金券