关于用于防止内存泄漏的空赋值修复的性质,有人可以为我挠痒痒吗?
我们都熟悉以下技术来阻止DOM对象和JS对象之间的循环引用,以防止内存泄漏:
function foo() {
var ele = document.getElementById("someParagraphId");
ele.onclick = function() {
//some action here
};
ele = null;
}
问题是为什么上面的方法会起作用?将“ele”设置为 null 肯定会停止循环引用,但它不会也阻止将来对“ele”的引用吗?
function foo() {
var ele = document.getElementById("someParagraphId");
ele.onclick = function() {
console.log("accessing ele ... after set to null " + ele.onclick);
};
ele = null;
}
然而事件监听器触发了。它会提示“ele”对象为空(这是我们所期望的)。
鉴于上述行为,我们是否可以正确推断 Javascript 引擎实现将保存对事件监听器的某种内部引用,并且在触发事件时调用该引用?
eventHandlerRef = //assignment to "ele.onclick" handler/listener;
如果有像上面那样的引用,那么赋值到空的修复是否会依赖于实现?或者,它是 ECMAScript 规范的一部分。
据我了解,此修复始终是跨浏览器安全的。我还没有遇到很多具体提到在应用空分配之前检测/嗅探浏览器类型的示例。
===============编辑==================
我认为由于我提出问题的方式,可能会在不知不觉中直接与我试图传达的内容进行讨论。正在引用的几个概念:
对象句柄/对象引用:
var obj1, obj2;
obj1 = {foo: "bar"}; //obj1 points to the an area of the heap allocated
//for the object literal.
obj2 = obj1; //obj2 points to the same position in the heap
//as obj1.
//obj1 = null or
obj1 = {/*another obj literal*/} //the new object literal is created
//and allocated in some other part
//in the heap, and obj1 now points
//to the new location.
//obj2 is unaffected by obj1's re-assignment.
上面不是我的痒处,我很遗憾添加了这行:
console.log("accessing ele ... after set to null " + ele.onclick);
以上内容使这看起来像是一个封闭问题。我完全预料到会按照原始帖子中的指示抛出错误。
我的渴望更多的是......出于某种原因,在我看来,我一直认为 Javascript 引擎会在事件触发和设置时直接调用 ele.onlick()元素为 null 类似于以下流程:
var obj = {};
obj.method = function() {};
obj = null;
//error calling obj.method(), i.e., obj is null
鉴于我们在原始帖子中知道在 ele 设置为 null 后事件处理程序仍然会触发,在我看来流程更类似于:
var obj = {};
obj.method = function() {};
var objRef = obj;
obj = null;
//The Javascript Engine calling objRef.method when event triggered.
我的问题归结为一个问题,上面的大多数 Javascript 实现是如何工作的,其中一些内部引用指向分配的事件处理程序,并且是在触发事件时调用这个内部引用?
或者换句话说,是什么阻止了 Javascript 引擎实现直接调用 an ele.onclick()(暂时搁置设计和架构问题)?
也许我的思维过程不同,但当其他人第一次遇到空赋值修复时,没有人再看一眼,元素引用为空,但处理程序的代码还是被处决了?
最佳答案
丢弃所有旧答案,并解决编辑问题:)
让我们举一个更简单的例子:textContent。
var ele = document.getElementById('foo');
ele.textContent = 'abc';
ele = null;
此代码将#foo 的textContent 设置为abc,并丢弃对ele 的引用。现在如果我再次请求 #foo 会发生什么?...
var ele = document.getElementById('foo');
ele.textContent = 'abc';
ele = null;
ele = document.getElementById('foo');
console.log('#foo is ' + ele.textContent);
它将记录 “#foo is abc”。这只是表明 #foo 在我的脚本之外继续存在,并且 document 保留对它的引用(因为我得到它调用 document<>)。它与事件处理程序的工作方式相同。
var ele = document.getElementById('foo');
ele.onclick = function() { console.log('abc'); };
ele = null;
ele = document.getElementById('foo');
console.log('#foo.onclick is ' + ele.onclick);
事件处理程序不是一种特殊的属性。它们只是您在其上编写引用(对函数)的变量。这些大致是函数式语言的特性,其中函数可以用作简单的值。 JavaScript 实现只是从 DOM 引用调用事件处理程序,而不是从 ele 引用调用。让我们再举一个更简单的例子,没有 document.getElementById。
var a = {};
var b = a;
b.foo = function() { console.log("hello world!"); };
console.log(a.foo + " is the same as " + b.foo);
b = null;
console.log("Even though b is " + b + ", a.foo is still " + a.foo);
a.foo();
如您所见,函数与我们为其分配的引用无关:它们与引用指向的对象相关联。您可以从对对象的任何引用调用它们,而不仅仅是从您用来分配函数的引用。
因此,您可以取消引用以打破循环依赖,而不影响对象的正确行为。
关于Javascript 内存泄漏 : why would assigning object to null work?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6326778/
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
ruby如何管理内存。例如:如果我们在执行过程中采用C程序,则以下是内存模型。类似于这个ruby如何处理内存。C:__________________|||stack|||------------------||||------------------|||||Heap|||||__________________|||data|__________________|text|__________________Ruby:? 最佳答案 Ruby中没有“内存”这样的东西。Class#allocate分配一个对象并返回该对象。这就是程序
你好,我无法成功如何在散列中删除key后释放内存。当我从哈希中删除键时,内存不会释放,也不会在手动调用GC.start后释放。当从Hash中删除键并且这些对象在某处泄漏时,这是预期的行为还是GC不释放内存?如何在Ruby中删除Hash中的键并在内存中取消分配它?例子:irb(main):001:0>`ps-orss=-p#{Process.pid}`.to_i=>4748irb(main):002:0>a={}=>{}irb(main):003:0>1000000.times{|i|a[i]="test#{i}"}=>1000000irb(main):004:0>`ps-orss=-p
我遇到了一个非常奇怪的问题,我很难解决。在我看来,我有一个与data-remote="true"和data-method="delete"的链接。当我单击该链接时,我可以看到对我的Rails服务器的DELETE请求。返回的JS代码会更改此链接的属性,其中包括href和data-method。再次单击此链接后,我的服务器收到了对新href的请求,但使用的是旧的data-method,即使我已将其从DELETE到POST(它仍然发送一个DELETE请求)。但是,如果我刷新页面,HTML与"new"HTML相同(随返回的JS发生变化),但它实际上发送了正确的请求类型。这就是这个问题令我困惑的
这会导致Ruby出现内存问题吗?我知道如果大小超过10KB,Open-URI会写入TempFile。但是HTTParty会在写入TempFile之前尝试将整个PDF保存到内存吗?src=Tempfile.new("file.pdf")src.binmodesrc.writeHTTParty.get("large_file.pdf").parsed_response 最佳答案 您可以使用Net::HTTP。参见thedocumentation(特别是标题为“流媒体响应机构”的部分)。这是文档中的示例:uri=URI('http://e
我有这个:AccountSummary我想单击该链接,但在使用link_to时出现错误。我试过:bot.click(page.link_with(:href=>/menu_home/))bot.click(page.link_with(:class=>'top_level_active'))bot.click(page.link_with(:href=>/AccountSummary/))我得到的错误是:NoMethodError:nil:NilClass的未定义方法“[]” 最佳答案 那是一个javascript链接。Mechan
我看到有关未找到文件min.map的错误消息:GETjQuery'sjquery-1.10.2.min.mapistriggeringa404(NotFound)截图这是从哪里来的? 最佳答案 如果ChromeDevTools报告.map文件的404(可能是jquery-1.10.2.min.map、jquery.min.map或jquery-2.0.3.min.map,但任何事情都可能发生)首先要知道的是,这仅在使用DevTools时才会请求。您的用户不会遇到此404。现在您可以修复此问题或禁用sourcemap功能。修复:获取文
在部署在heroku上的Rails应用程序(v:3.1)中,我在内存中获得了更多具有相同ID的对象。我的heroku控制台日志:>>Project.find_all_by_id(92).size=>2>>ActiveRecord::Base.connection.execute('select*fromprojectswhereid=92').to_a.size=>1这怎么可能?可能是什么问题? 最佳答案 解决方案根据您的SQL查询,您的数据库中显然没有重复条目。也许您的类项目中的size或length方法已被覆盖。我试过find_
我的两个不同的Rails应用程序的内存有一些奇怪的问题。这两个应用程序都使用rails3.0.7。每个Controller请求分配20-30-50MB的内存。在生产模式下,这个数量减少到5-10。但这是同样的事情。这是两个应用程序使用的gem列表:gem'pg'gem'haml'gem'sass'gem'devise'gem'simple_form'gem'state_machine'gem"globalize3","0.1.0.beta"gem"easy_globalize3_accessors"gem'paperclip'gem'andand'关闭所有这些gem不会给我任何结果。我
我有一个用Rails3编写的站点。我的帖子模型有一个名为“内容”的文本列。在帖子面板中,html表单使用tinymce将“content”列设置为textarea字段。在首页,因为使用了tinymce,post.html.erb的代码需要用这样的原始方法来实现。.好的,现在如果我关闭浏览器javascript,这个文本区域可以在没有tinymce的情况下输入,也许用户会输入任何xss,比如alert('xss');.我的前台会显示那个警告框。我尝试sanitize(@post.content)在posts_controller中,但sanitize方法将相互过滤tinymce样式。例如