摘要:下面這四個重新替換封裝非常重要,具體的方法作用已經(jīng)在下面的代碼注釋中寫明了。使輸出的性能數(shù)據(jù)中添加數(shù)據(jù)。我們這里不考慮和內(nèi)存,那么發(fā)現(xiàn)給每隔設(shè)置了一個的起始時鐘周期。
XHProf 簡要概念
重新封裝zend的原生方法
如果要檢測CPU的話,會有5ms的延遲,因?yàn)樾枰嬎鉩pu頻率
內(nèi)部使用了鏈表
源碼地址:/root/Downloads/xhprof/extension/xhprof.c
最重要的兩個結(jié)構(gòu)體/* Xhprof"s global state. * * This structure is instantiated once. Initialize defaults for attributes in * 這個結(jié)構(gòu)體只初始化一次 * hp_init_profiler_state() Cleanup/free attributes in * hp_clean_profiler_state() */ typedef struct hp_global_t { /* ---------- Global attributes: ----------- */ /* Indicates if xhprof is currently enabled 是否當(dāng)前可用 */ int enabled; /* Indicates if xhprof was ever enabled during this request 在本次請求過程中是否其用過xhprof */ int ever_enabled; /* Holds all the xhprof statistics */ zval *stats_count; /* Indicates the current xhprof mode or level 當(dāng)前的運(yùn)行模式和等級*/ int profiler_level; /* Top of the profile stack 堆棧中的第一個*/ hp_entry_t *entries; /* freelist of hp_entry_t chunks for reuse... */ hp_entry_t *entry_free_list; /* Callbacks for various xhprof modes 代表不同模式的回調(diào)么?*/ hp_mode_cb mode_cb; /* ---------- Mode specific attributes: ----------- */ /* Global to track the time of the last sample in time and ticks */ struct timeval last_sample_time; uint64 last_sample_tsc; /* XHPROF_SAMPLING_INTERVAL in ticks */ uint64 sampling_interval_tsc; /* This array is used to store cpu frequencies for all available logical * cpus. For now, we assume the cpu frequencies will not change for power * saving or other reasons. If we need to worry about that in the future, we * can use a periodical timer to re-calculate this arrary every once in a * while (for example, every 1 or 5 seconds). 處理器的執(zhí)行頻率?*/ double *cpu_frequencies; /* The number of logical CPUs this machine has. 邏輯cpu的數(shù)量*/ uint32 cpu_num; /* The saved cpu affinity. */ cpu_set_t prev_mask; /* The cpu id current process is bound to. (default 0) 當(dāng)前進(jìn)程在的處理器的id*/ uint32 cur_cpu_id; /* XHProf flags */ uint32 xhprof_flags; /* counter table indexed by hash value of function names. 方法的調(diào)用次數(shù)的表*/ uint8 func_hash_counters[256]; /* Table of ignored function names and their filter 忽略統(tǒng)計的方法的表格*/ char **ignored_function_names; uint8 ignored_function_filter[XHPROF_IGNORED_FUNCTION_FILTER_SIZE]; } hp_global_t;
typedef struct hp_entry_t { char *name_hprof; /* function name 方法名稱*/ int rlvl_hprof; /* recursion level for function 方法的遞歸層級*/ uint64 tsc_start; /* start value for TSC counter 開始的時鐘周期*/ long int mu_start_hprof; /* memory usage 內(nèi)存使用量*/ long int pmu_start_hprof; /* peak memory usage 內(nèi)存使用峰值*/ struct rusage ru_start_hprof; /* user/sys time start */ struct hp_entry_t *prev_hprof; /* ptr to prev entry being profiled 指向上一個被分析的指針*/ uint8 hash_code; /* hash_code for the function name 每個方法名稱對應(yīng)的hash*/ } hp_entry_t;XHProf 在php中的使用
我們先看下XHProf的使用方法
save_run($data,"test"); // 我這里直接將可視化的鏈接地址打印了出來,方便調(diào)試 echo "執(zhí)行結(jié)果如下:(可以直接跳過結(jié)果,看下面,但是要記住有ct、wt這兩個值)
array(7) { ["test==>range"]=> array(2) { ["ct"]=> int(2) ["wt"]=> int(4463) } ["main()==>test"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(3069) } ["main()==>eval::/var/www/html/index2.php(9) : eval()"d code"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(16) } ["eval==>test"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(2614) } ["main()==>eval"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(2617) } ["main()==>xhprof_disable"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(0) } ["main()"]=> array(2) { ["ct"]=> int(1) ["wt"]=> int(5716) } }XHProf 源碼 xhprof_enable()首先我們來看xhprof_enable(),這個方法定義了要接受的三個參數(shù),并且將這三個參數(shù)分別傳遞給兩個方法使用,其中最重要的是hp_begin()
/** * Start XHProf profiling in hierarchical mode. * * @param long $flags flags for hierarchical mode * @return void * @author kannan */ PHP_FUNCTION(xhprof_enable) { long xhprof_flags = 0; /* XHProf flags */ zval *optional_array = NULL; /* optional array arg: for future use */ /* 獲取參數(shù)并且允許傳遞一個l 和z的可選參數(shù) 分別代表xhprof_flags 和 optional_array 關(guān)于TSRMLS_CC 可以看http://www.laruence.com/2008/08/03/201.html 另外關(guān)于zend_parse_parameters的返回值failure 代表參數(shù)的處理是否成功 */ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lz", &xhprof_flags, &optional_array) == FAILURE) { return; } /* 從參數(shù)中獲取需要被忽略的方法 參照手冊參數(shù)的說明 http://php.net/manual/zh/function.xhprof-enable.php */ hp_get_ignored_functions_from_arg(optional_array); hp_begin(XHPROF_MODE_HIERARCHICAL, xhprof_flags TSRMLS_CC); }hp_begin()這個方法看起來很長,但是世界上邏輯很簡單,主要是進(jìn)行了一些初始化。
下面一共進(jìn)行了四次replace,用來封裝zend的方法。下面這四個重新替換封裝非常重要,具體的方法作用已經(jīng)在下面的代碼注釋中寫明了。
zend_compile_file => hp_compile_file
zend_compile_string => hp_compile_string
zend_execute => hp_execute
zend_execute_internal => hp_execute_internal
/** * This function gets called once when xhprof gets enabled. * 這個方法在enable的時候調(diào)用一次 * It replaces all the functions like zend_execute, zend_execute_internal, * etc that needs to be instrumented with their corresponding proxies. * 他用來替換zend的一些需要被代理的方法意思就是xhprof劫持了原生方法 * hp_begin(XHPROF_MODE_HIERARCHICAL, xhprof_flags TSRMLS_CC); * * level 等級 * xhprof_flags 運(yùn)行方式 */ static void hp_begin(long level, long xhprof_flags TSRMLS_DC) { /* 如果xhprof 沒有開啟,也就是沒有調(diào)用enable方法,那么走這里買的邏輯,這個是通過hp_globals來判斷的, */ if (!hp_globals.enabled) { int hp_profile_flag = 1; hp_globals.enabled = 1; /* 這里修改了enbale狀態(tài),保證enable在整個請求過程中只會被第一次調(diào)用觸發(fā) */ hp_globals.xhprof_flags = (uint32)xhprof_flags; /* 格式化為32位的無符號整數(shù) */ /* 下面一共進(jìn)行了四次replace,用來封裝zend的方法 1. zend_compile_file => hp_compile_file zend_compile_file負(fù)責(zé)將要執(zhí)行的腳本文件編譯成由ZE的基本指令序列構(gòu)成的op codes , 然后將op codes交由zend_execute執(zhí)行,從而得到我們腳本的結(jié)果。 http://www.laruence.com/2008/08/14/250.html 2. zend_compile_string => hp_compile_string 這個是把php代碼編譯成為opcode的過程 http://www.phpchina.com/portal.php?mod=view&aid=40347 3. zend_execute => hp_execute zend_compile_file() zend_compile_file() is the wrapper for the lexer, parser, and code generator. It compiles a file and returns a zend_op_array. zend_execute() After a file is compiled, its zend_op_array is executed by zend_execute(). http://php.find-info.ru/php/016/ch23lev1sec2.html 4. zend_execute_internal => hp_execute_internal There is also a companion zend_execute_internal() function, which executes internal functions. */ /* Replace zend_compile with our proxy 先對其進(jìn)行了備份_,通過加入_下劃線的方式,然后使用hp_compile_file來替換*/ _zend_compile_file = zend_compile_file; zend_compile_file = hp_compile_file; /* Replace zend_compile_string with our proxy */ _zend_compile_string = zend_compile_string; zend_compile_string = hp_compile_string; /* Replace zend_execute with our proxy */ #if PHP_VERSION_ID < 50500 _zend_execute = zend_execute; zend_execute = hp_execute; #else _zend_execute_ex = zend_execute_ex; zend_execute_ex = hp_execute_ex; #endif /* Replace zend_execute_internal with our proxy */ _zend_execute_internal = zend_execute_internal; /* XHPROF_FLAGS_NO_BUILTINGS 是用來標(biāo)識,不需要統(tǒng)計內(nèi)置函數(shù)性能 通過位運(yùn)算&來判斷是否用戶傳遞的flags包含了NO_BUILTINGS 除此之外還包含一下三種flags 1. HPROF_FLAGS_NO_BUILTINS (integer) 使得跳過所有內(nèi)置(內(nèi)部)函數(shù)。 2. XHPROF_FLAGS_CPU (integer) 使輸出的性能數(shù)據(jù)中添加 CPU 數(shù)據(jù)。 3. XHPROF_FLAGS_MEMORY (integer) 使輸出的性能數(shù)據(jù)中添加內(nèi)存數(shù)據(jù)。 */ if (!(hp_globals.xhprof_flags & XHPROF_FLAGS_NO_BUILTINS)) { /* if NO_BUILTINS is not set (i.e. user wants to profile builtins), * then we intercept internal (builtin) function calls. * 如果沒有設(shè)置的話,那么就代表用戶想分析內(nèi)置函數(shù)性能,并且我們就會攔截內(nèi)置的方法請求 */ zend_execute_internal = hp_execute_internal; } /* Initialize with the dummy mode first Having these dummy callbacks saves * us from checking if any of the callbacks are NULL everywhere. * 首先來初始化一下這些方法,可以避免在回調(diào)方法為NULL的時候*/ hp_globals.mode_cb.init_cb = hp_mode_dummy_init_cb; hp_globals.mode_cb.exit_cb = hp_mode_dummy_exit_cb; hp_globals.mode_cb.begin_fn_cb = hp_mode_dummy_beginfn_cb; hp_globals.mode_cb.end_fn_cb = hp_mode_dummy_endfn_cb; /* Register the appropriate callback functions Override just a subset of * all the callbacks is OK. 根據(jù)不同的處理模式,簡單還是詳細(xì)*/ switch(level) { /* 一般都是使用的這個模式,所以我們專注看這個mode */ case XHPROF_MODE_HIERARCHICAL: hp_globals.mode_cb.begin_fn_cb = hp_mode_hier_beginfn_cb; hp_globals.mode_cb.end_fn_cb = hp_mode_hier_endfn_cb; break; case XHPROF_MODE_SAMPLED: hp_globals.mode_cb.init_cb = hp_mode_sampled_init_cb; hp_globals.mode_cb.begin_fn_cb = hp_mode_sampled_beginfn_cb; hp_globals.mode_cb.end_fn_cb = hp_mode_sampled_endfn_cb; break; } /* one time initializations 初始化分析器,內(nèi)部搞定了cpu頻率、initcb、可忽略的方法*/ hp_init_profiler_state(level TSRMLS_CC); /* start profiling from fictitious main() */ BEGIN_PROFILING(&hp_globals.entries, ROOT_SYMBOL, hp_profile_flag); } }hp_init_profiler_state()/** * Initialize profiler state * 初始化分析器狀態(tài) * * 這里最開始的時候傳遞進(jìn)來的level是XHPROF_MODE_HIERARCHICAL * * @author kannan, veeve */ void hp_init_profiler_state(int level TSRMLS_DC) { /* Setup globals */ if (!hp_globals.ever_enabled) { /* 如果之前沒有開啟過xhprof,那么將這個值初始化為1,現(xiàn)在就算開啟了 */ hp_globals.ever_enabled = 1; /* 堆棧的第一個設(shè)置空 */ hp_globals.entries = NULL; } /* 分析器的等級 */ hp_globals.profiler_level = (int) level; /* Init stats_count 初始化統(tǒng)計數(shù)量 */ if (hp_globals.stats_count) { /* 釋放這個內(nèi)存 */ zval_dtor(hp_globals.stats_count); /* 通知垃圾回收機(jī)制來回收這個內(nèi)存 */ FREE_ZVAL(hp_globals.stats_count); } /* 創(chuàng)建一個zval變量,并且初始化為數(shù)組 參考 http://www.cunmou.com/phpbook/8.3.md */ MAKE_STD_ZVAL(hp_globals.stats_count); array_init(hp_globals.stats_count); /* NOTE(cjiang): some fields such as cpu_frequencies take relatively longer * to initialize, (5 milisecond per logical cpu right now), therefore we * calculate them lazily. 一些字段初始化起來要花費(fèi)非常長的時間,那么我們要懶計算,就是放到后面計算*/ if (hp_globals.cpu_frequencies == NULL) { get_all_cpu_frequencies(); restore_cpu_affinity(&hp_globals.prev_mask); } /* bind to a random cpu so that we can use rdtsc instruction. 這里竟然是隨機(jī)綁定一個cpu*/ bind_to_cpu((int) (rand() % hp_globals.cpu_num)); /* Call current mode"s init cb 根據(jù)不同的模式,調(diào)用初始方法,看line:1933*/ hp_globals.mode_cb.init_cb(TSRMLS_C); /* Set up filter of functions which may be ignored during profiling 設(shè)置被過濾的方法*/ hp_ignored_functions_filter_init(); }get_cpu_frequency()在上面的方法中調(diào)用了一個get_all_cpu_frequencies(),這個方法內(nèi)部調(diào)用了一個get_cpu_frequency很有意思,因?yàn)檫@個方法將導(dǎo)致如果開啟CPU的檢測,那么會有5ms的延遲
/** * This is a microbenchmark to get cpu frequency the process is running on. The * returned value is used to convert TSC counter values to microseconds. * * @return double. * @author cjiang */ static double get_cpu_frequency() { struct timeval start; struct timeval end; /* gettimeofday 獲取當(dāng)前的時間,并且放到start中 */ if (gettimeofday(&start, 0)) { perror("gettimeofday"); return 0.0; } uint64 tsc_start = cycle_timer(); /* Sleep for 5 miliseconds. Comparaing with gettimeofday"s few microseconds * execution time, this should be enough. * 這個是為了獲取CPU的執(zhí)行頻率,用5000微秒的時間中cpu的執(zhí)行次數(shù),來得到每秒cpu能執(zhí)行的頻率 * TSC 自從啟動CPU開始記錄的時鐘周期 * */ usleep(5000); if (gettimeofday(&end, 0)) { perror("gettimeofday"); return 0.0; } uint64 tsc_end = cycle_timer(); /* 時鐘周期的數(shù)量除以微秒時間間隔的數(shù)量得到cpu頻率 */ return (tsc_end - tsc_start) * 1.0 / (get_us_interval(&start, &end)); }BEGIN_PROFILING 重要!這個就是分析的邏輯,他的要點(diǎn)在于生成了一個單項(xiàng)鏈表。
/* * Start profiling - called just before calling the actual function * 開始分析,只在正式方法調(diào)用之前要調(diào)用 * NOTE: PLEASE MAKE SURE TSRMLS_CC IS AVAILABLE IN THE CONTEXT * OF THE FUNCTION WHERE THIS MACRO IS CALLED. * TSRMLS_CC CAN BE MADE AVAILABLE VIA TSRMLS_DC IN THE * CALLING FUNCTION OR BY CALLING TSRMLS_FETCH() * TSRMLS_FETCH() IS RELATIVELY EXPENSIVE. * entries 這里傳遞進(jìn)來的是hp_entry_t的一個指向指針的地址 * 這個地方實(shí)際上生成的是一個單鏈表,都是用prev_hprof 來進(jìn)行關(guān)聯(lián) * * 這里do while(0) 是用來封裝宏的 * */ #define BEGIN_PROFILING(entries, symbol, profile_curr) do { /* Use a hash code to filter most of the string comparisons. */ uint8 hash_code = hp_inline_hash(symbol); /* 判斷這個方法是否是需要忽略的方法,如果不是需要被忽略的,那么進(jìn)行分析 */ profile_curr = !hp_ignore_entry(hash_code, symbol); if (profile_curr) { /* 返回一個指針(地址),開辟了一個內(nèi)存空間給cur_entry,包括了hash_code、方法名稱、堆棧指針 */ hp_entry_t *cur_entry = hp_fast_alloc_hprof_entry(); (cur_entry)->hash_code = hash_code; (cur_entry)->name_hprof = symbol; /* 這里的*entries 指向的是指針hp_global_t.entires 堆棧的首地址 */ (cur_entry)->prev_hprof = (*(entries)); /* Call the universal callback*/ hp_mode_common_beginfn((entries), (cur_entry) TSRMLS_CC); /* Call the mode"s beginfn callback 這個方法除卻cpu和mem 只是設(shè)置了tsc_Start */ hp_globals.mode_cb.begin_fn_cb((entries), (cur_entry) TSRMLS_CC); /* Update entries linked list */ (*(entries)) = (cur_entry); } } while (0)我們可以看上面的鏈表在生成的過程中,調(diào)用了 hp_globals.mode_cb.begin_fn_cb方法。我們這里不考慮CPU和內(nèi)存,那么發(fā)現(xiàn)給每隔current設(shè)置了一個tsc的起始時鐘周期。
/** * XHPROF_MODE_HIERARCHICAL"s begin function callback * * @author kannan */ void hp_mode_hier_beginfn_cb(hp_entry_t **entries, hp_entry_t *current TSRMLS_DC) { /* Get start tsc counter */ current->tsc_start = cycle_timer(); /* Get CPU usage 如果要計算cpu的話*/ if (hp_globals.xhprof_flags & XHPROF_FLAGS_CPU) { getrusage(RUSAGE_SELF, &(current->ru_start_hprof)); } /* Get memory usage 如果要計算內(nèi)存的話*/ if (hp_globals.xhprof_flags & XHPROF_FLAGS_MEMORY) { current->mu_start_hprof = zend_memory_usage(0 TSRMLS_CC); current->pmu_start_hprof = zend_memory_peak_usage(0 TSRMLS_CC); } }hp_execute 代碼執(zhí)行部分每次有代碼執(zhí)行的時候,都會走這個地方,這段代碼主要是在執(zhí)行zend_execute的前后,粉分別調(diào)用了BEGIN_PROFILING 和END_PROFILING
#if PHP_VERSION_ID < 50500 ZEND_DLEXPORT void hp_execute (zend_op_array *ops TSRMLS_DC) { #else ZEND_DLEXPORT void hp_execute_ex (zend_execute_data *execute_data TSRMLS_DC) { zend_op_array *ops = execute_data->op_array; #endif char *func = NULL; int hp_profile_flag = 1; func = hp_get_function_name(ops TSRMLS_CC); if (!func) { #if PHP_VERSION_ID < 50500 _zend_execute(ops TSRMLS_CC); #else _zend_execute_ex(execute_data TSRMLS_CC); #endif return; } BEGIN_PROFILING(&hp_globals.entries, func, hp_profile_flag); #if PHP_VERSION_ID < 50500 _zend_execute(ops TSRMLS_CC); #else _zend_execute_ex(execute_data TSRMLS_CC); #endif if (hp_globals.entries) { END_PROFILING(&hp_globals.entries, hp_profile_flag); } efree(func); }END_PROFILINGhp_globals.mode_cb.end_fn_cb((entries) TSRMLS_CC);這段代碼最終指向了hp_mode_hier_endfn_cb,這段代碼中主要構(gòu)成了一個"==>"數(shù)據(jù)格式,并且計算了每個方法的調(diào)用次數(shù)。
void hp_mode_hier_endfn_cb(hp_entry_t **entries TSRMLS_DC) { /* 整個堆棧的最后一個調(diào)用 */ hp_entry_t *top = (*entries); zval *counts; struct rusage ru_end; char symbol[SCRATCH_BUF_LEN]; long int mu_end; long int pmu_end; /* Get the stat array */ hp_get_function_stack(top, 2, symbol, sizeof(symbol)); if (!(counts = hp_mode_shared_endfn_cb(top, symbol TSRMLS_CC))) { return; } if (hp_globals.xhprof_flags & XHPROF_FLAGS_CPU) { /* Get CPU usage */ getrusage(RUSAGE_SELF, &ru_end); /* Bump CPU stats in the counts hashtable */ hp_inc_count(counts, "cpu", (get_us_interval(&(top->ru_start_hprof.ru_utime), &(ru_end.ru_utime)) + get_us_interval(&(top->ru_start_hprof.ru_stime), &(ru_end.ru_stime))) TSRMLS_CC); } if (hp_globals.xhprof_flags & XHPROF_FLAGS_MEMORY) { /* Get Memory usage */ mu_end = zend_memory_usage(0 TSRMLS_CC); pmu_end = zend_memory_peak_usage(0 TSRMLS_CC); /* Bump Memory stats in the counts hashtable */ hp_inc_count(counts, "mu", mu_end - top->mu_start_hprof TSRMLS_CC); hp_inc_count(counts, "pmu", pmu_end - top->pmu_start_hprof TSRMLS_CC); } }hp_mode_shared_endfn_cb這個方法統(tǒng)計了調(diào)用次數(shù)和消耗時間,實(shí)際上最終所有的數(shù)據(jù)都存儲在hp_entry_t所構(gòu)造的鏈表中
zval * hp_mode_shared_endfn_cb(hp_entry_t *top, char *symbol TSRMLS_DC) { zval *counts; uint64 tsc_end; /* Get end tsc counter */ tsc_end = cycle_timer(); /* Get the stat array */ if (!(counts = hp_hash_lookup(symbol TSRMLS_CC))) { return (zval *) 0; } /* Bump stats in the counts hashtable */ hp_inc_count(counts, "ct", 1 TSRMLS_CC); hp_inc_count(counts, "wt", get_us_from_tsc(tsc_end - top->tsc_start, hp_globals.cpu_frequencies[hp_globals.cur_cpu_id]) TSRMLS_CC); return counts; }
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/23171.html
摘要:一背景項(xiàng)目即將上線,想通過一些工具來分析代碼的穩(wěn)定性和效率,想起在上個團(tuán)隊(duì)時使用過的擴(kuò)展因?yàn)閾Q了新電腦,所以需要重新編譯此擴(kuò)展,現(xiàn)將安裝與實(shí)際排查過程完整記錄下來,方便自己回顧和幫助更多的讀者。作者湯青松微信日期 一、背景 項(xiàng)目即將上線,想通過一些工具來分析代碼的穩(wěn)定性和效率,想起在上個團(tuán)隊(duì)時使用過的xhprof擴(kuò)展;因?yàn)閾Q了新電腦,所以需要重新編譯此擴(kuò)展,現(xiàn)將安裝與實(shí)際排查過程完整記...
摘要:什么是開源的輕量級性能分析工具。它報告函數(shù)級別的請求次數(shù)和各種指標(biāo),包括阻塞時間,時間和內(nèi)存使用情況。基于瀏覽器的性能分析用戶界面能更容易查看,或是與同行們分享成果。對于本地開發(fā)環(huán)境來說,進(jìn)行性能分析是夠用了。 什么是 XHPROF? XHPROF:Facebook 開源的輕量級PHP性能分析工具。 它報告函數(shù)級別的請求次數(shù)和各種指標(biāo),包括阻塞時間,CPU時間和內(nèi)存使用情況。 XHPr...
摘要:性能分析此版本為第三方擴(kuò)展官房不支持目錄為擴(kuò)展源碼安狀擴(kuò)展即可編輯啟用擴(kuò)展性能分析數(shù)據(jù)文件存放位置需要用戶有可寫可讀權(quán)限對項(xiàng)目入口文件添加代碼在第一步后的文件夾里面生成數(shù)據(jù)文件后綴或者創(chuàng)建網(wǎng)占目錄為例在第一步后的文件夾里面訪問上面虛擬主機(jī) xhprof php性能分析 1.clone xhprof 此版本為github第三方擴(kuò)展 (php官房不支持 php 7) https://git...
摘要:是開發(fā)的一個測試性能的擴(kuò)展,本文記錄了在應(yīng)用中使用對進(jìn)行性能優(yōu)化,查找性能瓶頸的方法。函數(shù)用于停止性能分析,并返回分析的數(shù)據(jù)。該參數(shù)用于為剖析結(jié)果添加額外的信息,該參數(shù)的值使用以下宏,如果需要提供多個值,使用進(jìn)行分隔。 XHProf是facebook 開發(fā)的一個測試php性能的擴(kuò)展,本文記錄了在PHP應(yīng)用中使用XHProf對PHP進(jìn)行性能優(yōu)化,查找性能瓶頸的方法。 安裝Xhprof擴(kuò)展...
閱讀 2774·2021-11-17 09:33
閱讀 3106·2021-10-25 09:44
閱讀 1213·2021-10-11 10:59
閱讀 2407·2021-09-27 13:34
閱讀 2915·2021-09-07 10:19
閱讀 2141·2019-08-29 18:46
閱讀 1540·2019-08-29 12:55
閱讀 932·2019-08-23 17:11