{"id":47457,"date":"2024-12-02T10:23:55","date_gmt":"2024-12-02T02:23:55","guid":{"rendered":"https:\/\/fwq.ai\/blog\/47457\/"},"modified":"2024-12-02T10:23:55","modified_gmt":"2024-12-02T02:23:55","slug":"php%e5%87%bd%e6%95%b0%e6%80%a7%e8%83%bd%e5%88%86%e6%9e%90%e5%b7%a5%e5%85%b7%e4%bb%8b%e7%bb%8d%ef%bc%9a%e4%bd%bf%e7%94%a8%e6%a1%88%e4%be%8b%e5%88%86%e4%ba%ab","status":"publish","type":"post","link":"https:\/\/fwq.ai\/blog\/47457\/","title":{"rendered":"php\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab"},"content":{"rendered":"<p><b><\/b> <\/p>\n<h1>php\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab<\/h1>\n<p>\u54c8\u55bd\uff01\u4eca\u5929\u5fc3\u8840\u6765\u6f6e\u7ed9\u5927\u5bb6\u5e26\u6765\u4e86<span style=\"color: #FF6600;, Helvetica, Arial, sans-serif;font-size: 14px;background-color: #FFFFFF\">\u300aphp\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab\u300b<\/span>\uff0c\u60f3\u5fc5\u5927\u5bb6\u5e94\u8be5\u5bf9<span style=\"color: #FF6600;, Helvetica, Arial, sans-serif;font-size: 14px;background-color: #FFFFFF\">\u6587\u7ae0<\/span>\u90fd\u4e0d\u964c\u751f\u5427\uff0c\u90a3\u4e48\u9605\u8bfb\u672c\u6587\u5c31\u90fd\u4e0d\u4f1a\u5f88\u56f0\u96be\uff0c\u4ee5\u4e0b\u5185\u5bb9\u4e3b\u8981\u6d89\u53ca\u5230<span style=\"color: #FF6600;, Helvetica, Arial, sans-serif;font-size: 14px;background-color: #FFFFFF\"><\/span>\uff0c\u82e5\u662f\u4f60\u6b63\u5728\u5b66\u4e60<span style=\"color: #FF6600;, Helvetica, Arial, sans-serif;font-size: 14px;background-color: #FFFFFF\">\u6587\u7ae0<\/span>\uff0c\u5343\u4e07\u522b\u9519\u8fc7\u8fd9\u7bc7\u6587\u7ae0~\u5e0c\u671b\u80fd\u5e2e\u52a9\u5230\u4f60\uff01<\/p>\n<p>\u672c\u6587\u4ecb\u7ecd\u4e86\u4e09\u79cd\u6d41\u884c\u7684 PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\uff1aPHPStan\uff1a\u9759\u6001\u5206\u6790\uff0c\u83b7\u53d6\u51fd\u6570\u6267\u884c\u65f6\u95f4\u4f30\u8ba1Blackfire\uff1a\u4ea4\u4e92\u5f0f\u706b\u7130\u56fe\uff0c\u8be6\u7ec6\u6027\u80fd\u62a5\u544a\uff0c\u53ef\u68c0\u6d4b\u5185\u5b58\u6cc4\u6f0fXdebug\uff1a\u7ec6\u7c92\u5ea6\u8c03\u8bd5\uff0c\u63d0\u4f9b\u51fd\u6570\u8c03\u7528\u5806\u6808\u4fe1\u606f<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20241026\/1729918667671c76cbca961.jpg\" class=\"aligncenter\" title=\"php\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab\u63d2\u56fe\" alt=\"php\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab\u63d2\u56fe\" \/><\/p>\n<p><strong>PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u5b9e\u6218\u6848\u4f8b\u5206\u4eab<\/strong><\/p>\n<p>\u51fd\u6570\u6027\u80fd\u5206\u6790\u5bf9\u4e8e\u4f18\u5316 PHP \u5e94\u7528\u7a0b\u5e8f\u81f3\u5173\u91cd\u8981\u3002\u672c\u6587\u4ecb\u7ecd\u4e86\u51e0\u79cd\u6d41\u884c\u7684 PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\uff0c\u5e76\u901a\u8fc7\u5b9e\u9645\u6848\u4f8b\u5206\u4eab\u5176\u4f7f\u7528\u3002<\/p>\n<p><strong>PHPStan<\/strong><\/p>\n<p>PHPStan \u662f\u4e00\u4e2a\u9759\u6001\u5206\u6790\u5de5\u5177\uff0c\u53ef\u4ee5\u68c0\u6d4b\u4ee3\u7801\u4e2d\u7684\u9519\u8bef\u5e76\u5bf9\u51fd\u6570\u6267\u884c\u65f6\u95f4\u8fdb\u884c\u4f30\u8ba1\u3002\u5b83\u652f\u6301\u7c7b\u578b\u58f0\u660e\u5e76\u63d0\u4f9b\u8be6\u7ec6\u7684\u6027\u80fd\u62a5\u544a\u3002<\/p>\n<p><strong>\u5b9e\u6218\u6848\u4f8b\uff1a<\/strong>\u4f18\u5316 laravel \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u67e5\u8be2\u6027\u80fd<\/p>\n<p>\u4f7f\u7528 PHPStan\uff0c\u53ef\u4ee5\u8f7b\u677e\u8bc6\u522b\u9700\u8981\u4f18\u5316\u7684\u9ad8\u8017\u65f6\u67e5\u8be2\u3002<\/p>\n<pre>use PHPStan\\Analyser\\Analyser;\nuse PHPStan\\Rules\\Rule;\nuse PHPStan\\Testing\\RuleTestCase;\n\nclass SlowQueryRuleTest extends RuleTestCase\n{\n    protected function getRules(): array\n    {\n        return [\n            new SlowQueryRule()\n        ];\n    }\n\n    public function testSlowQuery()\n    {\n        $this-&gt;analyse(\n            [\n                '&lt;?php',\n                'class QueryExample',\n                '{',\n                '    public function getSlowQuery()',\n                '    {',\n                '        $result = DB::select(\"SELECT * FROM users WHERE name = 'John'\");',\n                '        return $result;',\n                '    }',\n                '}'\n            ],\n            [\n                [\n                    'description' =&gt; 'Query \"SELECT * FROM users WHERE name = \\'John\\'\"; may be slow.',\n                    'line' =&gt; 7\n                ]\n            ]\n        );\n    }\n}<\/pre>\n<p>\u6267\u884c\u5206\u6790\u5668\u53ef\u4ee5\u8bc6\u522b\u51fa\u67e5\u8be2\u53ef\u80fd\u6027\u80fd\u8f83\u5dee\uff0c\u5e76\u63a8\u8350\u4f18\u5316\u63d0\u793a\u3002<\/p>\n<p><strong>Blackfire<\/strong><\/p>\n<p>Blackfire \u662f\u4e00\u4e2a\u529f\u80fd\u5f3a\u5927\u7684\u5206\u6790\u5de5\u5177\uff0c\u53ef\u4ee5\u6d4b\u91cf\u51fd\u6570\u7684\u6267\u884c\u65f6\u95f4\u548c\u5185\u5b58\u4f7f\u7528\u60c5\u51b5\u3002\u5b83\u63d0\u4f9b\u4ea4\u4e92\u5f0f\u706b\u7130\u56fe\u548c\u8be6\u7ec6\u7684\u6027\u80fd\u62a5\u544a\u3002<\/p>\n<p><strong>\u5b9e\u6218\u6848\u4f8b\uff1a<\/strong>\u5206\u6790 Symfony \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u5185\u5b58\u6cc4\u6f0f<\/p>\n<p>Blackfire \u53ef\u4ee5\u5e2e\u52a9\u8bc6\u522b\u5e76\u89e3\u51b3 Symfony \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u5185\u5b58\u6cc4\u6f0f\u3002<\/p>\n<pre>use Psr\\Http\\Message\\ServerRequestInterface;\nuse Psr\\Http\\Message\\ResponseInterface;\n\nclass MemoryLeakController\n{\n    public function __construct(private EntityManagerInterface $entityManager)\n    {\n    }\n\n    public function __invoke(ServerRequestInterface $request): ResponseInterface\n    {\n        \/\/ \u83b7\u53d6\u7528\u6237\u5bf9\u8c61\uff0c\u9700\u8981\u6ce8\u610f\u4fdd\u6301\u5bf9\u8c61\u5f15\u7528\n        $user = $this-&gt;entityManager-&gt;getRepository(User::class)-&gt;find(1);\n\n        \/\/ \u6267\u884c\u8017\u65f6\u7684\u64cd\u4f5c\n\n        \/\/ \u6e05\u9664\u5bf9\u8c61\u5f15\u7528\uff0c\u9632\u6b62\u5185\u5b58\u6cc4\u6f0f\n        $user = null;\n\n        return new Response(200);\n    }\n}<\/pre>\n<p>\u901a\u8fc7\u8fd0\u884c Blackfire \u5206\u6790\uff0c\u53ef\u4ee5\u8bc6\u522b\u51fa\u63a7\u5236\u5668\u5728\u4e0d\u53ca\u65f6\u6e05\u9664\u7528\u6237\u5bf9\u8c61\u5f15\u7528\u65f6\u4f1a\u53d1\u751f\u5185\u5b58\u6cc4\u6f0f\u3002<\/p>\n<p><strong>Xdebug<\/strong><\/p>\n<p>Xdebug \u662f\u4e00\u4e2a\u5f3a\u5927\u7684\u8c03\u8bd5\u5de5\u5177\uff0c\u53ef\u4ee5\u63d0\u4f9b\u51fd\u6570\u6267\u884c\u65f6\u95f4\u548c\u51fd\u6570\u8c03\u7528\u5806\u6808\u7684\u4fe1\u606f\u3002\u5b83\u5141\u8bb8\u5728 IDE \u4e2d\u8fdb\u884c\u7ec6\u7c92\u5ea6\u7684\u8c03\u8bd5\u3002<\/p>\n<p><strong>\u5b9e\u6218\u6848\u4f8b\uff1a<\/strong>\u8c03\u8bd5 Zend Framework \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u6027\u80fd\u95ee\u9898<\/p>\n<p>Xdebug \u53ef\u4ee5\u5e2e\u52a9\u8c03\u8bd5 Zend Framework \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u6027\u80fd\u95ee\u9898\uff0c\u4f8b\u5982\u8def\u7531\u89e3\u6790\u3002<\/p>\n<pre>use Zend\\Mvc\\Router\\RouteStack;\nuse Zend\\Mvc\\Router\\PluginManager;\nuse Xdebug\\Debugger;\n\nclass RouteDebugExample\n{\n    public function testRoutes()\n    {\n        \/\/ \u6253\u5f00 Xdebug \u8ddf\u8e2a\n        Debugger::start();\n\n        \/\/ \u521b\u5efa\u8def\u7531\u5806\u6808\n        $routerStack = new RouteStack();\n\n        \/\/ \u521b\u5efa\u63d2\u4ef6\u7ba1\u7406\u5668\n        $pluginManager = new PluginManager();\n\n        \/\/ \u6ce8\u518c\u8def\u7531\u63d2\u4ef6\n        $routerStack-&gt;setPluginManager($pluginManager);\n\n        \/\/ \u6267\u884c\u8def\u7531\u89e3\u6790\uff0c\u5e76\u8f93\u51fa\u6267\u884c\u65f6\u95f4\u548c\u5176\u4ed6\u8c03\u8bd5\u4fe1\u606f\n        $result = $routerStack-&gt;match('\/foo\/bar');\n        echo Debugger::dump($result);\n    }\n}<\/pre>\n<p>\u901a\u8fc7\u542f\u7528 Xdebug \u8ddf\u8e2a\uff0c\u53ef\u4ee5\u8f7b\u677e\u8bc6\u522b\u8def\u7531\u89e3\u6790\u4e2d\u7684\u8017\u65f6\u6b65\u9aa4\u5e76\u91c7\u53d6\u63aa\u65bd\u8fdb\u884c\u4f18\u5316\u3002<\/p>\n<p><strong>\u9009\u62e9\u5408\u9002\u7684\u5de5\u5177<\/strong><\/p>\n<p>\u9009\u62e9\u54ea\u79cd\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u53d6\u51b3\u4e8e\u5e94\u7528\u7a0b\u5e8f\u7684\u5177\u4f53\u9700\u6c42\u3002PHPStan \u5bf9\u4e8e\u9759\u6001\u5206\u6790\u548c\u6027\u80fd\u4f30\u8ba1\u5f88\u6709\u7528\uff0cBlackfire \u63d0\u4f9b\u8be6\u7ec6\u7684\u6027\u80fd\u62a5\u544a\uff0c\u800c Xdebug \u975e\u5e38\u9002\u5408\u7ec6\u7c92\u5ea6\u7684\u8c03\u8bd5\u3002\u901a\u8fc7\u5229\u7528\u8fd9\u4e9b\u5de5\u5177\uff0c\u5f00\u53d1\u4eba\u5458\u53ef\u4ee5\u8bc6\u522b\u548c\u89e3\u51b3 PHP \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u6027\u80fd\u95ee\u9898\uff0c\u4ece\u800c\u63d0\u9ad8\u6574\u4f53\u6027\u80fd\u3002<\/p>\n<p>\u672c\u7bc7\u5173\u4e8e\u300aphp\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab\u300b\u7684\u4ecb\u7ecd\u5c31\u5230\u6b64\u7ed3\u675f\u5566\uff0c\u4f46\u662f\u5b66\u65e0\u6b62\u5883\uff0c\u60f3\u8981\u4e86\u89e3\u5b66\u4e60\u66f4\u591a\u5173\u4e8e\u6587\u7ae0\u7684\u76f8\u5173\u77e5\u8bc6\uff0c\u8bf7\u5173\u6ce8\u7c73\u4e91\u516c\u4f17\u53f7\uff01<\/p>\n","protected":false},"excerpt":{"rendered":"<p>php\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab \u54c8\u55bd\uff01\u4eca\u5929\u5fc3\u8840\u6765\u6f6e\u7ed9\u5927\u5bb6\u5e26\u6765\u4e86\u300aphp\u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u4f7f\u7528\u6848\u4f8b\u5206\u4eab\u300b\uff0c\u60f3\u5fc5\u5927\u5bb6\u5e94\u8be5\u5bf9\u6587\u7ae0\u90fd\u4e0d\u964c\u751f\u5427\uff0c\u90a3\u4e48\u9605\u8bfb\u672c\u6587\u5c31\u90fd\u4e0d\u4f1a\u5f88\u56f0\u96be\uff0c\u4ee5\u4e0b\u5185\u5bb9\u4e3b\u8981\u6d89\u53ca\u5230\uff0c\u82e5\u662f\u4f60\u6b63\u5728\u5b66\u4e60\u6587\u7ae0\uff0c\u5343\u4e07\u522b\u9519\u8fc7\u8fd9\u7bc7\u6587\u7ae0~\u5e0c\u671b\u80fd\u5e2e\u52a9\u5230\u4f60\uff01 \u672c\u6587\u4ecb\u7ecd\u4e86\u4e09\u79cd\u6d41\u884c\u7684 PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\uff1aPHPStan\uff1a\u9759\u6001\u5206\u6790\uff0c\u83b7\u53d6\u51fd\u6570\u6267\u884c\u65f6\u95f4\u4f30\u8ba1Blackfire\uff1a\u4ea4\u4e92\u5f0f\u706b\u7130\u56fe\uff0c\u8be6\u7ec6\u6027\u80fd\u62a5\u544a\uff0c\u53ef\u68c0\u6d4b\u5185\u5b58\u6cc4\u6f0fXdebug\uff1a\u7ec6\u7c92\u5ea6\u8c03\u8bd5\uff0c\u63d0\u4f9b\u51fd\u6570\u8c03\u7528\u5806\u6808\u4fe1\u606f PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\u4ecb\u7ecd\uff1a\u5b9e\u6218\u6848\u4f8b\u5206\u4eab \u51fd\u6570\u6027\u80fd\u5206\u6790\u5bf9\u4e8e\u4f18\u5316 PHP \u5e94\u7528\u7a0b\u5e8f\u81f3\u5173\u91cd\u8981\u3002\u672c\u6587\u4ecb\u7ecd\u4e86\u51e0\u79cd\u6d41\u884c\u7684 PHP \u51fd\u6570\u6027\u80fd\u5206\u6790\u5de5\u5177\uff0c\u5e76\u901a\u8fc7\u5b9e\u9645\u6848\u4f8b\u5206\u4eab\u5176\u4f7f\u7528\u3002 PHPStan PHPStan \u662f\u4e00\u4e2a\u9759\u6001\u5206\u6790\u5de5\u5177\uff0c\u53ef\u4ee5\u68c0\u6d4b\u4ee3\u7801\u4e2d\u7684\u9519\u8bef\u5e76\u5bf9\u51fd\u6570\u6267\u884c\u65f6\u95f4\u8fdb\u884c\u4f30\u8ba1\u3002\u5b83\u652f\u6301\u7c7b\u578b\u58f0\u660e\u5e76\u63d0\u4f9b\u8be6\u7ec6\u7684\u6027\u80fd\u62a5\u544a\u3002 \u5b9e\u6218\u6848\u4f8b\uff1a\u4f18\u5316 laravel \u5e94\u7528\u7a0b\u5e8f\u4e2d\u7684\u67e5\u8be2\u6027\u80fd \u4f7f\u7528 PHPStan\uff0c\u53ef\u4ee5\u8f7b\u677e\u8bc6\u522b\u9700\u8981\u4f18\u5316\u7684\u9ad8\u8017\u65f6\u67e5\u8be2\u3002 use PHPStan\\Analyser\\Analyser; use PHPStan\\Rules\\Rule; use PHPStan\\Testing\\RuleTestCase; class SlowQueryRuleTest extends RuleTestCase { protected function getRules(): array { return [ new SlowQueryRule() ]; } public function testSlowQuery() { $this-&gt;analyse( [ &#8216;&lt;?php&#8217;, &#8216;class QueryExample&#8217;, &#8216;{&#8216;, &#8216; public function [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[16],"tags":[],"class_list":["post-47457","post","type-post","status-publish","format-standard","hentry","category-16"],"_links":{"self":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts\/47457","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/comments?post=47457"}],"version-history":[{"count":0,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts\/47457\/revisions"}],"wp:attachment":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/media?parent=47457"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/categories?post=47457"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/tags?post=47457"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}