PHP 7.4中使用预加载的方法详解

前言

PHP 7.4增加了预加载支持,这一功能可以显着提高代码的性能。

这是一个简单的预加载:

  • 为了预加载文件,您需要编写自定义PHP脚本
  • 此脚本在服务器启动时执行一次
  • 所有预加载的文件都可在内存中用于所有请求
  • 在重新启动服务器之前,对源文件所做的更改不会产生任何影响

让我们深入研究一下。

Opcache,但更多

虽然预加载是在顶级操作opcache上构建的,但它并不完全相同。Opcache将获取您的PHP源文件,将其编译为“操作码”,并将这些编译后的文件存储在磁盘上。

您可以将“操作码”视为代码的低级表示,可以在运行时轻松解释。因此,opcache会跳过源文件之间的转换步骤以及PHP解释器在运行时实际需要的内容。一场巨大的胜利!

但是,还有更多的东西可以获得。Opcached文件不了解其他文件。如果你有一个A从类扩展的类B,你仍然需要在运行时将它们链接在一起。此外,opcache执行检查以查看源文件是否已被修改,并将基于此文件使其缓存无效。

所以这就是预加载发挥作用的地方:它不仅将源文件编译为操作码,还将相关的类,特征和接口链接在一起。然后,它将保留这个“已编译”的可运行代码blob - 即:PHP解释器可用的代码 - 在内存中。

当请求到达服务器时,它现在可以使用已经加载到内存中的部分代码库,而没有任何开销。

那么,我们谈论的是“代码库的哪些部分”?

在实践中预加载

为了使预加载工作,开发人员必须告诉服务器要加载哪些文件。这是通过一个简单的PHP脚本完成的,所以没有什么可怕的。

规则很简单:

  • 您提供了一个预加载脚本,并使用您的php.ini文件链接到它 opcache.preload
  • 您要预加载的每个PHP文件都应该opcache_compile_file()从preload脚本中传递到

假设您想要预加载一个框架,例如Laravel。您的脚本必须遍历目录中的所有PHP文件vendor/laravel,并逐个包含它们。

以下是您在php.ini中链接到此脚本的方法:

opcache.preload=/path/to/project/preload.php

这是一个虚拟实现:

$files = /* An array of files you want to preload */;

foreach ($files as $file) {
 opcache_compile_file($file);
}

请注意opcache_compile_file,您也可以使用include该文件代替使用。虽然似乎有一个bug,因为在编写时这似乎不起作用。

警告:无法预加载未链接的类

坚持下去,有一个警告!为了预先加载文件,还必须预先加载它们的依赖项 - 接口,特征和父类。

如果类依赖项有任何问题,您会在服务器启动时注意到它:

Can't preload unlinked class
Illuminate\Database\Query\JoinClause:
Unknown parent
Illuminate\Database\Query\Builder

请参阅opcache_compile_file()解析文件,但不执行它。这意味着如果一个类具有未预加载的依赖项,则其本身也不能被预加载。

这不是一个致命的问题,您的服务器将正常工作; 但是你不会拥有你真正想要的所有预装文件。

这就是为什么你应该注意要预加载哪些文件,以确保解决所有依赖项。手动执行此操作可能看起来像是一件苦差事,所以很自然人们已经在开发自动化解决方案。

Composer支持

最有前途的自动化解决方案来自composer,现在已经被大多数现代PHP项目所使用。

人们正在努力添加预加载配置选项composer.json,进而为您生成预加载文件!就像预加载一样,此功能仍在进行中,但可以在此处进行操作

幸运的是,如果你不想,你将不需要手动配置预加载文件,composer将能够为你做到这一点。

服务器要求

关于使用预加载时的devops方面,还有两个更重要的事情需要提及。

您已经知道需要在php.ini中指定一个条目才能使预加载工作。这意味着如果您使用共享主机,则无法根据需要自由配置PHP。

实际上,您需要一个专用(虚拟)服务器才能为单个项目优化预加载的文件。所以记住这一点。

还要记住php-fpm,每次要重新加载内存中的文件时,都需要重新启动服务器(如果你正在使用它就足够了)。这对大多数人来说似乎是显而易见的,但仍值得一提。

性能

现在谈到最重要的问题:预加载实际上是否提高了性能?

答案是肯定的,当然:Ben Morel分享了一些基准,可以在与之前相关的同一个composer问题中找到。

有趣的是,您可以决定只预加载“热门类”:代码库中经常使用的类。Ben的基准测试表明,只加载大约100个热门类,实际上比预加载所有产生更好的性能提升。这是性能提升13%和17%的差异。

应该预先加载哪些类依赖于您的特定项目。在开始时尽可能简单地预加载是明智的。如果您确实需要增加几个百分比,则必须在运行时监控代码。

所有这些当然也可以自动化,并且可能在将来完成。

现在,最重要的是要记住,comopser会添加支持,这样你就不必自己制作预装文件了,而且这个功能很容易在服务器上设置,因为你可以完全控制它。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。

(0)

相关推荐

  • 说说PHP的autoLoad自动加载机制

    __autoload的使用方法1: 最经常使用的就是这种方法,根据类名,找出类文件,然后require_one 复制代码 代码如下: function __autoload($class_name) { $path = str_replace('_', '/', $class_name); require_once $path . '.php'; } // 这里会自动加载Http/File/Interface.php 文件 $a = new Http_File_Interface(); 这种方法

  • php自动加载的两种实现方法

    php自动载方法有两种. 第一种方案用__autoload,这个函数较简单,也较弱. 但有一问题没有解决, 就是在include前判断文件是否存在的问题. 复制代码 代码如下: set_include_path('aa' . PATH_SEPARATOR . get_include_path()); function __autoload($className) { //如果加这个检测, 因为此文件不在当前目录下,它就会检测不到文件存在, //但include是能成功的 if (file_exi

  • php自动加载方式集合

    php加载文件方式: 1.include,include_once,requice,requice_one常规加载 2.__autoload() 3.spl_autoload_register() 常规加载方式 假设我们有一个类文件A.php,里面定义了一个名字为A的类: <?php class A { public function __construct() { echo 'Got it.'; } } 然后我们有一个index.php需要用到这个类A,常规的写法就是 <?php requi

  • PHP 7.4中使用预加载的方法详解

    前言 PHP 7.4增加了预加载支持,这一功能可以显着提高代码的性能. 这是一个简单的预加载: 为了预加载文件,您需要编写自定义PHP脚本 此脚本在服务器启动时执行一次 所有预加载的文件都可在内存中用于所有请求 在重新启动服务器之前,对源文件所做的更改不会产生任何影响 让我们深入研究一下. Opcache,但更多 虽然预加载是在顶级操作opcache上构建的,但它并不完全相同.Opcache将获取您的PHP源文件,将其编译为"操作码",并将这些编译后的文件存储在磁盘上. 您可以将&qu

  • 基于Python实现配置热加载的方法详解

    目录 背景 如何实现 使用多进程实现配置热加载 使用signal信号量来实现热加载 采用multiprocessing.Event 来实现配置热加载 结语 背景 由于最近工作需求,需要在已有项目添加一个新功能,实现配置热加载的功能.所谓的配置热加载,也就是说当服务收到配置更新消息之后,我们不用重启服务就可以使用最新的配置去执行任务. 如何实现 下面我分别采用多进程.多线程.协程的方式去实现配置热加载. 使用多进程实现配置热加载 如果我们代码实现上使用多进程, 主进程1来更新配置并发送指令,任务的

  • 微信小程序实现瀑布流布局与无限加载的方法详解

    前言 瀑布流布局是一种比较流行的页面布局方式,最典型的就是Pinterest.com,每个卡片的高度不都一样,形成一种参差不齐的美感. 在HTML5中,我们可以找到很多基于jQuery之类实现的瀑布流布局插件,轻松做出这样的布局形式.在微信小程序中,我们也可以做出这样的效果,不过由于小程序框架的一些特性,在实现思路上还是有一些差别的. 今天我们就来看一下如何在小程序中去实现这种瀑布流布局: 小程序瀑布流布局 我们要实现的是一个固定2列的布局,然后将图片数据动态加载进这两列中(而加载进来的图片,会

  • PHP从零开始打造自己的MVC框架之类的自动加载实现方法详解

    本文实例讲述了PHP从零开始打造自己的MVC框架之类的自动加载实现方法.分享给大家供大家参考,具体如下: 前面介绍了MVC框架的入口文件,接下来我们希望完成一个"自动加载类"的功能,我们把这个功能放到Imooc这个基础类当中. core\imooc.php: <?php namespace core; class Imooc { public static $classMap = array(); static public function run() { p('ok'); $

  • Android ListView异步加载图片方法详解

    本文实例讲述了Android ListView异步加载图片方法.分享给大家供大家参考,具体如下: 先说说这篇文章的优点把,开启线程异步加载图片,然后刷新UI显示图片,而且通过弱引用缓存网络加载的图片,节省了再次连接网络的开销. 这样做无疑是非常可取的方法,但是加载图片时仍然会感觉到轻微的卡屏现象,特别是listview里的item在进行快速滑动的时候. 我找了一下原因,可能是在listview快速滑动屏幕的时候划过的item太多 而且每次调用getView方法后就会异步的在过去某个时间内用han

  • Android 开发中fragment预加载问题

    我们在做应用开发的时候,一个Activity里面可能会以viewpager(或其他容器)与多个Fragment来组合使用,而如果每个fragment都需要去加载数据,或从本地加载,或从网络加载,那么在这个activity刚创建的时候就变成需要初始化大量资源.这样的结果,我们当然不会满意.那么,能不能做到当切换到这个fragment的时候,它才去初始化呢? 答案就在Fragment里的setUserVisibleHint这个方法里.请看关于Fragment里这个方法的API文档(国内镜像地址:ht

  • 基于vue中css预加载使用sass的配置方式详解

    1.安装sass的依赖包 npm install --save-dev sass-loader //sass-loader依赖于node-sass npm install --save-dev node-sass 2.在build文件夹下的webpack.base.conf.js的rules里面添加配置,如下红色部分 { test: /\.sass$/, loaders: ['style', 'css', 'sass'] } <span style="color:#454545;"

  • 利用CSS、JavaScript及Ajax实现图片预加载的方法

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布,也可帮助用户在浏览你网站内容时获得更好的用户体验.本文将分享三个不同的预加载技术,来增强网站的性能与可用性. 实现图片预加载可以使用css.JavaScript.Ajax三种方法.下面分别介绍这些方法的实现. 使用CSS 单纯的使用css可以将图片加载到页面元素的背景上,这种方法简单.高效: #d

  • js预加载图片方法汇总

    本文实例汇总了js预加载图片方法.分享给大家供大家参考.具体分析如下: 1. 纯CSS: #preload-01 { background: url(http://domain.tld/image-01.png) no-repeat -9999px -9999px; } #preload-02 { background: url(http://domain.tld/image-02.png) no-repeat -9999px -9999px; } #preload-03 { backgroun

  • web.xml中如何设置配置文件的加载路径实例详解

    web.xml中如何设置配置文件的加载路径实例详解 web应用程序通过Tomcat等容器启动时,会首先加载web.xml文件,通常我们工程中的各种配置文件,如日志.数据库.spring的文件等都在此时被加载,下面是两种常用的配置文件加载路径,即配置文件可以放到 SRC目录下或者可以放到WEB-INF根目录下 第一种在web.xml中这样配置: <context-param> <param-name >contextConfigLocation </param-name >

随机推荐