java 中String与StringBuilder 效率
之前印象中string与stringbuilder操作时,如果多次改变string就使用stringbuilder,效率会提高;
今天实际遇到了问题,亲身经历过之后,这性能不是一般的影响啊;不是同一个数量级的;
场景描述:
一个包含50719条记录的excel文件,读取其中的内容,通过逗号分隔的方式拼接成字符串;
如果使用string要耗时三四十分钟;有可能更长,并且耗费更多的内存;
如果使用stringbuilder只需要几秒钟;有可能更短;
测试代码如下:
使用stringbuilder
/**
* v1.1 excel转txt 输入excel流,返回txt流
* @param ins excel流
* @param filename2 txt临时文件路径
* @param split_char 分割符
* @param Column 总列数
* */
private boolean ExcelToTxt(InputStream ins, String filename2, String split_char, String Column){
boolean flag = false;
Workbook rwb = null;
FileOutputStream fops = null;
Sheet rs = null;
Cell cell = null;
String split_char_real = ",";//实际分隔符
if(split_char!=null && split_char.trim().length()>0){
split_char_real = split_char;
}
//获得文件名
//String filename1 = filename.substring(0,filename.lastIndexOf("."));
//路径+文件名+.txt后缀
//String filename2 = path+"/"+filename1+".txt";
//创建txt临时文件
File newFile = new File(filename2);
try {
//从输入流创建Workbook
rwb = Workbook.getWorkbook(ins);
//获取第一张Sheet表
rs = rwb.getSheet(0);
//获取Sheet表中所包含的总列数
int rsColumns = rs.getColumns();
if(Column!=null && Column.trim().length()>0){
rsColumns = Integer.valueOf(Column);//使用配置的列数
}
//获取Sheet表中所包含的总行数
int rsRows = rs.getRows();
//拼接内容
StringBuilder s = new StringBuilder();
for(int i=0;i<rsRows;i++){ // cell = rs.getCell(0,i);//每行的第一个
// if(cell.getContents().trim().length()==0){//如果第一列为空就结束读取excel
// break;
// }
//判断是否为空行
try {
StringBuilder rowData = new StringBuilder();
for(int j=0;j<rsColumns;j++){
cell = rs.getCell(j,i);
rowData.append(cell.getContents());
}
if (new String(rowData).trim().length() == 0) { //如果为空行,就结束读取excel
break;
}
} catch (Exception e) {
System.out.println("*****DataImport*********** 判断空行出错");
e.printStackTrace();
break;
} for(int j=0;j<rsColumns;j++){
cell = rs.getCell(j,i);
String content = cell.getContents();
s.append(content.replaceAll("\n", "") );
s.append(split_char_real);//支持换行
}
if(s.length()>0) s.delete(s.length()-split_char_real.length(),s.length());//s = s.substring(0,s.length()-split_char_real.length());
//
/*if(s.length()>0 ){
int count = StringUtils.countMatches(s, split_char_real);
//&& s.split(split_char_real).length>0 && s.split(split_char_real).length < rsColumns
s = s + split_char_real+ " ";
}*/
if(s.lastIndexOf(split_char_real) == (s.length() - split_char_real.length())){
s.append(" ");
}
if(i<rsRows-1) s.append("\r\n"); }
System.out.println("--------DataImport---------excel转换生成的txt文件内容:"+newFile.getAbsolutePath());
if(!newFile.exists()){//如果文件不存在,就创建一个新的
newFile.createNewFile();
}
fops = new FileOutputStream(newFile);//使用输出流
fops.write(s.toString().getBytes());//写入文本文件
flag = true;//转换成功
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (BiffException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally{
//操作完成时,关闭对象,释放占用的内存空间
rwb.close();
try {
fops.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return flag; } public static void main(String[] args) {
DataImport di = new DataImport();
try
{
di.ExcelToTxt(new FileInputStream("f:\\团圆行动营销成功客户上传_20140610.xls"), "f:\\2014.txt", ",", "10");
} catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
使用string
/**
* v1.1 excel转txt 输入excel流,返回txt流
* @param ins excel流
* @param filename2 txt临时文件路径
* @param split_char 分割符
* @param Column 总列数
* */
private boolean ExcelToTxt(InputStream ins, String filename2, String split_char, String Column){
boolean flag = false;
Workbook rwb = null;
FileOutputStream fops = null;
Sheet rs = null;
Cell cell = null;
String split_char_real = ",";//实际分隔符
if(split_char!=null && split_char.trim().length()>0){
split_char_real = split_char;
}
//获得文件名
//String filename1 = filename.substring(0,filename.lastIndexOf("."));
//路径+文件名+.txt后缀
//String filename2 = path+"/"+filename1+".txt";
//创建txt临时文件
File newFile = new File(filename2);
try {
//从输入流创建Workbook
rwb = Workbook.getWorkbook(ins);
//获取第一张Sheet表
rs = rwb.getSheet(0);
//获取Sheet表中所包含的总列数
int rsColumns = rs.getColumns();
if(Column!=null && Column.trim().length()>0){
rsColumns = Integer.valueOf(Column);//使用配置的列数
}
//获取Sheet表中所包含的总行数
int rsRows = rs.getRows();
//拼接内容
String s = "";
for(int i=0;i<rsRows;i++){ // cell = rs.getCell(0,i);//每行的第一个
// if(cell.getContents().trim().length()==0){//如果第一列为空就结束读取excel
// break;
// }
//判断是否为空行
try {
StringBuilder rowData = new StringBuilder();
for(int j=0;j<rsColumns;j++){
cell = rs.getCell(j,i);
rowData.append(cell.getContents());
}
if (new String(rowData).trim().length() == 0) { //如果为空行,就结束读取excel
break;
}
} catch (Exception e) {
System.out.println("*****DataImport*********** 判断空行出错");
e.printStackTrace();
break;
} for(int j=0;j<rsColumns;j++){
cell = rs.getCell(j,i);
String content = cell.getContents();
s = s + content.replaceAll("\n", "") + split_char_real;//支持换行
}
if(s.length()>0) s = s.substring(0,s.length()-split_char_real.length());
//
if(s.length()>0 && s.split(split_char_real).length>0 && s.split(split_char_real).length < rsColumns){
s = s + split_char_real+ " ";
}
if(i<rsRows-1) s = s + "\r\n"; }
System.out.println("--------DataImport---------excel转换生成的txt文件内容:"+s);
if(!newFile.exists()){//如果文件不存在,就创建一个新的
newFile.createNewFile();
}
fops = new FileOutputStream(newFile);//使用输出流
fops.write(s.getBytes());//写入文本文件
flag = true;//转换成功
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (BiffException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally{
//操作完成时,关闭对象,释放占用的内存空间
rwb.close();
try {
fops.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return flag; } public static void main(String[] args) {
DataImport di = new DataImport();
try
{
di.ExcelToTxt(new FileInputStream("f:\\团圆行动营销成功客户上传_20140610.xls"), "f:\\2014.txt", ",", "10");
} catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
java 中String与StringBuilder 效率的更多相关文章
- 探秘Java中String、StringBuilder以及StringBuffer
探秘Java中String.StringBuilder以及StringBuffer 相信String这个类是Java中使用得最频繁的类之一,并且又是各大公司面试喜欢问 到的地方,今天就来和大家一起学习 ...
- JAVA中String和StringBuilder类的特点及使用
转自:https://www.imooc.com/code/2202 仅做个人学习记录之用,侵删! 什么是 Java 中的字符串 在 Java 中,字符串被作为 String 类型的对象处理. Str ...
- java中String、StringBuilder、StringBuffer三者的区别
在Java项目开发中,字符串是最长使用的数据类型,而有关字符串的String.StringBuilder.StringBuffer三者又常常让人分不清楚什么时候该使用哪个. 特此整理一下. Strin ...
- Java中String、StringBuilder以及StringBuffer
原文出处: 海子 相信String这个类是Java中使用得最频繁的类之一,并且又是各大公司面试喜欢问到的地方,今天就来和大家一起学习一下String.StringBuilder和StringBuffe ...
- java中String与StringBuilder的区别
相信大家对 String 和 StringBuffer 的区别也已经很了解了,但是估计还是会有很多同志对这两个类的工作原理有些不清楚的地方,今天我在这里重新把这个概念给大家复习一下,顺便牵出 J2SE ...
- java中 String StringBuffer StringBuilder的区别
* String类是不可变类,只要对String进行修改,都会导致新的对象生成. * StringBuffer和StringBuilder都是可变类,任何对字符串的改变都不会产生新的对象. 在实际使用 ...
- java中String,StringBuffer,StringBuilder之间的区别
文章转载自:http://www.cnblogs.com/frankliiu-java/archive/2010/07/05/1771537.html String是固定长度的字符串,如果要发生变化必 ...
- Java中String、StringBuilder和StringBuffer
StringBuilder和StringBuffer内部都是通过char[]来实现的.(jdk1.9后,底层把char 数组变成了byte[].)唯一不同的就是我们的StringBuffer内部操作方 ...
- java中String、stringbuilder、stringbuffer区别
1.可变与不可变 String类中使用字符数组保存字符串,如下就是,因为有"final"修饰符,所以可以知道string对象是不可变的.每次对String对象进行改变的时候其实都等 ...
随机推荐
- 线程间的通信----wait/notify机制
wait/notify机制 实现多个线程之间的通信可以使用wait.notify.notifyAll三个方法.这三个方法都是Object类的方法.wait():导致当前线程等待,直到另一个线程调用此对 ...
- [C++]_[获取Utf8字符串的字符个数和子字符串]
场景: 1.有时候须要统计utf8字符串的个数,单纯统计字节个数是不行的. 2.有时候也须要获取从某个位置開始的n个连续字符用于显示或计算. static int GetUtf8LetterNumbe ...
- express 不是内部或外部命令(windows)解决方式
请环境变量设置 注意 "D:\soft\nodejs\"这个是笔者自己安装在电脑上的node js文件夹,请改动成自己的 假设安装的express是4.X就要安装 express- ...
- ios6--UILabel
// // ViewController.m // 02-UILabel的使用 // // UILabel显示一段文字. #import "ViewController.h" @i ...
- XAML实例教程系列 - 事件(Event) 五
Kevin Fan分享开发经验,记录开发点滴 XAML实例教程系列 - 事件(Event) 2012-06-19 01:36 by jv9, 1727 阅读, 7 评论, 收藏, 编辑 Events, ...
- go语言笔记——切片底层本质是共享数组内存!!!绝对不要用指针指向 slice切片本身已经是一个引用类型就是指针
切片 切片(slice)是对数组一个连续片段的引用(该数组我们称之为相关数组,通常是匿名的),所以切片是一个引用类型(因此更类似于 C/C++ 中的数组类型,或者 Python 中的 list 类型) ...
- hdu 6119 小小粉丝度度熊(尺取)
小小粉丝度度熊 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- nova service-list for juno kilo,liberty openstack
- js获取request参数值(javascript 获取request参数值的方法)
jsp 中的js,可以用el表达式来提取:var value = "${requestScope.XXX}"; 注:XXX为你的参数名 如:http://localhost:808 ...
- String Successor(模拟)
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3490 题意:给出一个字符串,一个操作次数,每次操作从当前字符串最右边的字符 ...