for循环奇怪错误,for(int i=0;i<10;i++)但循环内的i居然能取到10了??大牛解答。

三爷_ 2011-08-16 01:07:55
        List<ErrorModel> errorList = new List<ErrorModel>();
static ManualResetEvent[] ThreadEvent = new ManualResetEvent[10];//线程事件

public void MakeHtmlByThread(out IList<ErrorModel> Errors, IList<PathModel> pathModels)
{

int count = pathModels.Count / 10;

for (int i = 0; i < 10; i++) //将任务分发给10个支线程
{
ThreadEvent[i] = new ManualResetEvent(false);
ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); });
}
WaitHandle.WaitAll(ThreadEvent); //等待所有支线程结束

Errors = errorList;
}

private void ToHtml(IList<PathModel> modelList, ManualResetEvent _theradEvent)
{
foreach (PathModel model in modelList)
{
Stopwatch watch = Stopwatch.StartNew();
try
{
string url = "http://" + Request.UrlReferrer.Authority + Request.ApplicationPath + model.aspxPath;
string html = model.htmlPath;
if (html.EndsWith(".html"))
{
int dot = html.LastIndexOf(".html");
int index = html.LastIndexOf("/") + 1;
string s1 = html.Substring(0, index);
string s2 = html.Substring(index, dot - index);
string fileDir = ZH.FileControl.FileBuild.BuildHtmls(url, Request.PhysicalApplicationPath, s1, s2);
}
}
catch (Exception ex)
{
errorList.Add(new ErrorModel() { errorMsg = Thread.CurrentThread.Name + "号线程执行的任务包含错误:" + ex.Message + ",耗时:" + watch.Elapsed.TotalSeconds + "秒。", id = model.id });
}
}

if (null != _theradEvent) _theradEvent.Set();//指示本线程已结束
}


这段代码是用多线程生成静态页面,其中:MakeHtmlByThread方式启动了10个线程,每个,但是ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); }此处却每次都提示数组超出索引,我调试看,发现i居然等于10了。。。

怎么解决这个问题?是什么原因导致?


而且我发现我只要规规矩矩的按F10单步调试,是不会出现这种问题的,但是我只要正常运行,或者按F5跳过,就会出现这种问题。


是多线程导致的吗?

怎么解决?
...全文
678 13 打赏 收藏 转发到动态 举报
写回复
用AI写文章
13 条回复
切换为时间正序
请发表友善的回复…
发表回复
三爷_ 2011-08-16
  • 打赏
  • 举报
回复
问题解决,我用了另外一个不是很优雅的办法:

List<ErrorModel> errorList = new List<ErrorModel>();
Thread[] thrads = new Thread[10];

public void MakeHtmlByThread(out IList<ErrorModel> Errors, IList<PathModel> pathModels)
{
StartThread(pathModels);
foreach (Thread t in thrads) t.Join();

Errors = errorList;
}

private System.Object lockThis = new System.Object();//此对象用于线程安全锁
public void StartThread(IList<PathModel> pathModels)
{
lock (lockThis)
{
int count = pathModels.Count / 10;

for (int i = 0; i < 10; i++) //将任务分发给10个支线程
{
thrads[i] = new Thread(ToHtml);
thrads[i].Start(pathModels.Skip(count * i).Take(count).ToList<PathModel>());
}
}
}
HIpity 2011-08-16
  • 打赏
  • 举报
回复
[Quote=引用 10 楼 bluepigya 的回复:]
for (int i = 0; i < 10; i++)
最後推出循環的時候是,1不就是等於10嗎?
[/Quote] 对啊
三爷_ 2011-08-16
  • 打赏
  • 举报
回复
[Quote=引用 9 楼 tomysea 的回复:]

引用 6 楼 nozeryu 的回复:
我大概猜到原因了,多线程问题,ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); }最后一次执行这句是,循环已经退出,此时i=10,然而因为多线程,在循环退出后……
[/Quote]

private System.Object lockThis = new System.Object();
public void MakeHtmlByThread(out IList<ErrorModel> Errors, IList<PathModel> pathModels)
{
lock (lockThis)
{
int count = pathModels.Count / 10;

for (int i = 0; i < 10; i++) //将任务分发给10个支线程
{
ThreadEvent[i] = new ManualResetEvent(false);
ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[(i)]); });
}

WaitHandle.WaitAll(ThreadEvent); //等待所有支线程结束
}

Errors = errorList;
}


还是相同的错误
bluepigya 2011-08-16
  • 打赏
  • 举报
回复
for (int i = 0; i < 10; i++)
最後推出循環的時候是,1不就是等於10嗎?
tomysea 2011-08-16
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 nozeryu 的回复:]
我大概猜到原因了,多线程问题,ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); }最后一次执行这句是,循环已经退出,此时i=10,然而因为多线程,在循环退出后才执行了这句代码。所以此时i等于10了,然后……
[/Quote]

lock(someObject)
{

}
子夜__ 2011-08-16
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 nozeryu 的回复:]

我大概猜到原因了,多线程问题,ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); }最后一次执行这句是,循环已经退出,此时i=10,然而因为多线程,在循环退出后才执行了这句代码。所以此时i等于10了,然……
[/Quote]
加个判断
小脩 2011-08-16
  • 打赏
  • 举报
回复
[Quote=引用 3 楼 zanfeng 的回复:]
退出前的最后一次不就是嘛?
[/Quote]

正解!
三爷_ 2011-08-16
  • 打赏
  • 举报
回复
我大概猜到原因了,多线程问题,ThreadPool.QueueUserWorkItem(delegate { ToHtml(pathModels.Skip(count * i).Take(count).ToList<PathModel>(), ThreadEvent[i]); }最后一次执行这句是,循环已经退出,此时i=10,然而因为多线程,在循环退出后才执行了这句代码。所以此时i等于10了,然后抛异常。

那么怎么解决这个问题呢?
qydvip 2011-08-16
  • 打赏
  • 举报
回复
不清楚


当i=9 的时候 后边i++ 这样i就等于10了 但是等于10的时候会跳出循环啊


难道没有跳吗
ajaxtop 2011-08-16
  • 打赏
  • 举报
回复
和多线程没关系,你调试到第八次时仔细看下i的值,和你那个数组的个数

问题是有点怪,至少i为10不会进入循环嘛

还有就是调试要是一步一步就正常,我觉得这个应该不对,你把代码重新生成一下

然后调试到最后一次在仔细看下i的值和数组的个数
足球中国 2011-08-16
  • 打赏
  • 举报
回复
退出前的最后一次不就是嘛?
三爷_ 2011-08-16
  • 打赏
  • 举报
回复
不是i从几开始,你看清楚,我循环的条件是i<10,而不是i<=10,那么循环内部怎么可能出现i=10的时候。
ajaxtop 2011-08-16
  • 打赏
  • 举报
回复
你那i从0开始的
​ 博主介绍:✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌项目名称基于Web的酒店客房管理系统的设计与实现系统说明3.2.1  用户模块功能分析前台模块应主要包括用户登录模块、用户注册模块、查看客房信息模块、客房预定 模块、用户留言模块、充值模块和个人信息维护模块。用户登录模块:用户通过输入注册的的账号和密码,然后进行身份验证,匹配成功 后实现登录功能。用户注册模块:用户输入合法的账号和密码可以实现注册功能。 查看客房信息模块:用户可以通过首页查看客房的价格,图片,详情等信息,从而   可以选择想要预定的房间。 客房预定模块:用户选择自己想要预定的客房后,可以通过输入相关信息进行订房操作。用户留言模块:用户可以向管理员发送留言。 充值模块:用户可以通过添加银行卡再为自己进行充值操作。 个人信息维护模块:用户可以修改自己的姓名、密码、身份证号等信息,还可以查看自己的订单。3.2.2  管理员模块功能分析对于管理员而言,一个好的管理系统总是能让酒店的管理工作事半功倍[7]。管理员 能够通过这个系统对自己的酒店情况一目了然,应该包括客户留言模块、客房管理模 块、订房信息管理模块、入住信息管理模块、统计分析模块、酒店新闻管理模块、会 员信息管理模块、员工信息管理模块、系统用户管理模块、个人信息维护模块。客户留言模块:管理员可以查看并回复用户的留言。客房管理模块:管理员可以管理客房信息,可以添加新的客房,删除已经停用的客房信息,还可以修改现有的客房信息。订房信息管理模块:管理员可以处理用户的订房请求,为用户办理入住手续。入住信息管理模块:对于没有注册的线下客户,也可以办理入住手续。统计分析模块:可以对酒店所有的入住信息进行记录和总结分析。酒店新闻管理模块:管理员可以更新网站上的新闻公告,展示图片等信息。会员信息管理模块:管理员可以查看到所有的注册会员信息,可以对会员信息进行删除,修改,添加操作。员工信息管理模块:管理员可以查看到自己公司所有的员工信息,而且还可以对员工的相关信息进行管理。系统用户管理模块:管理员可以查看到所有的系统管理员信息并对管理员信息进行管理。个人信息维护模块:管理员可以更改自己的登录密码或者是姓名、性别、手机号等 相关个人信息。​编辑 环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。2.IDE环境:IDEA,Eclipse,Myeclipse都可以。推荐IDEA;3.tomcat环境:Tomcat 7.x,8.x,9.x版本均可4.硬件环境:windows 7/8/10 1G内存以上;或者 Mac OS; 5.数据库:MySql 5.7版本;6.是否Maven项目:否;技术栈1. 后端:Spring+SpringMVC+Mybatis2. 前端:JSP+CSS+JavaScript+jQuery使用说明1. 使用Navicat或者其它工具,在mysql中创建对应名称的数据库,并导入项目的sql文件;2. 使用IDEA/Eclipse/MyEclipse导入项目,Eclipse/MyEclipse导入时,若为maven项目请选择maven;若为maven项目,导入成功后请执行maven clean;maven install命令,然后运行;3. 将项目中springmvc-servlet.xml配置文件中的数据库配置改为自己的配置;4. 运行项目,在浏览器中输入http://localhost:8080/ 登录运行截图​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑 用户管理控制层:package com.houserss.controller;import javax.servlet.http.HttpSession;import org.apache.commons.lang3.StringUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.RequestParam;import org.springframework.web.bind.annotation.ResponseBody;import com.houserss.common.Const;import com.houserss.common.Const.Role;import com.houserss.common.ServerResponse;import com.houserss.pojo.User;import com.houserss.service.IUserService;import com.houserss.service.impl.UserServiceImpl;import com.houserss.util.MD5Util;import com.houserss.util.TimeUtils;import com.houserss.vo.DeleteHouseVo;import com.houserss.vo.PageInfoVo;/** * Created by admin */@Controller@RequestMapping(/user/)public class UserController if (ip != null && ip.length() > 0) String[] ips = ip.split(,);for (int i = 0; i  

62,046

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术交流专区
javascript云原生 企业社区
社区管理员
  • ASP.NET
  • .Net开发者社区
  • R小R
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

.NET 社区是一个围绕开源 .NET 的开放、热情、创新、包容的技术社区。社区致力于为广大 .NET 爱好者提供一个良好的知识共享、协同互助的 .NET 技术交流环境。我们尊重不同意见,支持健康理性的辩论和互动,反对歧视和攻击。

希望和大家一起共同营造一个活跃、友好的社区氛围。

试试用AI创作助手写篇文章吧