ホームページ  >  記事  >  バックエンド開発  >  PHPにおける効率改善・最適化手法の紹介

PHPにおける効率改善・最適化手法の紹介

黄舟
黄舟オリジナル
2017-07-27 10:06:241589ブラウズ

これまでのミニプロジェクトから学んだ経験と教訓を考慮すると、多くの場所は標準化されておらず、十分に効率的ではありません。そのため、PHP をさらに学習する際に、さらに遠回りしないように、インターネットのリソースに基づいて PHP 関連の知識を一定にまとめました。

PHPとは、英語のハイパーテキスト前処理言語Hypertext Preprocessorの略称です。 PHP は HTML 埋め込み言語であり、HTML ドキュメントに埋め込まれ、サーバー側で実行されるスクリプト言語です。言語スタイルは C 言語に似ており、広く使用されています。

PHP の独自の構文は、C、Java、perl、および PHP 独自の構文を組み合わせたものです。 CGI や Perl よりも高速に動的 Web ページを実行できます。他のプログラミング言語と比較して、PHP で作成された動的ページは HTML ドキュメントにプログラムを埋め込んで実行します。PHP は HTML タグを完全に生成する CGI よりも実行効率が高く、コンパイルされたコードも実行でき、コンパイルによりコードの暗号化と最適化が可能です。実行すると、コードの実行が速くなります。

PHP プログラミングに関するいくつかの要約:

0. 文字列を含めるには二重引用符の代わりに一重引用符を使用すると高速になります。 PHP は二重引用符で囲まれた文字列内の変数を検索しますが、一重引用符は検索しません。 注: これを実行できるのは echo だけです。これは複数の文字列をパラメータとして受け取ることができる「関数」です (注釈: PHP マニュアル echo は実際の関数ではなく言語構造であるため、関数は二重引用符で囲まれています)。

1. クラスメソッドを静的に定義できる場合は、それを静的に定義してみると、速度が 4 倍近く向上します。

2. $row[’id’] は $row[id] より 7 倍高速です。

3. Echo は print より高速で、echo $str1、$str2 などの文字列連結の代わりに echo の複数のパラメーター (注釈: ピリオドの代わりにカンマを使用) を使用します。

4. for ループを実行する前に最大ループ数を決定します。ループごとに最大値を計算するのではなく、代わりに foreach を使用することをお勧めします。

5. 未使用の変数、特に大きな配列の登録を解除して、メモリを解放します。

6. __get、__set、__autoload の使用は避けてください。

7. require_once() は高価です。

8. ファイルをインクルードする場合は、絶対パスを使用するようにしてください。これにより、PHP による include_path 内のファイルの検索速度が低下し、オペレーティング システムのパスの解析にかかる時間が短縮されます。

9. スクリプトの実行開始時刻 (注釈: サーバーがクライアント要求を受信する) を知りたい場合は、time() よりも $_SERVER[‘REQUEST_TIME’] を使用する方が良いでしょう。

10. 関数は正規表現を置き換えて同じ関数を完成させます。

11. str_replace 関数は preg_replace 関数よりも高速ですが、strtr 関数は str_replace 関数よりも 4 倍効率的です。

12. 文字列置換関数が配列または文字をパラメーターとして受け入れることができ、パラメーターの長さが長すぎない場合は、渡される各パラメーターが 1 つだけを記述するのではなく、文字になるように追加の置換コードを記述することを検討できます。コード行。クエリおよび置換のパラメータとして配列を受け入れます。

13. 複数の if、else if ステートメントを使用するよりも、選択的分岐ステートメント (翻訳アノテーション: switch case) を使用する方が良いです。

14. @ を使用してエラー メッセージをブロックするのは非常に非効率的です。

15. Apache の mod_deflate モジュールをオンにして、Web ページの閲覧速度を向上させます。

16. データベース接続は、使用が終了したら閉じる必要があります。長い接続は使用しないでください。

17. エラーメッセージは高価です。

18. メソッド内でローカル変数をインクリメントするのが最も速いです。関数内でローカル変数を呼び出すのとほぼ同じ速度です。

19. グローバル変数のインクリメントは、ローカル変数のインクリメントより 2 倍遅くなります。

20. オブジェクト プロパティ ($this->prop++ など) のインクリメントは、ローカル変数のインクリメントより 3 倍遅くなります。

21. 未定義のローカル変数をインクリメントするのは、事前定義されたローカル変数をインクリメントするよりも 9 ~ 10 倍遅くなります。

22. 関数内で呼び出さずにローカル変数を定義するだけでも、(ローカル変数をインクリメントするのと同じ程度に) 速度が低下します。 PHP はおそらく、グローバル変数が存在するかどうかを確認します。

23. 10 個のメソッドを追加したため (メソッドのテスト前とテスト後の両方)、メソッド呼び出しはクラスで定義されているメソッドの数に依存していないように見えますが、パフォーマンスに変化はありませんでした。

24. 派生クラスのメソッドは、基本クラスで定義された同じメソッドよりも高速に実行されます。

25. 1 つのパラメーターを指定して空の関数を呼び出すには、7 ~ 8 回のローカル変数のインクリメント操作を実行するのと同じ時間がかかります。同様のメソッド呼び出しには、15 近くのローカル変数の増分が必要です。

26. Apache が PHP スクリプトを解析する時間は、静的な HTML ページを解析する場合よりも 2 ~ 10 倍遅くなります。使用する静的な HTML ページを増やし、スクリプトを減らすようにしてください。

27. スクリプトをキャッシュできない限り、呼び出されるたびに再コンパイルされます。 PHP キャッシュ メカニズムを導入すると、通常、コンパイルのオーバーヘッドが排除され、パフォーマンスが 25% ~ 100% 向上します。

28. できるだけキャッシュするようにしてください。memcached を使用できます。 Memcached は、動的 Web アプリケーションを高速化し、データベースの負荷を軽減するために使用できる高性能メモリ オブジェクト キャッシュ システムです。 OP コードのキャッシュは、リクエストごとにスクリプトを再コンパイルする必要がないように便利です。

29. 文字列を操作し、その長さが特定の要件を満たしているかどうかを確認する必要がある場合は、当然 strlen() 関数を使用します。この関数は計算を行わず、zval 構造体 (PHP 変数の格納に使用される C の組み込みデータ構造体) に格納されている既知の文字列長を返すだけなので、非常に高速に実行されます。ただし、strlen() は関数であるため、関数呼び出しは小文字などの多くの手順を経るため、多少遅くなります (注釈: 小文字の関数名を指します。PHP は関数名の大文字と小文字を区別しません)。 )、ハッシュ検索、呼び出された関数と一緒に実行されます。場合によっては、 isset() トリックを使用してコードの実行を高速化できます。

(以下の例)

if (strlen($foo) < 5) { echo “Foo is too short”$$ }

(以下の手法と比較してください)

if (!isset($foo{5})) { echo “Foo is too short”$$ }

isset() の呼び出しは strlen() よりも高速です。後者とは異なり、isset() は言語構造であるためです。その実行には関数の検索や小文字の文字は必要ありません。つまり、実際には、最上位コードで文字列の長さをチェックするオーバーヘッドはそれほどかかりません。

30. ファイルロック関数flockの定数パラメータ——。

共有ロック (読み取り操作) - LOCK_SH

排他的ロック (書き込み操作) - LOCK_EX

ロックの解放 (共有か排他かを問わず) - LOCK_UN

アンチブロッキング - —LOCK_NB

ロック操作は fclose() 関数を通じて解放できます。

31. 文字列が正当な IP かどうかを確認します:

正規表現は必要ありません。正当な場合は数値を返し、正当でない場合は false を返します。 。

32. PHP 5.3 以降では、リアルパス (__FILE__) を必要とせずに、__DIR__ を使用して現在のスクリプトが配置されているディレクトリを取得できます。

33. checkdnsrr() を使用して、ドメイン名の存在を通じて一部の電子メール アドレスの有効性を確認します。

この組み込み関数は、インクリメントを実行するときに、各ドメイン名が IP アドレスに対応していることを確認します。または変数 $i の減分、 $i++ は ++$i よりも遅くなります。この違いは PHP に固有のものであり、他の言語には当てはまりません。そのため、C または Java コードを変更して、すぐに高速になることを期待しないでください。実際には機能しません。 ++$i は 3 つの命令 (オペコード) しか必要としないため高速ですが、$i++ は 4 つの命令を必要とします。ポストインクリメントでは、実際には、後でインクリメントされる一時変数が作成されます。プレフィックスの増分は、元の値に直接増加します。これは、Zend の PHP オプティマイザーによって行われるような最適化の一種です。すべてのコマンド オプティマイザーが同じ最適化を実行するわけではなく、コマンド オプティマイザーを備えていないインターネット サービス プロバイダー (ISP) やサーバーが多数存在するため、この最適化を念頭に置くことをお勧めします。

35. オブジェクト指向 (OOP) である必要はありません。オブジェクト指向は多くの場合非常に高価であり、各メソッドとオブジェクトの呼び出しは大量のメモリを消費します。

36. すべてのデータ構造を実装するためにクラスを使用する必要はありません。配列も非常に便利です。

37. 本当に再利用するコードを慎重に考えてください。

38. 必要に応じていつでもコードをメソッドに分割できます。

行数が少なく頻度の高いメソッドについては、コードを直接記述するようにしてください。これにより、関数スタックのオーバーヘッドが軽減されます。また、メソッドのネストが深すぎないように注意してください。そうしないと、PHP の動作効率に大きな影響が生じます。

39. できるだけ多くの PHP 組み込み関数を使用するようにしてください。

40. コード内に時間のかかる関数が多数ある場合は、C 拡張機能を使用して実装することを検討できます。

41. コードをプロファイリングします。チェッカーは、コードのどの部分にどれくらいの時間がかかっているかを示します。 Xdebug デバッガーには、コードの全体的な整合性を評価し、コード内のボトルネックを明らかにする検査プログラムが含まれています。

42. mod_zip を Apache モジュールとして使用すると、データを即座に圧縮し、データ送信量を 80% 削減できます。

43. file、fopen、feof、fgets などの一連のメソッドの代わりに file_get_contents を使用できる場合は、より効率的な file_get_contents を使用してください。 file_get_contents は、ファイル ハンドルが正常に開かれたかどうかを判断する必要はありません。ただし、URL ファイルを開くときは、file_get_contents の PHP バージョンの問題に注意してください。

44. PHP のファイル操作効率は低くありませんが、可能な限り Select SQL ステートメントを最適化します。挿入と更新の操作はできるだけ少なくします (更新ページで批判されました)。

46. PHP の内部関数をできるだけ使用します (ただし、PHP に存在しない関数を見つけるのに時間を無駄にしました)。カスタム関数を書くのは経験の問題です!

47. ループ内で変数、特に大きな変数を宣言しないでください (これは PHP だけの問題ではないようです)。多次元配列のループ内でネストされた代入を行うには、

50. PHP の内部文字列操作関数を使用できる場合は、正規表現を使用しないでください。while および for ループの代わりに foreach を使用してください。 ;

51. 文字列を引用符で囲むには二重引用符の代わりに一重引用符を使用します。

52.「i=i+1 の代わりに i+=1 を使用します。」グローバル変数の場合は、使用後に Unset() する必要があります。

54. 中括弧 "{}" は、"[]" が配列を操作して指定された位置の文字を取得するように文字列を操作できます。

55. PHP タグ「bb9bd6d87db7f8730c53cb084e6b4d2d」は、出力やレポートのエラーを引き起こす誤ったスペースを避けるために、スタンドアロンの PHP スクリプトで終了タグを記述する必要はありません。コメントを使用してスクリプトの終了を示すことができます。

56. エコーは関数ではなく文法構造です。複数の文字列が続く場合は、コンマ「,」を使用する方が効率的です。

57. 配列内で、1、'1'、true がインデックスの場合は強制的に 1 に変換されます。また、「01」は変換されず、文字列として処理されます。

58. クラスのコードを異なる PHP タグに記述することは違法であり、構文エラーが報告されます。機能は問題ありません。

59. セッションとクッキーの違いと関係。

セッションはサーバーに保存され、Cookieはクライアントのブラウザに保存されます。

セッションはハードドライブ上のファイル、データベース、またはmemcachedに保存でき、Cookieはハードドライブに保存できます。 (永続的な cookie) とメモリ (セッション cookie);

session_id を渡すには 2 つの方法があり、1 つは cookie で、もう 1 つは get メソッドです (session.name 設定項目を通じて session_id を保存するための変数名を指定できます)。

60. time() の代わりに $_SERVER['REQUEST_TIME'] を使用して現在のタイムスタンプを取得すると、関数呼び出しが 1 つ削減され、より効率的になります。

61. _REQUEST を使用する場合、データを取得する際の優先順位は E(nv)G(et)P(ost)C(ookie)S(ession) になる可能性があるので注意してください。優先度の高いデータは優先度の低いデータで上書きされます。

62. header() 関数の後に終了する必要があります。そうしないと、後続のコードが引き続き実行されます。

63. メモリ使用量を削減するために大きな配列は参照によって渡され、使用される場合は unset() が使用されます。

64. set_time_limit() の制限。制限できるのはスクリプト自体の実行時間だけであり、system() 関数、ストリーム操作、データベース クエリなどの外部実行時間は制御できません。

65. echo、print、print_r、var_dump、var_export の違い:

echo と print は関数ではなく、基本的な型のみを表示できます。その他は関数であり、配列とオブジェクトを表示できます。 ;

echo はカンマで区切って複数の変数を表示できます。

print_r 2 番目のパラメータは、変数を出力するか、戻り値として使用するかを決定します。

var_dump は、変数の長さや型などの情報、および複数の変数をパラメータとして渡すことができます。

var_export は有効な PHP コード形式を返します。 66. 検証メール: filter_var($email, FILTER_VALIDATE_EMAIL);

67. ファイル拡張子を取得するメソッド:

1. 拡張子の値、または pathinfo($filename, PATHINFO_EXTENSION) )。

2、end(explode('.',$filename))。 68. 三項演算子 (?:) を使用してみてください。

69. error_reporting(0) 関数を使用して、機密情報がユーザーに表示されないようにします。

理想的には、php.ini ファイルでエラー報告を完全に無効にする必要があります。ただし、共有仮想ホストを使用していて php.ini を変更できない場合は、error_reporting(0) 関数を追加して各スクリプト ファイルの最初の行に配置する (または require_once() でロードする) ことをお勧めします。これにより、エラーが発生したときに、機密の SQL クエリとパスが表示されないよう効果的に保護できます。

70. 変数を不用意にコピーしないでください

PHP コードを整理するために、一部の PHP 初心者 (私を含む) が事前定義された変数を使用することがあります。変数の名前を短くすると、実際にはメモリ消費量が 2 倍になり、プログラムの速度が遅くなるだけです。次の例で、ユーザーが悪意を持って 512KB のテキストをテキスト入力ボックスに挿入すると、1MB のメモリが消費されると想像してください。

BAD:
$description = $_POST[&#39;description&#39;];
echo $description;
GOOD:
echo $_POST[&#39;description&#39;];

1.在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题; 2.尽量的少进行文件操作,虽然PHP的文件操作效率也不低的; 3.优化Select SQL语句,在可能的情况下尽量少的进行Insert、Update操作(在update上,我被恶批过); 4.尽可能的使用PHP内部函数(但是我却为了找个PHP里面不存在的函数,浪费了本可以写出一个自定义函数的时间,经验问题啊!); 5.循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?); 6.多维数组尽量不要循环嵌套赋值; 7.在可以用PHP内部字符串操作函数的情况下,不要用正则表达式; 8.foreach效率更高,尽量用foreach代替while和for循环; 9.用单引号替代双引号引用字符串; 10.“用i+=1代替i=i+1。符合c/c++的习惯,效率还高”; 11.对global变量,应该用完就unset()掉; 以下是一篇关于提高PHP效率的文章 

榨干 PHP,提高效率

这篇杂文翻译整理自网络各路文档资料(见最末的参考资料),尤其是 Ilia Alshanetsky (佩服之至) 在多个 PHP 会议上的演讲,主要是各类提高 PHP 性能的技巧。为求精准,很多部分都有详细的效率数据,以及对应的版本等等。偷懒,数据就不一一给出了,直接给结论。 ======================================================== 

静态调用的成员一定要定义成 static (PHP5 ONLY)

贴士:PHP 5 引入了静态成员的概念,作用和 PHP 4 的函数内部静态变量一致,但前者是作为类的成员来使用。静态变量和 Ruby 的类变量(class variable)差不多,所有类的实例共享同一个静态变量。

QUOTE: class foo { function bar() {  echo &#39;foobar&#39;; } } $foo = new foo; // instance way $foo->bar(); // static way foo::bar();

静态地调用非 static 成员,效率会比静态地调用 static 成员慢 50-60%。主要是因为前者会产生 E_STRICT 警告,内部也需要做转换。 ======================================================== 

使用类常量 (PHP5 ONLY)

贴士:PHP 5 新功能,类似于 C++ 的 const。 使用类常量的好处是: - 编译时解析,没有额外开销 - 杂凑表更小,所以内部查找更快 - 类常量仅存在于特定「命名空间」,所以杂凑名更短 - 代码更干净,使除错更方便 ======================================================== (暂时)不要使用 require/include_once require/include_once 每次被调用的时候都会打开目标文件! - 如果用绝对路径的话,PHP 5.2/6.0 不存在这个问题 - 新版的 APC 缓存系统已经解决这个问题 文件 I/O 增加 => 效率降低如果需要,可以自行检查文件是否已被 require/include。不要调用毫无意义的函数有对应的常量的时候,不要使用函数。

QUOTE: php_uname(&#39;s&#39;) == PHP_OS; php_version() == PHP_VERSION; php_sapi_name() == PHP_SAPI;

虽然使用不多,但是效率提升大概在 3500% 左右。 最快的 Win32 检查 $is_win = DIRECTORY_SEPARATOR == '\\'; ======================================================== 

- 不用函数

- Win98/NT/2000/XP/Vista/Longhorn/Shorthorn/Whistler…通用 - 一直可用 时间问题 (PHP>5.1.0 ONLY) 你如何在你的软件中得知现在的时间?简单,「time() time() again, you ask me…」。不过总归会调用函数,慢。现在好了,用 $_SERVER[‘REQUEST_TIME’],不用调用函数,又省了。 

加速 PCRE

- 对于不用保存的结果,不用 (),一律用 (?。这样 PHP 不用为符合的内容分配内存,省。效率提升 15% 左右。 - 能不用正则,就不用正则,在分析的时候仔细阅读手册「字符串函数」部分。有没有你漏掉的好用的函数? 例如:

 strpbrk() strncasecmp() strpos()/strrpos()/stripos()/strripos()

======================================================== 

加速 strtr

如果需要转换的全是单个字符的时候,用字符串而不是数组来做

strtr:   $addr = strtr($addr, "abcd", "efgh"); // good $addr = strtr($addr, array(&#39;a&#39; => &#39;e&#39; ,//..)); // bad

效率提升:10 倍。 ======================================================== 

不要做无谓的替换

即使没有替换,str_replace 也会为其参数分配内存。很慢!解决办法: - 用 strpos 先查找(非常快),看是否需要替换,如果需要,再替换 效率: - 如果需要替换:效率几乎相等,差别在 0.1% 左右。 - 如果不需要替换:用 strpos 快 200%。 ======================================================== 

邪恶的 @ 操作符

不要滥用 @ 操作符。虽然 @ 看上去很简单,但是实际上后台有很多操作。用 @ 比起不用 @,效率差距:3 倍。 特别不要在循环中使用 @,在 5 次循环的测试中,即使是先用 error_reporting(0) 关掉错误,在循环完成后再打开,都比用 @ 快。 ======================================================== 

善用 strncmp

当需要对比「前 n 个字符」是否一样的时候,用 strncmp/strncasecmp,而不是 substr/strtolower,更不是 PCRE,更千万别提 ereg。strncmp/strncasecmp 效率最高(虽然高得不多)。 慎用 substr_compare (PHP5 ONLY),按照上面的道理,substr_compare 应该比先 substr 再比较快咯。答案是否定的,除非: - 无视大小写的比较 - 比较较大的字符串 不要用常量代替字符串 为什么: - 需要查询杂凑表两次 - 需要把常量名转换为小写(进行第二次查询的时候) - 生成 E_NOTICE 警告 - 会建立临时字符串 效率差别:700%。 ======================================================== 

不要把 count/strlen/sizeof 放到 for 循环的条件语句中

贴士:一个经典做法

QUOTE: for ($i = 0, $max = count($array);$i < $max; ++$i); ?>

效率提升相对于: - count 50% - strlen 75% ======================================================== 

短的代码不一定快

// longest if ($a == $b) { $str .= $a; } else { $str .= $b; } // longer if ($a == $b) { $str .= $a; } $str .= $b; // short $str .= ($a == $b ? $a : $b);

  你觉得哪个快? 效率比较:

 - longest: 4.27 - longer: 4.43 - short: 4.76

不可思议?再来一个:

  // original $d = dir(&#39;.&#39;); 
  while (($entry = $d->read()) !== false) { if ($entry == &#39;.&#39; || $entry == &#39;..&#39;) { continue; } } 
  // versus glob(&#39;./*&#39;); 
  // versus (include . and ..) scandir(&#39;.&#39;);

  哪个快? 效率比较: - original: 3.37 - glob: 6.28 - scandir: 3.42 - original without OO: 3.14 - SPL (PHP5): 3.95 画外音:从此也可以看出来 PHP5 的面向对象效率提高了很多,效率已经和纯函数差得不太多了。 ======================================================== 

提高 PHP 文件访问效率

需要包含其他 PHP 文件的时候,使用完整路径,或者容易转换的相对路径。  

include &#39;file.php&#39;; // bad approach incldue &#39;./file.php&#39;; // good include &#39;/path/to/file.php&#39;; // ideal

======================================================== 

物尽其用

PHP 有很多扩展和函数可用,在实现一个功能的之前,应该看看 PHP 是否有了这个功能?是否有更简单的实现?  

$filename = "./somepic.gif"; 
$handle = fopen($filename, "rb"); 
$contents = fread($handle, filesize($filename)); 
fclose($handle); // vs. much simpler file_get_contents(&#39;./somepic.gif&#39;);

  ======================================================== 

关于引用的技巧

引用可以: - 简化对复杂结构数据的访问 - 优化内存使用  

$a[&#39;b&#39;][&#39;c&#39;] = array(); 
// slow 2 extra hash lookups per access for ($i = 0; $i < 5; ++$i) $a[&#39;b&#39;][&#39;c&#39;][$i] = $i; 
// much faster reference based approach $ref =& $a[&#39;b&#39;][&#39;c&#39;]; for ($i = 0; $i < 5; ++$i) $ref[$i] = $i; ?> $a = &#39;large string&#39;; 
// memory intensive approach function a($str){ return $str.&#39;something&#39;; }
 // more efficient solution function a(&$str){ $str .= &#39;something&#39;; }   
Aug 25th, 2009
Comments

以上がPHPにおける効率改善・最適化手法の紹介の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。