`
jspine
  • 浏览: 103653 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

10万个字符串相加会发生什么??

    博客分类:
  • java
阅读更多

最近在检查公司平台的一个BUG,BUG的具体症状是:平台用户上传了一个附件,提交时需要等待30-50分钟,有时根本传不上去。

解决BUG:

      1、我先检查了平台的日志,发现相关业务没有发生异常,检查了整个WEB服务器的日志,发现有很多"TNS:connection close"。接着我就怀疑是由于平台的链接池被耗尽,于是我就重新配置链接池,定期回收超时链接。

       2、代码上线后,用户反应该问题还是存在,但是现在可以提交了,不会出现提交不上去。但是现在还是很慢速度并没有上去。并且运维同事反一旦用户提交后,系统的CPU负载会达到一个很可怕的数量级。于是我把该业务模块的所有代码拿出来一个个剖析跟踪。结果真发现了问题,即使在本地,提交一个10万行左右的文件,也很慢。。。

慢慢一行代码段出现在了我的面前具体如下:

while((fileLine = in.readLine()) != null) {
                line_num++;
                if(CTools.checkMobileValid(fileLine)!=0) {
                  text_error = "error";
                  send_hash.put(String.valueOf(hash_num), line_num + "_" + "您输入的手机号码无效!");
                  hash_num++;
                }
                else {
                 mobilenumber += fileLine + ",";
                }
              }

 String[] mobilenumber_arry = mobilenumber .split(",");

 

mobilenumber += fileLine + ",";

 String[] mobilenumber_arry = mobilenumber .split(",");

 

 请注意代码中红色 字体部分,另外该循环大概有10万行。

看到上面的代码我很是郁闷....,这样的思路我是理解不了,也看不明白,贴一个字符串测试例子:

http://blog.csdn.net/chuan122345/archive/2007/08/12/1739806.aspx

 

通过这个例子,我们可以学习到字符串链接,有时候真会死机的。。。如果传一个20万行的文件,我将毫不怀疑会直接把服务器给挂死。。。(当然,这也取决于服务器的运算力。。。)

 

分享到:
评论
51 楼 JavaFans 2010-02-16  
<div class="quote_title">猜想:</div>
<div class="quote_title">对于编译期间就能确定的String值,的确就是如楼上所说。</div>
<div class="quote_title">理由是编译器做过优化了,他能够用对待StringBuffer的方式来优化String。</div>
<div class="quote_title">但是在运行时才确定的String值,也就是楼主的情况下,我对此表示怀疑。<br>
</div>
<div class="quote_title"><br></div>
<div class="quote_title">chenguanwei2008 写道</div>
<div class="quote_div">
<div class="quote_title">kevintse 写道</div>
<div class="quote_div">
<div class="quote_title">binlaniua 写道</div>
<div class="quote_div">其实 <br>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br><br>是一样的.. <br><br>和 <br>String a = "1"; <br>a += "b"; <br>就不一样了。。 <br><br>
</div>
<br><br>Oh, my god...不知道你是在哪里看到的~ <br>"1" + ..... + ...居然跟StringBuffer.append()是一样的? <br>String.+ 与StringBuffer.append()相比,性能是很低效的。String.+每加一次就生成一个临时对象,StringBuffer.append()则是append过程中不断重新分配内存(次数要远远低于String.+)。 <br>拼接大量字符串的时候,用StringBuffer.append()的性能远远高于String.+,只是相对耗费比较多的内存。 <br>你可以测试一下,或者看一下String和StringBuffer的源代码。</div>
<p>binlaniua说的是对的,你可以写代码试试</p>
<p>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br>把编译得到的class文件用java -p反编译一次,观察得到的字节码你就知道啦</p>
<p> </p>
</div>
<p> </p>
50 楼 JavaFans 2010-02-16  
<div class="quote_title">有意思,学习<br>
</div>
<div class="quote_title"><br></div>
<div class="quote_title">chenguanwei2008 写道</div>
<div class="quote_div">
<div class="quote_title">kevintse 写道</div>
<div class="quote_div">
<div class="quote_title">binlaniua 写道</div>
<div class="quote_div">其实 <br>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br><br>是一样的.. <br><br>和 <br>String a = "1"; <br>a += "b"; <br>就不一样了。。 <br><br>
</div>
<br><br>Oh, my god...不知道你是在哪里看到的~ <br>"1" + ..... + ...居然跟StringBuffer.append()是一样的? <br>String.+ 与StringBuffer.append()相比,性能是很低效的。String.+每加一次就生成一个临时对象,StringBuffer.append()则是append过程中不断重新分配内存(次数要远远低于String.+)。 <br>拼接大量字符串的时候,用StringBuffer.append()的性能远远高于String.+,只是相对耗费比较多的内存。 <br>你可以测试一下,或者看一下String和StringBuffer的源代码。</div>
<p>binlaniua说的是对的,你可以写代码试试</p>
<p>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br>把编译得到的class文件用java -p反编译一次,观察得到的字节码你就知道啦</p>
<p> </p>
</div>
<p> </p>
49 楼 chenguanwei2008 2010-02-14  
<div class="quote_title">kevintse 写道</div>
<div class="quote_div">
<div class="quote_title">binlaniua 写道</div>
<div class="quote_div">其实 <br>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br><br>是一样的.. <br><br>和 <br>String a = "1"; <br>a += "b"; <br>就不一样了。。 <br><br>
</div>
<br><br>Oh, my god...不知道你是在哪里看到的~ <br>"1" + ..... + ...居然跟StringBuffer.append()是一样的? <br>String.+ 与StringBuffer.append()相比,性能是很低效的。String.+每加一次就生成一个临时对象,StringBuffer.append()则是append过程中不断重新分配内存(次数要远远低于String.+)。 <br>拼接大量字符串的时候,用StringBuffer.append()的性能远远高于String.+,只是相对耗费比较多的内存。 <br>你可以测试一下,或者看一下String和StringBuffer的源代码。</div>
<p>binlaniua说的是对的,你可以写代码试试</p>
<p>String a = "1" + ..... + ... <br><br>和 <br><br>StringBuffer sbuff = new StringBuffer(); <br>sbuff.append(..).append(..) <br>把编译得到的class文件用java -p反编译一次,观察得到的字节码你就知道啦</p>
<p> </p>
48 楼 shellfj 2010-02-10  
楼主,明显对Java代码的优化没有什么概念。
47 楼 ftp51423121 2010-02-10  
StringBuilder
46 楼 lkj107 2010-02-08  
10W个简单的字符串,OOM的几率很小吧,得有多大的并发a
StringBuffer比较可靠
45 楼 ronalke 2010-02-08  
在数据库拼接查询串等地方可以用StringBuffer,应该就不会遇到这样的问题了
44 楼 luckyEveryOne 2010-02-03  
我很佩服楼主 不用StringBuilder
43 楼 tiny051401 2010-02-03  
很恐怖的代码,像我们这样的小服务器,碰到这样的代码,也只有死的份了。觉得开始设计的就有问题,就不应该这么相加。
42 楼 zcy860511 2010-02-02  
自己写个缓存吧,数量过大,真大到JVM分配的内存都不够用了,一样OutOfMemory
41 楼 andy54321 2010-02-02  
到底这个需求该如何解决呢?
用什么数据结构来存放才是王道?
楼主的解决方法效率、性能提升多少?
40 楼 抛出异常的爱 2010-02-02  
传arry不传iterator吧.
39 楼 elmar 2010-02-02  
zhangdp_neu 写道
如果相加频率很频繁的话,挺可怕的,算一下,你这10万字符串占多少空间。

一个字符串20个字符,40个byte,10万也就4M而已
38 楼 elmar 2010-02-02  
ansjsun 写道
听哥一句用数组吧..先弄个足够大的数组...一行一行放.弄个数字记录.到了数组哪行了.如果不够数组扩容..至于那个需求..list和map都不行..容易内存溢出

你以为ArrayList会溢出的话,数组就不溢出了??
37 楼 billy1977 2010-02-02  
kasirin 写道
Scenery 写道
我觉得是设计思想的问题,不要说上传十万个字符串,就是上传一百万个字符串,就现在而言也不是没有,问题不是在于把String改成StringBuilder就OK了,问题在于怎么往上传,如果把这些字符串往数据库里这么一存,肯定不是一个好注意,无论是上传还是下载都会花费大量的时间,更别提对这些内容进行修改。所以我认为大于五千个字符串的不如保存成文件,在数据库里只要记一下这个文件的路径就可以了。




感觉这蛮好的

这个显然自以为是了,从楼主透露出的一些信息能看出那个文件里保存的是电话号码,他需要把这些电话号码保存到数据库里才能进行查询、统计,或者避免重复等操作,如果四个人上传了十个这样的文件,文件里的电话号码可能重复,现在需要统计一下一共有多少电话号码,数据库里select count(*)...,文件里就麻烦了呀。如果他需要给这些号码发短信,并记录发送次数,这个发送次数难道也记录在文件里吗?

36 楼 kasirin 2010-02-02  
Scenery 写道
我觉得是设计思想的问题,不要说上传十万个字符串,就是上传一百万个字符串,就现在而言也不是没有,问题不是在于把String改成StringBuilder就OK了,问题在于怎么往上传,如果把这些字符串往数据库里这么一存,肯定不是一个好注意,无论是上传还是下载都会花费大量的时间,更别提对这些内容进行修改。所以我认为大于五千个字符串的不如保存成文件,在数据库里只要记一下这个文件的路径就可以了。




感觉这蛮好的
35 楼 kevintse 2010-02-02  
binlaniua 写道
其实
String a = "1" + ..... + ...



StringBuffer sbuff = new StringBuffer();
sbuff.append(..).append(..)

是一样的..


String a = "1";
a += "b";
就不一样了。。



Oh, my god...不知道你是在哪里看到的~
"1" + ..... + ...居然跟StringBuffer.append()是一样的?
String.+ 与StringBuffer.append()相比,性能是很低效的。String.+每加一次就生成一个临时对象,StringBuffer.append()则是append过程中不断重新分配内存(次数要远远低于String.+)。
拼接大量字符串的时候,用StringBuffer.append()的性能远远高于String.+,只是相对耗费比较多的内存。
你可以测试一下,或者看一下String和StringBuffer的源代码。
34 楼 bonny 2010-02-02  
蜗牛创业网 写道
mazzystar 写道
蜗牛创业网 写道
append太多了会溢出。
我的意思是 buffer append太多,在很久以前这件事在我面前发生了

能说说最后怎么解决的吗?


ByteArrayOutputStream

正道,以前处理一个10m的文件,string半个小时都处理不完,流瞬间就好了。
33 楼 Scenery 2010-02-01  
我觉得是设计思想的问题,不要说上传十万个字符串,就是上传一百万个字符串,就现在而言也不是没有,问题不是在于把String改成StringBuilder就OK了,问题在于怎么往上传,如果把这些字符串往数据库里这么一存,肯定不是一个好注意,无论是上传还是下载都会花费大量的时间,更别提对这些内容进行修改。所以我认为大于五千个字符串的不如保存成文件,在数据库里只要记一下这个文件的路径就可以了。
32 楼 shoushou2001 2010-02-01  
楼主不错啊,能发现这问题,呵呵。

相关推荐

    C代码实现超长整数字符串 相加,及相应执行程序

    3. 按位相加:从低位到高位遍历两个字符串,将对应位的数字相加,并考虑进位。对于每一位,我们计算两数之和加上上一位的进位,然后根据结果更新当前位。如果和大于9,则需要进行进位,将值减去10,并将进位设置为1...

    输入10个字符串到一个字符串数组

    ` 声明了一个长度为 10 的字符串数组。 - **数组初始化**:`str[i] = scanner.next();` 从用户处接收输入并将其存储在数组中。 #### 3.2 遍历数组 - **for 循环**:使用 for 循环遍历数组。例如,`for (int i = 0; ...

    java-leetcode题解之第415题字符串相加.zip

    在本压缩包“java-leetcode题解之第415题字符串相加.zip”中,包含的是关于LeetCode第415题“字符串相加”(Add Strings)的Java解决方案。这道题目属于计算机编程领域,特别是Java语言的学习与算法实践。LeetCode是...

    字符串相加(竖式相加+进位考虑)1

    在给定的问题中,我们需要实现一个函数来计算两个以字符串形式表示的非负整数的和。这个问题可以被归类为字符串处理和基本数学运算的结合。以下是对这个任务的详细解析: 首先,题目提供了几个关键限制: 1. `num1`...

    将字符串转会为十六进制

    在IT领域,将字符串转换为十六进制是一个常见的需求,尤其是在数据处理、加密解密算法、网络通信等场景中。本文将深入解析如何实现这一功能,包括基础概念、代码实现及注意事项。 ### 基础概念 #### 字符串与十六...

    字符串比较问题对于长度相同的2 个字符串A和B,其距离定义为相应位置字符距离之和。2 个非空格

    对于长度相同的2 个字符串A和B,其距离定义为相应位置字符距离之和。2 个非空格 字符的距离是它们的ASCII码之差的绝对值。空格与空格的距离为0;空格与其它字符的距 离为一定值k。 在一般情况下,字符串A和B的...

    获取一个字符串中的数字组

    标题中的“获取一个字符串中的数字组”是指从包含数字和非数字字符的字符串中提取连续的数字,并将它们作为整数处理。描述中提到的任务是输入一个字符串,例如"a123x456_17960? 302tab5876",从中找出所有的连续数字...

    c语言两个字符串的连接

    在C语言中,字符串连接是将两个或多个字符串合并成一个新字符串的过程。这个操作在编程中非常常见,尤其是在处理用户输入或者格式化输出时。本文将深入探讨如何在C语言中实现两个字符串的连接。 首先,理解C语言中...

    字符串处理函数列表,字符串处理函数列表

    15. strcat:连接两个字符串,将第二个字符串追加到第一个字符串的末尾。 16. strchr:查找字符串中第一个出现的指定字符,返回指向该字符的指针。 17. strcmp:比较两个字符串,根据字典顺序返回它们的相对关系。...

    C# 字符串转十六进制串,16进制反向转回原字符串

    注意,由于Unicode编码中每个字符可能需要两个字节来表示,所以在转换过程中必须确保原始字符串和十六进制串的长度是偶数,否则可能会丢失字符信息。 这些方法可以帮助你在C#中灵活地处理字符串和十六进制串之间的...

    我的vb程序(10进制替换10个字符串程序源代码).zip

    标题中的“我的vb程序(10进制替换10个字符串程序源代码).zip”表明这是一个使用Visual Basic(VB)编程语言编写的程序,主要功能是实现10进制数值与10个预设字符串之间的替换操作。VB是一种广泛使用的事件驱动编程...

    汇编程序 字符串排序

    从键盘输入若干个字符串(5~15个),每一串的长度不超过20个字符,请将它们做升序排序并在屏幕上显示。编程要求:Enter键结束一个字串的输入,连续两个Enter键结束整个字串的输入。人机对话输入数据,界面友好,容错...

    截取字符串

    在编程领域,字符串操作是一项基础且重要的任务,而“截取字符串”是其中的一个关键功能。这个小demo展示了如何在不同的编程环境中实现字符串截取,同时也涉及到字符串的空格处理。接下来,我们将深入探讨这两个主题...

    labview 16进制字符串强制转换为10进制字符串

    本VI实现16进制字符串强制转换为10进制字符串,即输入16进制的03DF本VI可以输出10进制的03DF,如果需要区分大小写,请自行添加强制转换为大写或小写

    labview 10进制字符串强制转换为16进制字符串

    本VI实现十进制字符串强制转换为16进制字符串,即输入10进制的0CDA可输出16进制的0CDA

    将俩个字符串连接起来,用C语言实现

    当你想要将两个字符串连接起来时,通常会使用`strcat`或`strncat`这样的库函数,但在这个例子中,我们将探讨如何不依赖这些内置函数来实现字符串连接。 首先,我们需要了解基本的字符串操作。在C语言中,字符串常量...

    wincc字符串函数大全

    strchr 函数的功能是在一个串中查找给定字符的第一个匹配之处。它的用法是 `char *strchr(char *str, char c);`,其中 `str` 是目标字符串,`c` 是要查找的字符。例如: ```c #include #include int main(void) ...

    字符串16进制转10进制

    在计算机编程中,字符串的16进制转10进制是一个常见的字符转换操作,特别是当处理二进制数据或在编程调试过程中遇到16进制数据时。16进制数,也称为十六进制数,是一种逢16进位的数制,使用数字0-9和字母A-F(或小写...

    c++中字符串的介绍

    定义一个字符串其实就是定义一个字符数组,例如:char name[10]; // 定义一个名字是 name 的字符串 也可以定义二维的或者多维的字符数组,例如:char names[10][10]; // 定义一个二维字符数组,通常称这样的数组为...

    分别计算字符串中字母、数字及其他字符的数目

    它包括一个栈区,分配了4096字节的空间,以及数据区,用于存储用户输入的字符串、计数变量和其他辅助信息。 程序首先定义了一些常量,如回车符(CR,0dh)和换行符(LF,0ah)。接着,它为输入提示、结果显示和字符...

Global site tag (gtag.js) - Google Analytics