Ric*_*hez 2 javascript php ajax jquery asynchronous
我正在开发基于 PHP 的 Web 应用程序(我没有构建)。
我正在运行这个ajax请求:
$.ajax({
type: 'POST',
url: "/potato/ajax.php?module=test_module",
dataType: 'json',
async: true,
data: {
start_ts: that.start_date,
stop_ts: that.end_date,
submitted: true
},
beforeSend: function() {
console.log('Start: ' + new Date().toLocaleString());
// Show Chart Loading
that.qwChart.showLoading({
color: '#00b0f0',
// text: that.returnNumWithPrecent(that.progress)
text: that.qwChartProgress
});
// If data div isn't displayed
if (!that.dataDisplayed) {
// Show divs loading
that.showMainDiv();
} else {
that.$qwTbody.slideUp('fast');
that.$qwTbody.html('');
}
},
complete: function(){},
success: function(result){
console.log('End: ' + new Date().toLocaleString());
// Clear timer
clearInterval(timer);
// Set progressbar to 100%
that.setProgressBarTo100();
// Show Download Button
that.downloadBtn.style.display = 'inline-block';
// Insert Chart Data
that.insertChartData(result);
// Insert Table Data
that.insertTableData(result);
}
});
Run Code Online (Sandbox Code Playgroud)
出于某种原因,它让我的整个网络应用程序卡住,直到它返回数据。我知道默认情况下 ajax 请求设置为“true”,但我还是添加了它以确保它是。如果它是异步的,它应该可以在不卡住我的网络应用程序的情况下完成工作,对吗?可能是什么问题?这是服务器端问题吗?我如何调试这种情况?
编辑:说“卡住”我的意思是 - 当我在提交 ajax 调用后等待响应时,刷新页面或并行打开其他页面(仅在我的 Web 应用程序中)显示一个白色的加载屏幕。每当 ajax 调用返回数据时 - 白页加载到请求的页面。
数据从 PHP 文件返回:
<?php
require_once("/www/common/api/db.php");
if (!empty($_POST['submitted'])) {
// error_reporting(-1);
// Users Array:
$users = get_qw_data($start_ts_to_date, $stop_ts_to_date);
// Summary Array:
$summary = get_qw_summary($users);
// QW Score Array:
$qws = get_qw_score($users);
// Generate CSV Report files
/* Remove old:*/
if (!is_file_dir_exist($customer))
create_qw_directory($customer);
/* Report #1: */ users_apps_google_macros_ma($users['users'], $customer);
/* Report #2: */ usage_and_qw_summary($summary, $customer);
/* Report #3: */ qw_score($qws, $customer);
/* Zip Files: */ zip_qw_files($customer);
echo json_encode($qws);
}
Run Code Online (Sandbox Code Playgroud)
PHP 会话是其他请求“卡住”的主要候选者,因为会话文件被写锁定,所以只要一个正在运行的脚本实例打开会话,所有其他请求都必须等待。
解决办法是尽快打电话session_write_close。
一点扩展的解释:
会话数据的默认存储机制就是文件系统。对于每个活动会话,PHP 只需将一个文件放入配置的会话目录中,并将 $_SESSION 的内容写入其中,以便在下一个需要访问它的请求时从那里读回。
现在,如果几个 PHP 脚本实例试图“同时”将更改的会话数据写入该文件,那显然会产生很大的冲突/错误。
因此,PHP 对会话文件设置了写锁,一旦一个脚本实例访问会话 - 其他所有人,其他请求(对同一脚本,或另一个也使用会话的请求),将不得不等待,直到第一个脚本与会话一起完成,并再次释放写锁。
默认情况下,当脚本运行完成时会发生这种情况。但是,如果您有更长的脚本运行时间,这很容易导致您在这里遇到的“阻塞”效果。解决方案是明确地告诉 PHP(通过session_write_close),“我已经完成了这里的会话,从现在开始不会向它写入任何新的/更改的数据 - 所以随意释放锁,以便下一个脚本可以开始读取会话数据。”
重要的是,您只能在脚本处理完任何会话数据后执行此操作。您仍然可以在脚本的其余部分从 $_SESSION 中读取- 但您不能再写入它了。(因此$_SESSION['foo'] = 'bar';,在您发布会话后,任何类似的操作都必须失败。)
如果会话此时(在此特定脚本中)服务的唯一目的是检查用户身份验证,那么您可以在此之后直接关闭会话。脚本的其余部分可以随心所欲地运行,而不会阻止其他脚本再访问同一个会话。
这不仅限于 AJAX 请求 - 这些只是您通常首先注意到此类内容的地方之一,否则您通常不会有那么多使用“并行”运行的会话的请求。但是,如果您要在多个浏览器选项卡中多次打开一个长时间运行的脚本,您会注意到相同的效果 - 在第一个选项卡中,脚本将运行并执行其业务,而在以下选项卡中,您应该注意到那些请求也会“挂起”,只要前一个脚本实例持有会话的写锁。