当前位置:首页 > 文章列表 > 文章 > php教程 > PHP获取CPU内存使用方法全解析

PHP获取CPU内存使用方法全解析

2025-10-12 12:26:27 0浏览 收藏

欢迎各位小伙伴来到golang学习网,相聚于此都是缘哈哈哈!今天我给大家带来《PHP获取CPU内存使用方法详解》,这篇文章主要讲到等等知识,如果你对文章相关的知识非常感兴趣或者正在自学,都可以关注我,我会持续更新相关文章!当然,有什么建议也欢迎在评论留言提出!一起学习!

PHP获取CPU和内存使用情况需借助系统命令或读取/proc文件,常用exec()或shell_exec()执行top、free等命令并解析输出,也可通过sys_getloadavg()获取负载、memory_get_usage()获取脚本内存;但执行外部命令有性能开销和安全风险,如命令注入、权限提升等;更优方案是读取Linux的/proc/meminfo和/proc/stat文件以减少进程开销,或使用APM工具如Datadog、New Relic等专业监控代理实现高效、安全的资源监控。

php如何获取CPU和内存使用情况?PHP系统资源监控与获取

PHP要获取CPU和内存使用情况,通常不会直接通过内置函数一步到位,因为PHP本身是一个应用层语言,它更多是与Web服务器交互,而不是直接操作系统底层。我们通常需要借助PHP执行系统命令的能力(如exec()shell_exec())来调用操作系统的工具,或者利用一些特殊的PHP扩展,甚至是通过读取特定的系统文件(如Linux下的/proc文件系统)来间接获取这些信息。这有点像让一个厨师去直接测量农田的土壤成分,虽然不是不可能,但总觉得有点绕远路,且需要一些额外的工具和技巧。

解决方案

要获取系统级的CPU和内存使用情况,我们最常用的方法就是利用PHP的exec()shell_exec()函数来执行操作系统提供的命令行工具。这玩意儿,用起来确实方便,但代价也不小,后面我们会聊到它的性能和安全考量。

1. 获取CPU使用情况:

说实话,直接获取一个精确到百分比的“当前CPU使用率”对PHP脚本来说是比较棘手的,因为CPU使用率是一个动态的、持续变化的指标,而且通常由操作系统内核负责统计和调度。PHP脚本在某个瞬间执行一个命令,只能得到那个瞬间的快照,或者一段时间内的平均值。

一个常见的做法是执行Linux系统下的top命令或者mpstat(如果安装了sysstat包)。

<?php
// 获取CPU使用率(Linux为例,需要解析top命令的输出)
function getCpuUsage() {
    $output = shell_exec("top -bn1 | grep 'Cpu(s)'");
    // 解析输出,例如:Cpu(s): 0.3% us, 0.3% sy, 0.0% ni, 99.3% id, 0.0% wa, 0.0% hi, 0.0% si, 0.0% st
    if (preg_match('/Cpu\(s\):\s*([\d.]+)\%\s*us,.*([\d.]+)\%\s*sy,.*([\d.]+)\%\s*id/', $output, $matches)) {
        $user_cpu = floatval($matches[1]); // 用户空间占用
        $system_cpu = floatval($matches[2]); // 内核空间占用
        $idle_cpu = floatval($matches[3]); // 空闲CPU
        $total_usage = $user_cpu + $system_cpu; // 总使用率(不包含nice, io wait等)
        return [
            'user' => $user_cpu,
            'system' => $system_cpu,
            'idle' => $idle_cpu,
            'total_usage' => $total_usage
        ];
    }
    return false;
}

// 获取系统平均负载(load average),这与CPU使用率不同,但也是一个重要的性能指标
// sys_getloadavg() 是PHP内置函数,更安全高效
function getSystemLoadAverage() {
    return sys_getloadavg(); // 返回一个包含1分钟、5分钟、15分钟平均负载的数组
}

// 示例调用
$cpuInfo = getCpuUsage();
if ($cpuInfo) {
    echo "CPU 用户空间使用率: " . $cpuInfo['user'] . "%\n";
    echo "CPU 内核空间使用率: " . $cpuInfo['system'] . "%\n";
    echo "CPU 总使用率: " . $cpuInfo['total_usage'] . "%\n";
} else {
    echo "无法获取CPU使用率。\n";
}

$loadAvg = getSystemLoadAverage();
echo "系统平均负载 (1min, 5min, 15min): " . implode(', ', $loadAvg) . "\n";
?>

这里有个小小的陷阱,sys_getloadavg()获取的是系统平均负载(load average),它表示的是在特定时间段内,系统处于可运行或不可中断状态的进程数量。这与CPU使用率是两个不同的概念,高负载不一定意味着CPU使用率高,但通常是CPU瓶颈的早期信号。

2. 获取内存使用情况:

内存使用情况可以分为两个层面:PHP脚本自身的内存使用和整个系统的内存使用。

  • PHP脚本自身内存使用: PHP提供了内置函数来获取当前脚本的内存消耗,这对于调试和优化PHP应用非常有用。

    <?php
    // 获取当前脚本已分配的内存
    echo "当前脚本内存使用: " . round(memory_get_usage() / (1024 * 1024), 2) . " MB\n";
    
    // 获取当前脚本在执行期间达到的内存峰值
    echo "当前脚本内存峰值: " . round(memory_get_peak_usage() / (1024 * 1024), 2) . " MB\n";
    ?>
  • 系统总内存使用: 与CPU类似,我们需要调用系统命令。在Linux下,free -mcat /proc/meminfo是获取系统内存信息的利器。

    <?php
    // 获取系统内存使用情况(Linux为例,解析free -m命令)
    function getSystemMemoryUsage() {
        $output = shell_exec("free -m");
        // 解析输出,例如:
        //               total        used        free      shared  buff/cache   available
        // Mem:           7983        2045        3000         400        2937        5300
        // Swap:          2047           0        2047
        if (preg_match('/Mem:\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)/', $output, $matches)) {
            return [
                'total_mb' => intval($matches[1]),
                'used_mb' => intval($matches[2]),
                'free_mb' => intval($matches[3]),
                'shared_mb' => intval($matches[4]),
                'buff_cache_mb' => intval($matches[5]),
                'available_mb' => intval($matches[6])
            ];
        }
        return false;
    }
    
    // 示例调用
    $memInfo = getSystemMemoryUsage();
    if ($memInfo) {
        echo "系统总内存: " . $memInfo['total_mb'] . " MB\n";
        echo "系统已用内存: " . $memInfo['used_mb'] . " MB\n";
        echo "系统空闲内存: " . $memInfo['free_mb'] . " MB\n";
        echo "系统可用内存: " . $memInfo['available_mb'] . " MB (包含buff/cache中可回收的部分)\n";
    } else {
        echo "无法获取系统内存使用情况。\n";
    }
    ?>

    对于Windows系统,情况会稍微复杂一些,可能需要使用wmic命令或PowerShell脚本,然后通过PHP来执行并解析。不过在Web服务器环境中,Linux更为常见。

PHP获取系统资源时常见的性能瓶颈与安全风险有哪些?

利用exec()shell_exec()这类函数来获取系统资源,虽然直接有效,但就像我前面说的,它其实是个两难的选择,伴随着不容忽视的性能开销和潜在的安全风险。我的经验是,任何时候涉及到执行外部命令,都得格外小心。

性能瓶颈:

  1. 进程创建开销: 每次调用exec()shell_exec(),操作系统都需要创建一个新的进程来执行你指定的命令。这个过程本身就需要消耗CPU和内存资源。如果你的PHP脚本频繁地调用这些命令,比如在一个高并发的Web请求中,那么这些额外的进程创建开销就会迅速累积,导致服务器负载飙升,响应时间变长。
  2. I/O开销: 命令执行的结果通常通过标准输出返回给PHP,这涉及到进程间的I/O操作。如果命令的输出内容很大,或者需要复杂的管道操作,那么I/O的消耗也会增加。
  3. 解析开销: PHP需要解析命令的文本输出,将其转换成有用的数据结构。这个解析过程,尤其是涉及到正则表达式匹配时,本身也是一个CPU密集型的操作。
  4. 实时性与准确性: 就像前面提到的,CPU使用率是一个非常动态的指标。你通过exec()获取的只是一个瞬间的快照。如果需要高频、实时的监控,这种方式的性能开销会变得无法承受。

安全风险:

  1. 命令注入: 这是最严重也是最常见的安全漏洞。如果你允许用户输入的数据作为命令的一部分,而没有进行严格的过滤和验证,攻击者就可以注入恶意的命令。例如,如果你的代码是shell_exec("ls " . $_GET['dir']),攻击者可以传递dir=; rm -rf /,后果不堪设想。
  2. 权限提升: PHP进程通常运行在一个特定的用户下(如www-datanginx)。如果这个用户拥有执行某些敏感系统命令的权限,或者这些命令本身存在漏洞,攻击者就可能利用命令注入来提升权限,甚至完全控制服务器。
  3. 信息泄露: 某些系统命令可能会输出敏感的系统信息,如果这些信息被攻击者获取,可能成为进一步攻击的跳板。
  4. 服务拒绝(DoS): 攻击者可以利用命令注入执行耗时或资源密集型的命令,从而耗尽服务器资源,导致服务不可用。

为了规避这些风险,我的建议是:

  • 永远不要直接拼接用户输入到系统命令中。 必须使用escapeshellarg()escapeshellcmd()函数对参数进行严格转义。
  • 最小化权限。 PHP运行的用户应该只拥有执行必要命令的最小权限。
  • 考虑替代方案。 如果可以,尽量避免直接执行系统命令。

除了直接执行系统命令,PHP还有哪些更优雅的资源监控方案?

虽然exec()很方便,但它的局限性和风险让我们不得不去寻找更“优雅”的解决方案。在我看来,更优雅的方案通常意味着更好的性能、更高的安全性、更强的可维护性,并且能够更好地融入现代的监控体系。

  1. 使用/proc文件系统(仅限Linux): Linux系统提供了一个虚拟文件系统/proc,它以文件的形式提供了内核和进程的信息。例如,/proc/meminfo包含了系统内存的详细信息,/proc/stat包含了CPU的统计数据。直接读取这些文件比执行外部命令要高效得多,因为它避免了进程创建的开销。

    <?php
    // 读取/proc/meminfo 获取内存信息
    function getMemInfoFromProc() {
        if (!file_exists('/proc/meminfo')) {
            return false;
        }
        $lines = file('/proc/meminfo');
        $memInfo = [];
        foreach ($lines as $line) {
            if (preg_match('/^(\w+):\s+(\d+)\s*kB/', $line, $matches)) {
                $memInfo[$matches[1]] = intval($matches[2]) / 1024; // 转换为MB
            }
        }
        // 计算一些常用的指标
        $total = $memInfo['MemTotal'] ?? 0;
        $free = $memInfo['MemFree'] ?? 0;
        $buffers = $memInfo['Buffers'] ?? 0;
        $cached = $memInfo['Cached'] ?? 0;
        $available = $memInfo['MemAvailable'] ?? ($free + $buffers + $cached); // MemAvailable在较新内核中才有
    
        return [
            'total_mb' => round($total, 2),
            'used_mb' => round($total - $available, 2),
            'free_mb' => round($free, 2),
            'available_mb' => round($available, 2)
        ];
    }
    
    // 获取CPU统计数据(需要两次采样计算)
    function getCpuStatFromProc() {
        if (!file_exists('/proc/stat')) {
            return false;
        }
        $lines = file('/proc/stat');
        foreach ($lines as $line) {
            if (str_starts_with($line, 'cpu ')) {
                $parts = explode(' ', $line);
                // user, nice, system, idle, iowait, irq, softirq, steal, guest, guest_nice
                return [
                    'user' => intval($parts[2]),
                    'nice' => intval($parts[3]),
                    'system' => intval($parts[4]),
                    'idle' => intval($parts[5]),
                    'iowait' => intval($parts[6]),
                    'irq' => intval($parts[7]),
                    'softirq' => intval($parts[8]),
                    'steal' => intval($parts[9])
                ];
            }
        }
        return false;
    }
    
    // 要计算CPU使用率,需要两次采样
    // 第一次采样
    $stat1 = getCpuStatFromProc();
    if ($stat1) {
        // 等待一小段时间(比如1秒)
        sleep(1);
        // 第二次采样
        $stat2 = getCpuStatFromProc();
    
        if ($stat2) {
            $total_delta = ($stat2['user'] + $stat2['nice'] + $stat2['system'] + $stat2['idle'] + $stat2['iowait'] + $stat2['irq'] + $stat2['softirq'] + $stat2['steal']) -
                           ($stat1['user'] + $stat1['nice'] + $stat1['system'] + $stat1['idle'] + $stat1['iowait'] + $stat1['irq'] + $stat1['softirq'] + $stat1['steal']);
            $idle_delta = $stat2['idle'] - $stat1['idle'];
    
            if ($total_delta > 0) {
                $cpu_usage = 100 * (1 - $idle_delta / $total_delta);
                echo "CPU 使用率 (通过/proc/stat计算): " . round($cpu_usage, 2) . "%\n";
            }
        }
    }
    
    $memInfoProc = getMemInfoFromProc();
    if ($memInfoProc) {
        echo "系统总内存 (通过/proc/meminfo): " . $memInfoProc['total_mb'] . " MB\n";
        echo "系统已用内存 (通过/proc/meminfo): " . $memInfoProc['used_mb'] . " MB\n";
        echo "系统可用内存 (通过/proc/meminfo): " . $memInfoProc['available_mb'] . " MB\n";
    }
    ?>

    这种方式虽然更高效,但代码实现起来更复杂,且仅限于Linux系统。

  2. 使用专门的监控代理/APM工具: 这是企业级应用中最推荐的方式。许多专业的应用性能监控(APM)工具,如New Relic, Datadog, Sentry, Prometheus等,都提供了PHP代理(Agent)。这些代理通常以PHP扩展的形式运行,能够深度集成到PHP-FPM或Web服务器中,直接从操作系统或PHP运行时获取各种指标(包括CPU、内存、网络、磁盘I/O等),并将数据发送到监控平台进行存储、分析和可视化。这种方式的优点是:

    • 低开销: 代理通常用C/C++编写,性能优化得很好。
    • 全面性: 不仅能监控系统资源,还能监控PHP应用本身的性能瓶颈(如函数调用时间、数据库查询、HTTP请求等)。
    • 易于集成: 一旦安装配置好,几乎不需要改动PHP代码。
    • 可视化与告警: 监控平台通常提供强大的仪表盘和告警功能。
  3. 构建一个独立的监控服务: 如果你不想引入大型APM工具,但又需要更灵活的监控,可以考虑构建一个独立的

好了,本文到此结束,带大家了解了《PHP获取CPU内存使用方法全解析》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!

PX与CM转换公式及实用技巧PX与CM转换公式及实用技巧
上一篇
PX与CM转换公式及实用技巧
PHP生成唯一有序数组对方法详解
下一篇
PHP生成唯一有序数组对方法详解
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ChatExcel酷表:告别Excel难题,北大团队AI助手助您轻松处理数据
    ChatExcel酷表
    ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
    3193次使用
  • Any绘本:开源免费AI绘本创作工具深度解析
    Any绘本
    探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
    3406次使用
  • 可赞AI:AI驱动办公可视化智能工具,一键高效生成文档图表脑图
    可赞AI
    可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
    3436次使用
  • 星月写作:AI网文创作神器,助力爆款小说速成
    星月写作
    星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
    4543次使用
  • MagicLight.ai:叙事驱动AI动画视频创作平台 | 高效生成专业级故事动画
    MagicLight
    MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
    3814次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码