java多线程批量读取文件( 八)--读写分离
package com.net.thread.future; import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.Set;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future; /**
* @author
* @Time:2017年8月16日 下午5:26:37
* @version 1.0
* @description
*/
public class CallableDemo3 { final static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); public static void main(String[] args)
{
File f = new File("C://Users//LENOVO//Desktop//file");
// 文件总数
final List<File> filePathsList = new ArrayList<File>();
File[] filePaths = f.listFiles();
for (File s : filePaths) {
filePathsList.add(s);
} CountDownLatch latch = new CountDownLatch(filePathsList.size());
ExecutorService pool = Executors.newFixedThreadPool(10); BlockingQueue<Future<Map<String, FileInputStream>>> queue =
new ArrayBlockingQueue<Future<Map<String, FileInputStream>>>(100); System.out.println("-------------文件读、写任务开始时间:" + sdf.format(new Date()));
for (int i = 0; i < filePathsList.size(); i++) {
File temp = filePathsList.get(i);
Future<Map<String, FileInputStream>> future = pool.submit(new MyCallableProducer(latch, temp));
queue.add(future); pool.execute(new MyCallableConsumer(queue));
} try {
latch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("-------------文件读、写任务结束时间:" + sdf.format(new Date()));
pool.shutdownNow();
} // 文件读线程
static class MyCallableProducer implements Callable<Map<String, FileInputStream>>
{
private CountDownLatch latch;
private File file;
private FileInputStream fis = null;
private Map<String, FileInputStream> fileMap = new HashMap<String, FileInputStream>(); public MyCallableProducer(CountDownLatch latch, File file)
{
this.latch = latch;
this.file = file;
} @Override
public Map<String, FileInputStream> call() throws Exception
{
System.out.println(Thread.currentThread().getName() + " 线程开始读取文件 :" + file.getName() + " ,时间为 "+ sdf.format(new Date()));
fis = new FileInputStream(file);
fileMap.put(file.getName(), fis);
doWork();
System.out.println(Thread.currentThread().getName() + " 线程读取文件 :" + file.getName() + " 完毕" + " ,时间为 "+ sdf.format(new Date()));
latch.countDown();
return fileMap;
} private void doWork()
{
//此方法可以添加一些业务逻辑,比如包装pojo等等操作,返回的值可以是任何类型
Random rand = new Random();
int time = rand.nextInt(10) * 1000;
try {
Thread.sleep(time);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
} // 文件写线程
static class MyCallableConsumer implements Runnable
{
private String fileName = "";
private BlockingQueue<Future<Map<String, FileInputStream>>> queue;
private FileInputStream fis = null;
private File dirFile = null; private BufferedReader br = null;
private InputStreamReader isr = null;
private FileWriter fw = null;
private BufferedWriter bw = null; public MyCallableConsumer(BlockingQueue<Future<Map<String, FileInputStream>>> queue2)
{
this.queue = queue2;
} @Override
public void run()
{
try {
Future<Map<String, FileInputStream>> future = queue.take();
Map<String, FileInputStream> map = future.get(); Set<String> set = map.keySet();
for (Iterator<String> iter = set.iterator(); iter.hasNext();) { fileName = iter.next().toString();
fis = map.get(fileName); System.out.println(Thread.currentThread().getName() + " 线程开始写文件 :" + fileName + " ,时间为 "+ sdf.format(new Date()));
try {
isr = new InputStreamReader(fis, "utf-8");
br = new BufferedReader(isr); dirFile = new File("d:" + File.separator + "gc3" + File.separator + fileName);
fw = new FileWriter(dirFile);
bw = new BufferedWriter(fw); String data = "";
bw.write("+++++++++++++" + Thread.currentThread().getName() + " 线程开始写文件++++++++++++");
while ((data = br.readLine()) != null) {
bw.write(data + "\r");
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
bw.close();
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
} }
说明:
1、其实构思很简单,阻塞队列是线程安全的,那么我多线程就用阻塞队列,这样可以保证每个写线程拿到的具体内容不同,不会导致重复写数据;
2、我使用异步线程进行读写,而非同步线程,这样有助于提升整体读、写性能。
3、CountDownLatch是信号灯,功能类似于join()方法,当然也可以使用CyclicBarrier
java多线程批量读取文件( 八)--读写分离的更多相关文章
- java多线程批量读取文件(七)
新公司入职一个多月了,至今没有事情可以做,十来个新同事都一样抓狂,所以大家都自己学习一些新东西,我最近在看zookeeper,感觉蛮不错的,和微服务的zuul以及eureka功能类似,只是代码复杂了一 ...
- java多线程批量下载文件
多线程下载文件 平时开发中有时会用到文件下载,为了提高文件的下载速率,采用多线程下载能够达到事半功倍的效果: package test; /** * 文件下载类 * @author luweichen ...
- JAVA多线程下载网络文件
JAVA多线程下载网络文件,开启多个线程,同时下载网络文件. 源码如下:(点击下载 MultiThreadDownload.java) import java.io.InputStream; im ...
- R8—批量生成文件夹,批量读取文件夹名称+R文件管理系统操作函数
一. 批量生成文件夹,批量读取文件夹名称 今日,工作中遇到这样一个问题:boss给我们提供了200多家公司的ID代码(如6007.7920等),需要根据这些ID号去搜索下载新闻,从而将下载到的新闻存到 ...
- 使用Perl批量读取文件最后行
使用Perl批量读取文件最后行 面对成百上千个文件,有时我们需要查看它的最后行,单个文件打开将耗费大量时间,而通过Perl提取出最后行,将快速的帮助我们处理繁琐的事务. 特性 整个目录完全遍历,自动提 ...
- matlab文件读写处理实例(二)——textread批量读取文件
问题:对文件夹下所有文件进行批量读取,跳过文件头部分,读取每个文件数据部分的7,8,9列,保存到变量并且输出到文件. 数据: 文件夹11m\
- Java io实现读取文件特殊内容进行替换
最近公司在做一个项目其中一个需求是读取文件中的特殊字符在其后进行添加或删除字符操作,本来想直接使用randomAccessFile按行读取,读取到特殊字符就进行添加或删除操作,但是randomAcce ...
- [JAVA 多种方式读取文件]
package com.file; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream ...
- Java相对路径读取文件
不管你是新手还是老鸟,在程序中读取资源文件总会遇到一些找不到文件的问题,这与Java底层的实现有关,不能算bug,只要方法得当,问题还是可以解决的. 项目的文件夹结构: repathtest ├─sr ...
随机推荐
- MVC之ViewData.Model
在MVC中前台Razor视图呈现数据的方式不止一种.举个简单的Demo,我们要把用户信息呈现给人民. 一.ViewData.Model的使用,先简单写一下Razor @model User---- ...
- Java基础入门 - Hello world
JDK安装完成后,将一下代码写入文件HelloWorld.java中 // HelloWorld.java public class HelloWorld { public static void m ...
- 使用swagger时遇到的问题
后端写好接口后开始和前端进行联调,为了减少时间成本或者说是后端不想写文档,所以便想使用一套可以自动化生成api接口文档的工具,swagger正是可以解决这一需求.于是很早之前就想把swagger集成到 ...
- C#设计模式--适配器模式(结构型模式)
一.适配器模式介绍: 适配器模式:将一个类的接口,转换成客户希望的另外一个接口.adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作 例子分析(充电器充电): 模式中的角色: 安 ...
- JavaScript 闭包的详细分享(三种创建方式)(附小实例)
JavaScript闭包的详细理解 一.原理:闭包函数--指有权访问私有函数里面的变量和对象还有方法等:通俗的讲就是突破私有函数的作用域,让函数外面能够使用函数里面的变量及方法. 1.第一种创建方式 ...
- java Date equals 的坑
今天在JDK6上做开发,遇到一个很诡异的问题. Domain中一个实体是Date,称为变量 a, 使用Calendar构造出来的Date,称为变量b, 虽然都是同一天,比如 2016-11-11 00 ...
- MySQL入门很简单: 10 mysql运算符
1. 算术运算符 例子: 将t1表中字段a的值进行加法,减法和乘法 2. 比较运算符 注:LIKE经常和通配符"_"和"%"一起使用,"_" ...
- "提取位于北坡的各类用地面积信息"的程序设计与实现
"提取位于北坡的各类用地面积信息"的程序设计与实现 程序员:左正康 发表时间:2013/12/20 14:24 代号:黑眼圈的日子 第一步:导入dem ...
- 7 MSSQL数据库备份与恢复
0 MSSQL数据库备份 1 SQLAgent配置 2 设置连接属性 3 输入SA账号密码 4 SQL备份脚本配置 5 生成SQL全量备份脚本 6 生成SQL差异备份脚本 7 修改SQL差异备份脚本 ...
- R 语言爬虫 之 cnblog博文爬取
Cnbolg Crawl a). 加载用到的R包 ##library packages needed in this case library(proto) library(gsubfn) ## Wa ...