ホームページ  >  記事  >  ウェブフロントエンド  >  パラメータを使用した JS ループ動的バインディング関数で発生する問題と解決策 [transfer]_javascript スキル

パラメータを使用した JS ループ動的バインディング関数で発生する問題と解決策 [transfer]_javascript スキル

WBOY
WBOYオリジナル
2016-05-16 18:16:381066ブラウズ

ご存知のとおり、パラメーターを使用しないバインディングは非常に簡単で、(構文: "document.getElementById("オブジェクト ID 名").attachEvent("イベント名 (onchange など)、関数名);") を使用するだけです (例: "ドキュメント。getElementById("select_0").attachEvent("onchange",modifyFunction);")。 (注: 以下は単なる例です)
パラメーターを使用したバインドはさらに複雑です: document.getElementById("select _0").attachEvent("onchange",function(){modifyFunction (obj,i););つまり、実行する必要がある関数を function() に記述するだけです。もちろん、document.getElementById("select _0").onchange=function(){modifyFunction (obj,i);); という別の書き方もあります。
バインドが成功しました。OK。しかし、このとき、渡されたパラメータ値がすべて同じであるという問題が発生しました。i の値を渡した後、各バインドされた関数のパラメータ値が異なるとは予想外でした。
それでは、オンラインで Baidu にアクセスしてください。難しい検索とテストの結果、次のような例も見つかりました。

コードをコピー コードは次のとおりです。

<script> <br>document.onclick=check; <br>function check() { <br>if(event.srcElement.type== "button ") <br>alert(event) .srcElement.name); <br></script>

この例では、イベントを通じてアクションを持つコンポーネントを検索し、そのソースを取得して、名前の値を抽出します。このようにして、受信した obj を通じて obj 番号を取得し、対応する操作を実行できます。
問題が 1 つあります。この操作の後、obj の値はどの選択を操作しても最後の値になります。
引き続き百度にアクセスしてください。
ついに、その理由が 1 つの記事で分かりました。この記事は次のように再投稿されます。
まず、ループ バインディング イベントを使用する Javascript の例を見てみましょう:
例: 無限の長さのリスト、マウスが特定の項目の上を通過するときに背景を変更する。





[Ctrl A すべて選択 注: 外部 Js を導入する必要がある場合は、更新して実行する必要があります ] これこの例では、グループ オブジェクト バインディング イベント ハンドラー関数にループします。 ただし、これに基づいていくつかの要件を追加すると。たとえば、特定のレコードをクリックすると、どのレコードがポップアップ表示されますか?
おそらくあなたは当然のようにこれを書くでしょう:




[Ctrl A すべて選択 注: 外部 Js を導入する必要がある場合は、 を実行するために更新する必要があります]
测试一下你会发现alert出来的都是:这是第6记录
其实这里for循环已将整个列表循环了一遍,并执行了i++,所以这里i变成了6,
有什么好的办法解决这个问题吗?
那就是闭包了,个人认为闭包是js中最难捉摸的地方之一,
看看什么是闭包:
闭包时是指内层的函数可以引用存在与包围他的函数内的变量,即使外层的函数的执行已经终止。
这个例子中我们可以这样做:

PS:闭包很难,很复杂!
经过以上文章可以得知,引起这个问题的原因其实是因为js的闭包难题。按照面向对象的JAVA语言的理解可以解释为:js循环动态绑定带参数函数中的参数,其实相当于java中的引用传递,而非值传递。传递进来的引用只相当于一个指针,指向的是一个内存地址,这个内存地址存放的才是具体的值,而外面的循环会不断的修改这个存放地址中的值,所以最后循环结束之后,参数的值只能找到最后一个。
知道了原因就很好解决了。New一个新的“函数类”(姑且这么称呼吧)。测试OK。一下是修改后的代码:
复制代码 代码如下:

//在新增按钮上绑定函数
document.getElementById("add").attachEvent("onclick",addFunction);
var jc_count = 0;//定义需要改变第几行的值
function txzmcFunction(x,y){//下拉框中绑定的函数
var sql="select txzjc from dm_txzmc where dm='"+x.value+"'";//取得下拉框中的代码,通过ajax获得相应的中文名称
jc_count = y;//定义当前行是第几行
ajaxSelect(sql,"txzjcFunction");//封装的ajax函数
}
function txzjcFunction(x){//接收封装的ajax函数返回值,并赋值
document.getElementById("_subarea_hxax_clzjxxb_hxax_txzxxb_update_txzjc_"+jc_count).value=x;
}
function bb(dx,sz){//解决动态绑定闭包问题要用到函数
this.clickFunc=function(){
txzmcFunction(dx,sz);//调用相应的函数
}
}
function addFunction(){ //动态循环绑定
var count=document.getElementById("_subarea_hxax_clzjxxb_hxax_txzxxb_update_maxcount").value;//获取最大的行数
for (var i=0;i{
var obj=document.getElementById("_subarea_hxax_clzjxxb_hxax_txzxxb_update_txzmc_" +i);
var tp = new bb(obj,i);//解决闭包问题,new一个新的函数类
obj.onchange = tp.clickFunc;
}
}
//显示页面时执行一次
addFunction();
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。