多线程并发写文件-文件锁
在项目中,遇到一个需求是读取日志文件内容,解析后将内容写入到html文件中。
日志文件介绍,每一行表示一条交易信息。读取一行的一条信息将其解析,即使对数据进行处理,之后写入到html文件中。
读文件采用的是正则表达式,每匹配到一条信息就解析。
在写入html文件时,会出现一个线程正在进行写操作,而另一个线程也要访问文件。为了避免写内容时出现混乱情况,这样的情况是不允许发生的。这时就需要对文件进行加锁处理。即使一个线程在对文件进行操作时,其他线程是不能对文件进行操作的。
解决的思路是,每当有线程访问文件时就对文件进行加锁处理,写操作完毕之后释放锁。其他线程只有获得锁才能对文件进行操作。否则就一直等待,直到获得文件锁。
之前用的是 ReentrantLock 这个类对文件进行加锁。下面是代码:
private final ReentrantLock lock = new ReentrantLock(); public void doVMenuAccessOutPutHtml(File file, ServiceData sd) { RandomAccessFile out = null; lock.lock(); try { if (!file.exists()) { file.createNewFile(); out = new RandomAccessFile(file, "rw"); writehead(out); writeVMenuHead(out); writefoot(out); } out = new RandomAccessFile(file, "rw"); String style = ""; if (count.get() % 2 == 0) { style = " <tr>\r\n<td> "; } else { style = " <tr class=\"alt\">\r\n<td> "; } StringBuffer sb = new StringBuffer(); String dateTime = sd.getString("dateTime"); String trandate = sd.getString("trandate"); String trancode = sd.getString("trancode"); String orgcode = sd.getString("orgcode"); String clerk = sd.getString("clerk"); String terminal = sd.getString("terminal"); String errcode = sd.getString("errcode"); String errstr = sd.getString("errstr"); sb.append(style + count.incrementAndGet() + " </td>\r\n<td> " + dateTime + " </td>\r\n<td> " + trandate + " </td>\r\n<td> " + trancode + " </td>\r\n<td> " + orgcode + " </td>\r\n<td> " + clerk + " </td>\r\n<td> " + terminal + " </td>\r\n<td> " + errcode + " </td>\r\n<td> " + errstr + " </td>\r\n</tr>\r\n "); long fileLength = out.length(); out.seek(fileLength - 26); out.write(sb.toString().getBytes("utf-8")); writefoot(out); } catch (IOException e) { // file.deleteOnExit(); System.out.println("Exception encountered: " + e); } finally { lock.unlock(); try { out.close(); out = null; } catch (IOException e) { System.out.println("Exception encountered: " + e); } } }
可是当文件两太大时,会产生很多线程对文件进行操作,就出现了写入文件混乱的情况。
如下图:
写入混乱导致页面出现混乱的情况。
问题原因我的理解是所有的线程有用的同一个锁导致。(这里具体的原因我还是不太明白,希望哪位博友能指点一下。感激不尽)
在网上找了一些资料后,将ReentrantLock换成了FileLock 。实现代码如下:
public static void doVMenuAccessOutPutHtml(File file, ServiceData sd) { RandomAccessFile out = null; try { if (!file.exists()) { file.createNewFile(); out = new RandomAccessFile(file, "rw"); writehead(out); writeVMenuHead(out); writefoot(out); } out = new RandomAccessFile(file, "rw"); FileChannel fcout = out.getChannel(); FileLock flout = null; while (true) { try { flout = fcout.lock(); break; } catch (Exception e) { System.out.println("有其他线程正在操作该文件,当前线程休眠1000毫秒"); } } String style = ""; if (countVMenu.get() % 2 == 0) style = " <tr>\r\n<td> "; else style = " <tr class=\"alt\">\r\n<td> "; StringBuffer sb = new StringBuffer(); String dateTime = sd.getString("dateTime"); String trandate = sd.getString("trandate"); String trancode = sd.getString("trancode"); String orgcode = sd.getString("orgcode"); String clerk = sd.getString("clerk"); String terminal = sd.getString("terminal"); String errcode = sd.getString("errcode"); String errstr = sd.getString("errstr"); sb.append(style + countVMenu.incrementAndGet() + " </td>\r\n<td> " + dateTime + " </td>\r\n<td> " + trandate + " </td>\r\n<td> " + trancode + " </td>\r\n<td> " + orgcode + " </td>\r\n<td> " + clerk + " </td>\r\n<td> " + terminal + " </td>\r\n<td> " + errcode + " </td>\r\n<td> " + errstr + " </td>\r\n</tr>\r\n "); long fileLength = out.length(); out.seek(fileLength - 26); out.write(sb.toString().getBytes("gbk")); writefoot(out); flout.release(); fcout.close(); out.close(); out = null; } catch (IOException e) { file.deleteOnExit(); System.out.println("Exception encountered: " + e); } }
程序为每一个线程都定义了锁。问题得到了解决。
为什么用第二种方法就可以了呢?
这里我也说不出具体的原因,如果你知道的话,希望能指点一下哟。
本文出自 “一个风向” 博客,请务必保留此出处http://lanffy.blog.51cto.com/6452125/1374360
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。