Rumah > Artikel > pembangunan bahagian belakang > analisis terperinci PHP phar
Artikel ini membawakan anda pengetahuan yang berkaitan tentang PHP, yang terutamanya memperkenalkan kandungan yang berkaitan tentang phar Nama penuh phar ialah PHP Archive menyediakan cara untuk mengintegrasikan keseluruhan aplikasi PHP lihat kaedah memasukkannya ke dalam fail phar untuk memudahkan pemindahan dan pemasangan saya harap ia akan membantu semua orang.
Pembelajaran yang disyorkan: "Tutorial Video PHP"
Fail Jar (Java Archive), aplikasi, termasuk semua fail boleh laku dan boleh diakses, dibungkus ke dalam fail JAR, menjadikan proses penggunaan sangat mudah.
Serupa dengan JAR. Nama penuh phar ialah PHP Archive Sambungan phar menyediakan cara untuk meletakkan keseluruhan aplikasi PHP ke dalam fail .phar untuk memudahkan pergerakan dan pemasangan. Ciri terbesar fail phar ialah cara mudah untuk menggabungkan beberapa fail ke dalam satu fail. Fail .phar menyediakan cara untuk mengedarkan program PHP yang lengkap dalam satu fail dan menjalankannya daripada fail tersebut.
Tidak seperti JAR, Phar boleh diproses oleh PHP sendiri, jadi tidak perlu menggunakan alat tambahan untuk mencipta atau menggunakannya. Ia boleh dibuat atau diekstrak menggunakan skrip php.
Fail Phar mempunyai tiga format: arkib tar, arkib zip dan arkib phar Dua jenis pelaksanaan yang pertama memerlukan Phar untuk memasang sokongan sambungan Phar, dan jarang digunakan di sini terutamanya tentang format arkib phar .
Status lalai fail PHAR dibaca -sahaja , menggunakan fail Phar tidak memerlukan sebarang konfigurasi. Deployment sangat mudah. Kerana kita kini perlu mencipta fail Phar kita sendiri, kita perlu membenarkan penulisan pada fail Phar, yang perlu diubah suai php.ini
Dalam fail php.ini saya, phar.readonly = On.
[Phar] ; http://php.net/phar.readonly ;phar.readonly = On
Mula-mula, ubah suai pilihan phar.readonly dalam php.ini, alih keluar koma bertitik sebelumnya, dan tukar nilai kepada mati Atas sebab keselamatan, pilihan ini akan dihidupkan secara lalai .ini Jika ia dinyahdayakan (nilainya 0 atau dimatikan), ia boleh dihidupkan atau dimatikan dalam skrip pengguna Jika ia dihidupkan dalam php.ini, skrip pengguna tidak boleh dimatikan, jadi ia ditetapkan kepada di sini untuk menunjukkan contoh.
Kini, kita boleh membungkus aplikasi PHP ke dalam fail Phar.
Di sini saya copy terus dari projek blog orang lain dan tidak buat demonstrasi kerana niat asal saya menganjurkan blog ini adalah Untuk memahami phar:/ / kerentanan, gunakan ctf. Oleh itu, nama fail tidak akan diubah suai mengikut nama pengarang asal. Artikel rujukan akan ditambah pada akhir.
Pertama sekali, saya perlu mencipta struktur direktori aplikasi mengikut peraturan Direktori akar adalah projek, dan direktori di bawah projek adalah seperti berikut:
file -yunek.js -yunke.css lib -lib_a.php template -msg.html index.php Lib.php<.>Folder fail mempunyai Dua fail js dan css dengan kandungan kosong, hanya untuk menunjukkan bahawa phar boleh mengandungi berbilang format fail kandungan lib_a.php adalah seperti berikut:
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/10 * Time: 9:23 */ function show(){ echo "l am show()"; }msg. kandungan html adalah seperti berikut:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>phar</title> </head> <body> <?=$str; ?> </body> </html>kandungan index.php adalah seperti berikut:
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/10 * Time: 9:17 */ require "lib/lib_a.php"; show(); $str = isset($_GET["str"]) ? $_GET["str"] : "hello world"; include "template/msg.html";Kandungan Lib.php adalah seperti berikut:
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/10 * Time: 9:20 */ function yunke() { echo "l am yunke()"; }3 Cipta fail phar Fail projek sudah sedia Sekarang, cipta yunkeBuild.php dalam direktori yang sama dengan folder projek, yang digunakan untuk menjana fail format phar Kandungannya adalah seperti berikut:
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/10 * Time: 9:36 */ //产生一个yunke.phar文件 $phar = new Phar('yunke.phar', 0, 'yunke.phar'); // 添加project里面的所有文件到yunke.phar归档文件 $phar->buildFromDirectory(dirname(__FILE__) . '/project'); //设置执行时的入口文件,第一个用于命令行,第二个用于浏览器访问,这里都设置为index.php $phar->setDefaultStub('index.php', 'index.php');Kemudian akses fail yunkeBuild.php ini dalam penyemak imbas, dan fail yunke.phar, struktur direktori akar pelayan adalah seperti berikut: projekyunkeBuild.php
yunke.pharIni adalah untuk menjana arkib phar Fail adalah proses yang paling mudah.
Di sini saya akan membuat beberapa tambahan lain untuk pemahaman yang lebih baik:
1)Fail phar dijana dengan mengakses yunkeBuild.php, yang cukup untuk pelaksanaan . Oleh itu, anda boleh melaksanakan kod berikut dalam terminal untuk menjana
aabouzekry@platinum:~/myapp$ php yunkeBuild.phpdan kemudian fail yunke.phar dijana. 2)
phar() baharu menjana objek phar. Mari kita tafsirkan parameter.
<?php $phar = new Phar("/yunke.phar", FilesystemIterator::CURRENT_AS_FILEINFO | FilesystemIterator::KEY_AS_FILENAME, "yunke.phar");Penjelasan: Penciptaan objek
baharu biasanya memerlukan tiga parameter. Phar
mengendalikan fail. Objek Phar
mewarisi objek PHP Phar
dan parameter ini dihantar terus ke kelas induk. Nilai yang disediakan di sini ialah nilai lalai RecursiveDirectoryIterator
dan memenuhi keperluan semasa. RecursiveDirectoryIterator
Tambahkan fail ke phar. Terdapat beberapa cara untuk menambah fail:
调用类方法Phar::addFile($filepath,$localpath=?)
添加文件,参数是文件绝对路径和(可选)存储到phar的相对路径
<?php $phar = new Phar('yunke.phar'); $phar->addFile('test.php'); include('phar://yunke.phar/test.php') // in test.php ?>
这里出现的phar://就是访问phar文件的一种方法,所以不需要太在意。
以字符串添加文件内容
调用类方法Phar::addFromString($localpath,$contents)
以字符串形式添加文件
<?php $phar = new Phar('yunke.phar'); $phar->addFromString('test.php','<?php echo \'in test.php\'?>'); include('phar://yunke.phar/test.php'); // in test.php ?>
调用类方法Phar::addEmptyDir($dirname)
添加空目录,使用方法Phar::getContent()
获取文件结构
<?php $phar = new Phar('yunke.phar'); $phar->addEmptyDir('test'); // yunke.phar/test/ ?>
调用类方法Phar::buildFromDirectory($dir,$pattern = "")
添加整个目录
<?php $phar = new Phar('yunke.phar'); $phar->buildFromDirectory('test'); // test.php in test/ include('phar://yunke.phar/test/test.php'); // in test/test.php ?>
4) 存根文件Stub,理解这个很重要。
归档文件中有一个存根文件stub,其实就是一段php执行代码,在制作归档时可以设置,直接执行归档文件时,其实就是执行它,所以它是启动文件;在脚本中包含归档文件时就像包含普通php文件一样包含它并运行,但直接以phar://的方式包含归档中某一个文件时不会执行存根代码, 往往在存根文件里面require包含要运行的其他文件,对存根文件的限制仅为以__HALT_COMPILER(); 结束,默认的存根设计是为在没有phar扩展时能够运行,它提取phar文件内容到一个临时目录再执行,不过从php5.3开始该扩展默认内置启用了。
stub是phar文件的文件头,格式为...<?php ...;__HALT_COMPILER();?>
,…可以是任意字符,包括留空,且php闭合符与最后一个分号之间不能有多于一个的空格符。另外php闭合符也可省略。最短省略闭合符的stub是__HALT_COMPILER();?></p>
<p>运行Phar文件时,stub文件被当做一个meta文件来初始化Phar, 并告诉Phar文件在被调用时该做什么。</p>
<p>在我们的例子中,使用的是 <code>createDefaultStub() 方法。
其他的方式如下:
方法一:调用类方法Phar::setStub($string)
为实例创建自定义stub
<?php $phar = new Phar('yunke.phar'); $phar->setStub('<?php echo \'in stub!\';__HALT_COMPILER();?>'); include('phar://yunke.phar'); // in stub! ?>
也可以
$phar->setStub($phar->createDefaultStub("index.php"));
生成的缺省stub文件包含如下的代码:
<?php Phar::mapPhar(); include "phar://yunke.phar/index.php"; __HALT_COMPILER();
createDefaultStub()
方法缺省创建的stub文件的内容很简单。 Phar::mapPhar()
用来分析Phar文件的元数据,并初始化它。stub文件的结尾处需要调用 __HALT_COMPILER()
方法,这个方法后不能留空格。__HALT_COMPILER()
会立即终止PHP的运行,防止include的文件在此方法后仍然执行。这是Phar必须的,没有它Phar将不能正常运行。
除此之外,我们还可以创建自己的stub文件来执行自定义的初始化过程,像这样加载自定义文件
<?php $phar->setStub(file_get_contents("stub.php"));
方法二:使用默认stub,调用类方法Phar::setDefaultStub()
为实例设置默认stub,使用方法Phar::getStub()
获取实例的stub
<?php $phar = new Phar('yunke.phar'); $phar->setDefaultStub(); print_r($phar->getStub()); // 2, 'c' => 'text/plain', 'cc' => 'text/plain', ... ?>
如果缺省创建stub,PHP会使用默认stub
<?php $phar = new Phar('yunke.phar'); $phar['demo.txt'] = 'demo'; print_r($phar->getStub()); // 2, 'c' => 'text/plain', 'cc' => 'text/plain', ... ?>
我们在服务器根目录建立一个index.php文件来演示如何使用上面创建的phar文件,内容如下:
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/8 * Time: 9:33 */ require "yunke.phar"; require "phar://yunke.phar/Lib.php"; yunke();
如果index.php文件中只有第一行,那么和不使用归档文件时,添加如下代码完全相同:
require "project/index.php";
如果没有第二行,那么第三行的yunke()将提示未定义,所以可见require一个phar文件时并不是导入了里面所有的文件,而只是导入了入口执行文件而已,但在实际项目中往往在这个入口文件里导入其他需要使用的文件,在本例中入口执行文件为project/index.php。
补充:
可以为归档设置别名,别名保存在归档文件中永久保存,它可以用一个简短的名字引用归档,而不管归档文件在文件系统中存储在那里,设置别名:
$phar = new Phar('lib/yunke.phar', 0); $phar->setAlias ( "yun.phar");
设置别名后可以如下使用:
<?php require "lib/yunke.phar"; require "phar://yun.phar/Lib.php"; //使用别名访问归档文件 require "phar://lib/yunke.phar/Lib.php"; //当然仍然可以使用这样的方式去引用
如果在制作phar文件时没有指定别名,也可以在存根文件里面使用Phar::mapPhar('yunke.phar');指定。
我们有时候会好奇phar里面包含的文件源码,这个时候就需要将phar文件还原,如果只是看一看的话可以使用一些ide工具,比如phpstorm 10就能直接打开它,如果需要修改那么就需要提取操作了,为了演示,我们下载一个composer.phar放在服务器目录,在根目录建立一个get.php文件,内容如下
<?php /** * Created by yunke. * User: yunke * Date: 2017/2/9 * Time: 19:02 */ $phar = new Phar('composer.phar'); $phar->extractTo('composer'); //提取一份原项目文件 $phar->convertToData(Phar::ZIP); //另外再提取一份,和上行二选一即可
用浏览器访问这个文件,即可提取出来,以上列子展示了两种提取方式:
Barisan kedua akan mencipta direktori komposer dan meletakkan kandungan yang diekstrak ke dalamnya;
Barisan ketiga akan menghasilkan fail composer.zip, yang boleh diekstrak dan dipulihkan dengan menyahmampatnya.
Bagi pintasan pengesahan pelayan yang disebabkan oleh phar:// pseudo-protocol:
Pembelajaran yang disyorkan: "Tutorial Video PHP"
Atas ialah kandungan terperinci analisis terperinci PHP phar. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!