- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个基于 apache、php、js/jquery 的网络应用程序。当加载文档时,应用程序的一个模块被初始化。该序列调用一个 init() 函数来执行许多任务,其中包括通过 ajax 获取两个 html 对话框并将它们插入到现有页面中。对这些对话框的引用保存为变量,因此我不必在整个脚本中指定 jquery 选择器,而可以简单地使用这些变量。这工作正常,除了在尝试将处理程序绑定(bind)到获取的对话框内的元素时变量“未定义”的情况很少(偶尔......)。这很奇怪,因为除了绑定(bind)之外,对话框在整个应用程序中都可用。这表明 ajax 调用成功,但显然存在某种竞争条件,因此有时仅在绑定(bind)尝试 之后才设置变量。
根据我的理解,这不应该发生,因为绑定(bind)是在 when() 的 .done() 部分完成的,并且只应在 when() 中对话框的 ajax 检索完成后执行。
显然我在这里遗漏了一些基本的东西。有人给我提示吗?
以下引用了工作实现中的代码摘录。它们的某些部分可能在语法上无效,但这只是由于删除了与此处无关的代码部分。未缩短的代码工作正常。
变量:
Shorty.Tracking.Dialog.List:{};
Shorty.Tracking.Dialog.Click:{};
初始化顺序:
$(window).load(function(){
//...
var dfd = new $.Deferred();
$.when(
// load layout of dialog to show the list of tracked clicks
Shorty.Tracking.init()
).done(function(){
// bind actions to basic buttons
Shorty.Tracking.Dialog.List.find('#close').on('click',function(){
// ...
});
// ...
});
// ...
})
简化的初始化函数:
Shorty.Tracking.init:function(){
// ...
var dfd=new $.Deferred();
// two dialogs are used by this plugin
var dialogs={
'list': Shorty.Tracking.Dialog.List,
'click': Shorty.Tracking.Dialog.Click
};
// load dialogs from server
$.each(['list','click'],function(i,dialog){
if (...){
// ...
dfd.resolve();
}else{
// load dialog layout via ajax and create a freshly populated dialog
$.ajax({
type: 'GET',
url: OC.filePath('shorty-tracking','ajax','layout.php'),
data: { dialog: dialog},
cache: false,
dataType: 'json'
}).pipe(
function(response){return Shorty.Ajax.eval(response)},
function(response){return Shorty.Ajax.fail(response)}
).done(function(response){
// create a fresh dialog
// insert it alongside the existing dialogs in the top controls bar
$('#controls').append(response.layout);
switch (dialog){
case 'list':
Shorty.Tracking.Dialog.List=$('#controls #shorty-tracking-list-dialog').first();
break;
case 'click':
Shorty.Tracking.Dialog.Click=$('#controls #shorty-tracking-click-dialog').first();
} // switch
dfd.resolve();
}).fail(dfd.reject)
} // else
}); // each
return dfd.promise();
},
通过 Bergi 的回答,我显然设法解决了原来的问题。到现在为止,我无法再检测到任何失败的绑定(bind)尝试。但是我无法完全遵循该建议:在他的回答中,他建议删除初始化方法中的 switch 语句以支持直接赋值。这当然要优雅得多,但那是行不通的。我的印象是我对 javascript 如何处理存储在变量中的引用和/或函数有一些误解。
也许您、Bergi 或其他人可以对此做出一些解释:
这是修改后的初始化方法:
init:function(){
if (Shorty.Debug) Shorty.Debug.log("initializing tracking list");
// check if dialogs already exist
if ( $.isEmptyObject(Shorty.Tracking.Dialog.List)
&& $.isEmptyObject(Shorty.Tracking.Dialog.Click) ){
// two dialogs are used by this plugin
var dialogs={
'list': Shorty.Tracking.Dialog.List,
'click': Shorty.Tracking.Dialog.Click
};
// load dialogs from server
var dfds=$.map(dialogs,function(obj,dialog){
// load dialog layout via ajax and append it to the collection of dialogs in the controls
return $.ajax({
type: 'GET',
url: OC.filePath('shorty-tracking','ajax','layout.php'),
data: { dialog: dialog},
cache: false,
dataType: 'json'
}).pipe(
function(response){return Shorty.Ajax.eval(response)},
function(response){return Shorty.Ajax.fail(response)}
).done(function(response){
// create a fresh dialog and insert it alongside the existing dialogs in the top controls bar
$('#controls').append(response.layout);
// obj=$('#controls #shorty-tracking-'+dialog+'-dialog').first();
switch(dialog){
case 'list':
Shorty.Tracking.Dialog.List=$('#controls #shorty-tracking-list-dialog').first();
break;
case 'click':
Shorty.Tracking.Dialog.Click=$('#controls #shorty-tracking-click-dialog').first();
break;
} // switch
})
}) // map
return $.when.apply(null, dfds);
}else{
// dialogs already loaded, just clean them for usage
Shorty.Tracking.Dialog.List.find('#list-of-clicks tbody tr').remove();
new Deferred().resolve();
} // else
},
在中间,您会看到赋值语句被注释掉,目前被下面的 switch 语句替换。我尝试了几种变体,比如 obj.element 等等,但都失败了。在函数外部,变量 Shorty.Tracking.Dialog.List 和 Shorty.Tracking.-Dialog.Click 保持为空。
我绝对是 web 东西 ans js/jquery 的新手。但我当然很想了解更多关于这种处理引用的方式。
最佳答案
这是有问题的代码:
var dfd=new $.Deferred();
$.each(['list','click'], function(){
...
dfd.resolve/fail();
});
您只创建了一个 Deferred,但多次解析它。因此,当第一个解析发生时,第二个 Ajax 不会完成。
因此,改为使用两个 Deferred,并通过 jQuery.when()
组合它们.另外,我认为您不需要自己创建 Deferred。只需使用从 pipe()
调用中获得的那两个。
function init() {
var dialogs={
'list': Shorty.Tracking.Dialog.List,
'click': Shorty.Tracking.Dialog.Click
};
var deferreds = $.map(dialogs, function(obj, dialog) {
return $.ajax(...).pipe(...).done(...
// really, don't use switch here:
obj.element = response.layout;
}); // done() returns the Deferred
});
return $.when.apply(null, deferreds);
}
...
$.ready(function($) {
init().done(...);
});
好的,我需要解释一下 switch 语句。的确,dialogs
对象中的值将从 [uninitialized] 变量分配并保持该值 (undefined
),当字段被重新分配给 Shorty
值不会改变。 Javascript 中没有“指针”。我认为只有在成员运算符中使用当前 dialog
变量而不是 switch 语句才有用。要分配给 Shorty.Tracking.Dialog.List
和 .Click
,您需要使用类似
var dialogs = ["list", "click"];
var shortcut = Shorty.Tracking.Dialog; // some object
for (var i=0; i<dialogs.length; i++) {
// loop over them,
// do the ajax stuff
var element = $(repsonse.layout); // get the added element from the response
$('#controls').append(element);
shortcut[dialogs[i]] = element;
}
(最终使用来自 How do I make the first letter of a string uppercase in JavaScript? 的片段)
关于javascript - jquery 中的竞争条件何时/完成序列?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11702141/
我正在阅读 Python 文档以真正深入了解 Python 语言,并遇到了 filter 和 map 函数。我以前使用过过滤器,但从未使用过映射,尽管我在 SO 上的各种 Python 问题中都见过这
当我尝试打印 BST 的级别顺序时,这个问题提示了我。 这是一个 Pre-Order Sequence: 4, 1, 2, 3, 5, 6, 7, 8 In_order Sequence : 1, 2
我的代码在 main(序列测试;)的第一行出现错误,指出它是对 sequence::sequence() 的 undefined reference 。我无法更改 main 中的代码。有谁知道我该如何
这可能很简单,但我在通常的 latex 指南中找不到任何相关内容。在这句话中: {\em hello\/} “\/”的目的是什么? 最佳答案 这就是所谓的斜体校正。其目的是确保斜体文本后有适当的间距。
当我从 Postgresql 表中删除所有记录,然后尝试重置序列以在插入时开始一个编号为 1 的新记录时,我得到不同的结果: SELECT setval('tblname_id_seq', (SELE
在版本10.0.3中,MariaDB引入了一种称为序列的存储引擎。 其ad hoc为操作生成整数序列,然后终止。 该序列包含正整数,以降序或升序排列,并使用起始,结束和递增值。 它不允许在多个查询中
如何在 Groovy 中获取给定数字的序列,例如: def number = 169 // need a method in groovy to find the consecutive number
基本上,如果这是 .NET,它看起来像这样: ISomething { string A { get; } int B { get; } } var somethings = new List
说以下代码部分(同一块): A <= 1 A <= 2 变量 A 总是被赋值为 2 吗?还是会出现竞争条件并分配 1 或 2? 我对非阻塞赋值的理解是,由硬件在 future 分配变量 A,因此它可能
在运行 WiX 设置时,我正在寻找操作列表及其顺序。不知何故,官方网站似乎没有提供任何信息。 基本问题是我想正确安排我的自定义操作。通常我需要使用 regsvr32.exe 注册一个 DLL,而这只能
F#初学者在这里 我想创建一个类型,它是具有至少一个元素的另一种具体类型(事件)的序列。任何其他元素都可以在以后随时添加。通常在 C# 中,我会创建一个具有私有(private) List 和公共(p
作为构建过程和不断发展的数据库的一部分,我试图创建一个脚本,该脚本将删除用户的所有表和序列。我不想重新创建用户,因为这将需要比所允许的更多的权限。 我的脚本创建了一个过程来删除表/序列,执行该过程,然
我想恢复两个向量的第一个日期和相同向量的第二个日期之间的日期序列,.... 这是一个例子: dates1 = as.Date(c('2015-10-01', '2015-03-27', '2015-0
这个问题已经有答案了: sql ORDER BY multiple values in specific order? (12 个回答) 已关闭 9 年前。 我有一个 sql 语句,我想要ORDER
我想恢复两个向量的第一个日期和相同向量的第二个日期之间的日期序列,.... 这是一个例子: dates1 = as.Date(c('2015-10-01', '2015-03-27', '2015-0
在用java编写代码时,我需要用“],[”分割字符串。下面是我的代码。 try (BufferedReader reader = new BufferedReader(new InputStreamR
这个问题已经有答案了: Project Euler Question 14 (Collatz Problem) (8 个回答) 已关闭 9 年前。 我正在尝试查找数字的 Collatz 序列。以下
我有一个例程函数process_letter_location(const char& c, string &word)。 在我的 main 中,我声明了一系列字符串变量,如下所示: string s
我需要找到最长的多米诺骨牌链,给定一组 12 个随机挑选的多米诺骨牌。我已经递归地生成了多米诺骨牌的所有可能性(使用 0 到 12 的面值有 91 种可能性)。多米诺骨牌由一 block “砖 blo
我有这个数据结构 Seq,它继承了类 vector 但有一些额外的功能。使用这个数据结构 Seq 我有这个预定义的数据结构: typedef Seq > MxInt2d; 我现在想要一个包含多个 Mx
我是一名优秀的程序员,十分优秀!