最近在闲逛XX站的时候,打算搞个破坏,试试有多少人还是用初始密码登陆。比较懒,所以直接打开控制台来写。

  所以问题可以描述为:

      向后端不断的post数据,id从1~5000自增,后端会根据情况来返回值res,需要把res=100的id输出。


最简单的想法是:for循环内部调用post数据

//错误示范 一
for(var i = 92000;i<92500;i++){
//直接借用一下网站内引用的jq
$.post("login.php", { ts:"login",username: i, password: i},function(data){
if(data=="100"){
console.log(i);
}
});
}

但是,运行结果是这样的:

post函数时异步的进行请求,拿到请求之后才会执行回调函数。for循环执行速度要快于post函数的执行速度。当执行post之后,for循环不会等待post拿到res并执行回调,而是继续遍历,for循环几百几千次的速度都快于post。所以当第一个post请求去执行其回调时,循环已经结束,i=92500。

  这和一道很经典的笔试题很像:

 for(var i = 0;i<10;i++){
setTimeout(function(){
console.log(i);
},1000);
}
//输出结果为10个 10

解决办法:利用闭包

//利用闭包和返回函数实现
for(var i=92000;i<92500;i++){
$.post("index.php?action=login",{ ts:"login",username: i, password: i,chekcode:9895 },(function(i){
return function(data){
if(data == "100"){
console.log(i)
}
}
})(i);
);
}

相关解释:

    通过把回调写成匿名函数闭包,将i变量保存并且立即调用函数,但是为了获取到返回的data数据,所以在闭包内部return function(data),用来作为真正的回调函数接受返回参数res-data

    当执行完for循环之后,console.log()首先能拿到正常返回数据data的值,因为js里函数访问参数时访问的作用域不是当前作用域,而是函数声明环境下的作用域,所以就可以直接访问到每个res=100对应的循环变量i。

  所以上面那个面试题的一种解法就是:

for(var i = 0;i<10;i++){
(function(i){setTimeout(function(){
console.log(i);
},1000)})(i);
}

对函数的一些理解:

  1.函数可以作为参数传入,参与运算。

  2.函数可以保存内部数据的状态,常见通过构造函数内部var变量实现类的私有成员

  3.还没想好怎么说,以后再补

转自:http://www.cnblogs.com/limingxi/p/4048705.html


写在最后:

上文中提到的匿名函数的闭包的参数值 i

//利用闭包和返回函数实现
for(var i=92000;i<92500;i++){
$.post("index.php?action=login",{ ts:"login",username: i, password: i,chekcode:9895 },(function(i){
return function(data){
if(data == "100"){
console.log(i)
}
}
})(i);
);
}

如果是数值或字符串,没有问题。如果参数值为对象或数组,不知道会是什么情况?请兄弟们告诉我。

结果如下:

var h = {f:""};
function a(c){
(function(d) {
setTimeout(function() {
console.log(d);
}, 1000)
})(c);
}
h.f = "fuck1";
a(h);
h.f = "fuck2";
a(h);
h.f = "fuck3";
a(h);
h.f = "fuck4";
a(h);
/*
以上结果,输出四次fuck4
*/ var h = [0];
function a(c){
(function(d) {
setTimeout(function() {
console.log(d);
}, 1000)
})(c);
}
h[0] = 1;
a(h);
h[0] = 2;
a(h);
h[0] = 3;
a(h);
h[0] = 4;
a(h);
/*
以上结果,输出四次[4]
*/

以上结果不难看出,如果参数为对象或数组时,只是修改了其值,并没有改变引用的地址,对于闭包来说,是同一个变量(只是里边的值不一样面已)。使用前多加注意

当然兄弟们估计还有更好的方法,请告知。

利用闭包向post回调函数传参数的更多相关文章

  1. [js]利用闭包向post回调函数传参数

    最近在闲逛校园XX站的时候,打算搞个破坏,试试有多少人还是用初始密码登陆.比较懒,所以直接打开控制台来写. 所以问题可以描述为: 向后端不断的post数据,id从1~5000自增,后端会根据情况来返回 ...

  2. scrapy回调函数传递参数

    scrapy.Request 的callback传参的两种方式 1.使用 lambda方式传递参数 def parse(self, response): for sel in response.xpa ...

  3. Chrome和Firefox浏览器执行new Date() 函数传参数得到不同结果的陷阱

    某日,同事问到关于new Date() 函数传参数,在火狐浏览器和谷歌浏览器控制台运行,会得到不同的结果,刚开始觉得不可能,后来实际操作才发现此陷阱 var date = new Date('2014 ...

  4. 关于main函数传参数的问题

    argc是命令行总的参数个数      argv[]是argc个参数,其中第0个参数是程序的全名,以后的参数      命令行后面跟的用户输入的参数,比如:      int   main(int   ...

  5. 解决有关flask-socketio中服务端和客户端回调函数callback参数的问题(全网最全)

    由于工作当中需要用的flask_socketio,所以自己学习了一下如何使用,查阅了有关文档,当看到回调函数callback的时候,发现文档里都描述的不太清楚,最后终于琢磨出来了,分享给有需要的朋友 ...

  6. PHP之回调函数传参(解决eval函数拼接对象参数的问题)

    在使用Smarty时,定义了一个统一调用控制器的函数,如下: function C($name, $method){//控制器的名称和其中方法的名称 require_once "contro ...

  7. 利用call与apply向函数传递参数

    Js中函数对象都有call与apply两个方法属性,二者使用方法和功能一样,只是传递参数的格式不同,call逐个传递单个参数,apply一次性传递一个参数数组. 这两个方法可以改变函数的调用对象,并且 ...

  8. 利用模块加载回调函数修改PE导入表实现注入

    最近整理PE文件相关代码的时候,想到如果能在PE刚刚读进内存的时候再去修改内存PE镜像,那不是比直接对PE文件进行操作隐秘多了么? PE文件在运行时会根据导入表来进行dll库的"动态链接&q ...

  9. javascript 利用匿名函数对象给你异步回调方法传参数

    先来创建一个匿名函数对象: /*** * 匿名函数 */ var callChangeBtn=new function(bugBtn){ this.chage=function(json){ bugB ...

随机推荐

  1. 3.SRS文档

    1.功能需求 本程序的使用者为局域网用户.程序实现的主要功能是局域网的常见格式的文件的传 输.其用例图如图1.本程序可通过可视化操作界面实现一对多的文件传输. 1.1模块分析 为实现局域网文件传输, ...

  2. openVPN报错:All TAP-Windows adapters on this system are currently in use

    解决办法: 1. 确定openVPN服务已打开. 2. 控制面板\网络和 Internet\网络连接,确定TAP-Windows Adapter V9已经启动.

  3. ajaxSubmit与ajaxFileUpload的空值

    ajaxSubmit发现正文无内容, ajaxFileUpload发现无上传内容, 嗯,你只是忘记了 html 中控件的 name 没有指定,这哥们不认 id 的.

  4. Spring依赖注入(IOC)那些事

    小菜使用Spring有几个月了,但是对于它的内部原理,却是一头雾水,这次借着工作中遇到的一个小问题,来总结一下Spring. Spring依赖注入的思想,就是把对象交由Spring容器管理,使用者只需 ...

  5. ASP.NET MVC中三方登录: 微软、谷歌、Office365

    创建一个MVC的工程,在Startup.Auth.cs文件中,我们能看到这样的一些代码: 这其实是微软已经帮我们实现好的三方登录的接口,我们只需要创建相应的开发者账号,并在其中配置好跟我们应用程序相关 ...

  6. java5 CountDownLatch同步工具

    好像倒计时计数器,调用CountDownLatch对象的countDown方法就将计数器减1,当到达0时,所有等待者就开始执行. java.util.concurrent.CountDownLatch ...

  7. struts2学习笔记之二:基本环境搭建

    学习struts2有一段时间了,作为一个运维人员学习的时间还是挺紧张的,写这篇文件为了方便以后复习时使用 环境: MyEclipse 10 tomcat6 jdk1.6   首先建立一个web项目,并 ...

  8. Singleton模式和Mono-State模式

    类和实例 对于大多数的类,都可以创建多个实例.在需要和不需要时,创建和删除这些实例.该过程会伴随着内存的分配和归还. 同时,有一些类,应该仅有一个实例.该实例在程序启动/结束时被创建和删除. root ...

  9. Bitbucket Repository size limits

    Repository size limits By Justen Stepka, Product Manager on May 30, 2014 In order to improve and mai ...

  10. 关于ScrollerView的一些小心得

    在项目开发时遇到一个问题,我在UIViewController上面直接创建了一个UIScrollerView,把UIScrollerView作为一个子视图添加到了UIViewController, 又 ...