我正在尝试根据servlet请求中的数据创建多个输出文本数据文件。我的servlet的约束是:
我写的代码是这样的:
doGet
未同步
在doGet
内我正在创建一个新的线程池(原因是我的servlet的调用应用程序在我的servlet返回响应之前不会发送下一个请求 - 所以我验证了请求并返回立即确认回来获取新请求)
将所有请求数据传递给在新线程池中创建的线程
我正在使用wait(60000)
。问题是代码在一分钟内生成具有正确阈值(名称)的文件,但是在一分钟超时之后,生成的文件(极少数)超出了容量,例如,名称超过了我在容量。
我认为它与线程有什么关系,当它们醒来时会引起问题?
我的代码是
if(!hashmap_dob.containsKey(key)){
request_count=0;
hashmap_count.put(key, Integer.toString(request_count));
sb1 = new StringBuilder();
sb2 = new StringBuilder();
sb3 = new StringBuilder();
hashmap_dob.put(key, sb1);
hashmap_firstname.put(key, sb2);
hashmap_surname.put(key, sb3);
}
if(hashmap_dob.containsKey(key)){
request_count = Integer.parseInt(hm_count.get(key));
request_count++;
hashmap_count.put(key, Integer.toString(request_count));
hashmap_filehasbeenprinted.put(key, Boolean.toString(fileHasBeenPrinted));
}
hashmap_dob.get(key).append(dateofbirth + "-");
hashmap_firstname.get(key).append(firstName + "-");
hashmap_surname.get(key).append(surname + "-");
if (hashmap_count.get(key).equals(capacity)){
request_count = 0;
dob = hashmap_dob.get(key).toString();
firstname = hashmap_firstname.get(key).toString();
surname = hashmap_surname.get(key).toString();
produceFile(required String parameters for file printing);
fileHasBeenPrinted = true;
sb1 = new StringBuilder();
sb2 = new StringBuilder();
sb3 = new StringBuilder();
hashmap_dob.put(key, sb1);
hashmap_firstname.put(key, sb2);
hashmap_surname.put(key, sb3);
hashmap_count.put(key, Integer.toString(request_count));
hashmap_filehasbeenprinted.put(key, Boolean.toString(fileHasBeenPrinted));
}
try{
wait(Long.parseLong(listenerWaitingTime));
}catch (InterruptedException ie){
System.out.println("Thread interrupted from wait");
}
if(hashmap_filehasbeenprinted.get(key).equals("false")){
dob = hashmap_dob.get(key).toString();
firstname = hashmap_firstname.get(key).toString();
surname = hm_surname.get(key).toString();
produceFile(required String parameters for file printing );
sb1 = new StringBuilder();
sb2 = new StringBuilder();
sb3 = new StringBuilder();
hashmap_dob.put(key, sb1);
hashmap_firstname.put(key, sb2);
hashmap_surname.put(key, sb3);
fileHasBeenPrinted= true;
request_count =0;
hashmap_filehasbeenprinted.put(key, Boolean.toString(fileHasBeenPrinted));
hashmap_count.put(key, Integer.toString(request_count));
}
如果你必须到这里,那么感谢你阅读我的问题并提前感谢你是否有任何想要解决的问题!
答案 0 :(得分:4)
我没有查看您的代码,但我发现您的方法非常复杂。试试这个:
在ServletContextListener
中创建线程和队列。中断线程以阻止它。在线程中,当您在队列中等待时收到InterruptedException
时,将最后剩余的项目刷新到文件中。
答案 1 :(得分:1)
根据我的理解,您希望在两种情况下创建/生成新文件:
我建议遵循:
答案 2 :(得分:1)
我试图阅读您的代码,但缺少很多信息,所以如果可以请提供更多详细信息:
1)缩进版本搞砸了,我不确定复制代码时是否有错误。
2)您发布的代码是什么?在doGet之后在其他一些线程上调用的代码?
3)也许你也可以添加变量声明。那些线程安全类型(ConcurrentHashMap)?
4)我不确定我们是否拥有关于fileHasBeenPrinted的所有信息。它似乎也是一个布尔值,它不是线程安全的。
5)你谈论“同步”功能,但你没有包含这些功能。
修改强>:
如果您复制的代码是同步方法,则表示如果您有许多请求,则只有其中一个请求只在给定时间运行。似乎总是调用60秒等待(压痕不太清楚,但我认为总是有60秒等待,无论文件是否写入)。因此,在处理另一个线程(请求)之前,您将同步方法锁定60秒。这可以解释为什么你没有在20个请求之后写文件,因为超过20个请求可以在60秒内到达。