JAVA多线程读写文件范例
在写之前先声明,本文是基于之前在博客园网站上检索到的一份JAVA多线程读写文件的示例,我在写自己的程序时是在那位作者写的基础上做了改良,但已不记得原文的地址。如果有知情者,烦请帖出地址,我在此文上加入引用或转载。
本程序是基于这么一种考虑,某系统后台有个将近2G大小的日志文件,你用任何编辑器去打开它,都将会很困难。针对这样的大文件解析处理,解决方案是使用多个线程,分割读取指定的大文件。获取我们所需要的信息。不多说,上代码了,有注释可以帮助理解。
- package com.thread.multipl.mysolution;
- import java.io.IOException;
- import java.io.RandomAccessFile;
- import java.util.concurrent.CountDownLatch;
- /**
- * 这个线程用来读取文件,当获取到指定关键字时,在指定的对象加1
- * @author 刘峰管理2
- *
- */
- public class ReadThread extends Thread{
- //定义字节数组(取水的竹筒)的长度
- private final int BUFF_LEN = 256;
- //定义读取的起始点
- private long start;
- //定义读取的结束点
- private long end;
- //将读取到的字节输出到raf中 randomAccessFile可以理解为文件流,即文件中提取指定的一部分的包装对象
- private RandomAccessFile raf;
- //线程中需要指定的关键字
- private String keywords;
- //此线程读到关键字的次数
- private int curCount = 0;
- /**
- * jdk1.5开始加入的类,是个多线程辅助类
- * 用于多线程开始前统一执行操作或者多线程执行完成后调用主线程执行相应操作的类
- */
- private CountDownLatch doneSignal;
- public ReadThread(long start, long end, RandomAccessFile raf,String keywords,CountDownLatch doneSignal){
- this.start = start;
- this.end = end;
- this.raf = raf;
- this.keywords = keywords;
- this.doneSignal = doneSignal;
- }
- public void run(){
- try {
- raf.seek(start);
- //本线程负责读取文件的大小
- long contentLen = end - start;
- //定义最多需要读取几次就可以完成本线程的读取
- long times = contentLen / BUFF_LEN+1;
- System.out.println(this.toString() + " 需要读的次数:"+times);
- byte[] buff = new byte[BUFF_LEN];
- int hasRead = 0;
- String result = null;
- for (int i = 0; i < times; i++) {
- //之前SEEK指定了起始位置,这里读入指定字节组长度的内容,read方法返回的是下一个开始读的position
- hasRead = raf.read(buff);
- //如果读取的字节数小于0,则退出循环! (到了字节数组的末尾)
- if (hasRead < 0) {
- break;
- }
- result = new String(buff,"gb2312");
- /// System.out.println(result);
- int count = this.getCountByKeywords(result, keywords);
- if(count > 0){
- this.curCount += count;
- }
- }
- KeyWordsCount kc = KeyWordsCount.getCountObject();
- kc.addCount(this.curCount);
- doneSignal.countDown();//current thread finished! noted by latch object!
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- public long getStart() {
- return start;
- }
- public void setStart(long start) {
- this.start = start;
- }
- public long getEnd() {
- return end;
- }
- public void setEnd(long end) {
- this.end = end;
- }
- public RandomAccessFile getRaf() {
- return raf;
- }
- public void setRaf(RandomAccessFile raf) {
- this.raf = raf;
- }
- public int getCountByKeywords(String statement,String key){
- return statement.split(key).length-1;
- }
- public int getCurCount() {
- return curCount;
- }
- public void setCurCount(int curCount) {
- this.curCount = curCount;
- }
- public CountDownLatch getDoneSignal() {
- return doneSignal;
- }
- public void setDoneSignal(CountDownLatch doneSignal) {
- this.doneSignal = doneSignal;
- }
- }
- package com.thread.multipl.mysolution;
- import java.io.File;
- import java.io.RandomAccessFile;
- import java.util.concurrent.CountDownLatch;
- public class MultiReadTest {
- /**
- * 多线程读取文件测试
- * @param args
- */
- public static void main(String[] args) {
- // TODO Auto-generated method stub
- final int DOWN_THREAD_NUM = 10;//起10个线程去读取指定文件
- final String OUT_FILE_NAME = "d:\\倚天屠龙记.txt";
- final String keywords = "无忌";
- //jdk1.5线程辅助类,让主线程等待所有子线程执行完毕后使用的类,
- //另外一个解决方案:自己写定时器,个人建议用这个类
- CountDownLatch doneSignal = new CountDownLatch(DOWN_THREAD_NUM);
- RandomAccessFile[] outArr = new RandomAccessFile[DOWN_THREAD_NUM];
- try{
- long length = new File(OUT_FILE_NAME).length();
- System.out.println("文件总长度:"+length+"字节");
- //每线程应该读取的字节数
- long numPerThred = length / DOWN_THREAD_NUM;
- System.out.println("每个线程读取的字节数:"+numPerThred+"字节");
- //整个文件整除后剩下的余数
- long left = length % DOWN_THREAD_NUM;
- for (int i = 0; i < DOWN_THREAD_NUM; i++) {
- //为每个线程打开一个输入流、一个RandomAccessFile对象,
- //让每个线程分别负责读取文件的不同部分
- outArr[i] = new RandomAccessFile(OUT_FILE_NAME, "rw");
- if (i != 0) {
- //
- // isArr[i] = new FileInputStream("d:/勇敢的心.rmvb");
- //以指定输出文件创建多个RandomAccessFile对象
- }
- if (i == DOWN_THREAD_NUM - 1) {
- // //最后一个线程读取指定numPerThred+left个字节
- // System.out.println("第"+i+"个线程读取从"+i * numPerThred+"到"+((i + 1) * numPerThred+ left)+"的位置");
- new ReadThread(i * numPerThred, (i + 1) * numPerThred
- + left, outArr[i],keywords,doneSignal).start();
- } else {
- //每个线程负责读取一定的numPerThred个字节
- // System.out.println("第"+i+"个线程读取从"+i * numPerThred+"到"+((i + 1) * numPerThred)+"的位置");
- new ReadThread(i * numPerThred, (i + 1) * numPerThred,
- outArr[i],keywords,doneSignal).start();
- }
- }
- }catch(Exception e){
- e.printStackTrace();
- }
- // finally{
- //
- // }
- //确认所有线程任务完成,开始执行主线程的操作
- try {
- doneSignal.await();
- } catch (InterruptedException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- //这里需要做个判断,所有做read工作线程全部执行完。
- KeyWordsCount k = KeyWordsCount.getCountObject();
- // Map<String,Integer> resultMap = k.getMap();
- System.out.println("指定关键字出现的次数:"+k.getCount());
- }
- }
- package com.thread.multipl.mysolution;
- /**
- * 统计关键字的对象
- * @author 刘峰管理2
- *
- */
- public class KeyWordsCount {
- private static KeyWordsCount kc;
- private int count = 0;
- private KeyWordsCount(){
- }
- public static synchronized KeyWordsCount getCountObject(){
- if(kc == null){
- kc = new KeyWordsCount();
- }
- return kc;
- }
- public synchronized void addCount(int count){
- System.out.println("增加次数:"+count);
- this.count += count;
- }
- public int getCount() {
- return count;
- }
- public void setCount(int count) {
- this.count = count;
- }
- }
运行结果如下:
每个线程读取的字节数:201260字节
Thread[Thread-0,5,main] 需要读的次数:787
Thread[Thread-1,5,main] 需要读的次数:787
Thread[Thread-2,5,main] 需要读的次数:787
Thread[Thread-3,5,main] 需要读的次数:787
Thread[Thread-4,5,main] 需要读的次数:787
Thread[Thread-5,5,main] 需要读的次数:787
Thread[Thread-6,5,main] 需要读的次数:787
Thread[Thread-7,5,main] 需要读的次数:787
Thread[Thread-8,5,main] 需要读的次数:787
Thread[Thread-9,5,main] 需要读的次数:787
增加次数:0
增加次数:146
增加次数:432
增加次数:539
增加次数:587
增加次数:717
增加次数:631
增加次数:467
增加次数:665
增加次数:538
指定关键字出现的次数:4722
我用10个线程去解析金庸大师写的《倚天屠龙记》,“无忌”这个词在这部小说中一共出现了4722次。实在找不到再大一些的文件了。倚天屠龙记.txt的大小4M出头。
关于CountDownLatch类的作用说明:
在API文档中,已经说明是一个辅助类。用于控制主线程与子线程之间切换的一个工具类。用法网上去搜下。ITEYE里也有人讨论过。我在这里使用它解决这样的问题:在确保10个线程都完成文件的解析工作后,系统调用主线程做剩下该做的事情,即:输出“出现的次数”。不确保这点的话,会导致执行完第4个线程,后面的线程还没开始,系统已经做最后一步输出统计结果,这样就达不到我们要的效果。这里解决方案有另一个简单的,自己写个计数器,从10记到1,10个线程嘛。这个看个人喜好吧。
原文:http://babystudyjava.iteye.com/blog/1732814
JAVA多线程读写文件范例的更多相关文章
- java StringBuffer读写文件
java StringBuffer读写文件 StringBuffer的优势 较String:String每更新一次就会new一个新的对象出来,更新次数上去之后,内存开销太大.而StringBuffer ...
- java io读写文件
java io读写文件相关阅读:http://www.cnblogs.com/wing011203/archive/2013/05/03/3056535.html public class DemoI ...
- java(IO)读写文件乱码转换UTF-8问题
java(IO)读写文件乱码转换UTF-8问题 读取文件 String Content = ""; // 文件很长的话建议使用StringBuffer try { FileInpu ...
- SAE java应用读写文件(TmpFS和Storage)-----绝世好代码
近期不少java用户都在提sae读写本地文件的问题,在这里结合TmpFS和Storage服务说说java应用应该如何读写文件TmpFS是一个供应用临时读写的路径,但请求过后将被销毁.出于安全考虑,sa ...
- SAE java应用读写文件(TmpFS和Storage)
近期不少java用户都在提sae读写本地文件的问题,在这里结合TmpFS和Storage服务说说java应用应该如何读写文件TmpFS是一个供应用临时读写的路径,但请求过后将被销毁.出于安全考虑,sa ...
- java 多线程下载文件 以及URLConnection和HttpURLConnection的区别
使用 HttpURLConnection 实现多线程下载文件 注意GET大写//http public class MultiThreadDownload { public static void m ...
- Java基础--读写文件
Java读写文件需要注意异常的处理,下面是一个例子 写文件 public class WriteFile { public static void write(String file, String ...
- 顺序、随机IO和Java多种读写文件性能对比
概述 对于磁盘的读写分为两种模式,顺序IO和随机IO. 随机IO存在一个寻址的过程,所以效率比较低.而顺序IO,相当于有一个物理索引,在读取的时候不需要寻找地址,效率很高. 基本流程 总体结构 我们编 ...
- java 多线程下载文件并实时计算下载百分比(断点续传)
多线程下载文件 多线程同时下载文件即:在同一时间内通过多个线程对同一个请求地址发起多个请求,将需要下载的数据分割成多个部分,同时下载,每个线程只负责下载其中的一部分,最后将每一个线程下载的部分组装起来 ...
随机推荐
- 链接 DB App.config 解析
<?xml version="1.0" encoding="utf-8"?><configuration> <startup> ...
- Java-悲观锁和乐观锁
Java中的乐观锁与悲观锁: 1. Java中典型的synchronized就是一种悲观锁,也就是独占锁,不过JDK1.6之后对synchronized已经做了许多优化,也不能说是完全的悲观锁了: 2 ...
- 看看PHP迭代器的内部执行过程
class myIterator implements Iterator { private $position = 0; private $array = array( "first_el ...
- Codeforces 799B - T-shirt buying(STL)
题目链接:http://codeforces.com/problemset/problem/799/B 题目大意:有n件T恤,每件T体恤都分别有价格(每件衣服的价格不重复).前面的颜色.背部的颜色三种 ...
- golang之结构体和方法
结构体的定义 结构体是将零个或者多个任意类型的命令变量组合在一起的聚合数据类型.每个变量都叫做结构体的成员. 其实简单理解,Go语言的结构体struct和其他语言的类class有相等的地位,但是GO语 ...
- fiddler添加监测请求的 ip地址
本文转载自:http://www.jackness.org/2014/12/26/%E7%BB%99fiddler%E6%B7%BB%E5%8A%A0%E7%9B%91%E6%B5%8B%E8%AF% ...
- linux中的vim 编辑一行内容,如何使光标快速移动到行首和行尾以及行中间某处啊?
光标定位G 移至行行首nG 移至第n行行首n+ 移n行行首n- 移n行行首n$ 移n行(1表示本行)行尾0 所行行首$ 所行行尾^ 所行首字母h,j,k,l 左移移移右移H 前屏幕首行行首M 屏幕显示 ...
- [你必须知道的.NET]第二十三回:品味细节,深入.NET的类型构造器
发布日期:2008.11.2 作者:Anytao © 2008 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. 说在,开篇之前 今天Artech兄在<关于Type Init ...
- IEEEXtreme 10.0 - Always Be In Control
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Always Be In Control 题目来源 第10届IEEE极限编程大赛 https://www.h ...
- c++ primer 2 变量和基本类型
2.1 基本内置类型 基本内置类型是C++“自带”的类型,区别于标准库定义的类型.使用时不需要应用标准库就可以使用,我们可以理解为数字型有下面这些 整形:就是整数或者是没有小数位的数.它包括bool( ...