`
jspine
  • 浏览: 103668 次
  • 性别: 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万行的文件,我将毫不怀疑会直接把服务器给挂死。。。(当然,这也取决于服务器的运算力。。。)

 

分享到:
评论
31 楼 lydawen 2010-02-01  
mazzystar 写道
蜗牛创业网 写道
append太多了会溢出。
我的意思是 buffer append太多,在很久以前这件事在我面前发生了

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


应该是已经超出JVM的内存了吧~
30 楼 zhangdp_neu 2010-02-01  
如果相加频率很频繁的话,挺可怕的,算一下,你这10万字符串占多少空间。
29 楼 ansjsun 2010-02-01  
听哥一句用数组吧..先弄个足够大的数组...一行一行放.弄个数字记录.到了数组哪行了.如果不够数组扩容..至于那个需求..list和map都不行..容易内存溢出
28 楼 m7788 2010-02-01  
不说别的,光说这10W次循环@_@
27 楼 蜗牛创业网 2010-02-01  
mazzystar 写道
蜗牛创业网 写道
append太多了会溢出。
我的意思是 buffer append太多,在很久以前这件事在我面前发生了

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


ByteArrayOutputStream
26 楼 icyiwh 2010-02-01  
引用
while((fileLine = in.readLine()) != null) { 


你这样如果有一万行,那需要进行10000次的IO,需要考虑Buffer,减少IO才是提升性能的王道.

  另外,即使用StringBuilder了,也要给其提供合适的capability才合适.
25 楼 hommy8 2010-02-01  
fish2007 写道
JDK5之后,完全可以用“+”操作符来连接字符串,因为在JDK5之后,java会用StringBuilder来处理

真的吗?头一次听到...
24 楼 sunnymoon 2010-02-01  
mobilenumber += fileLine + ",";

String[] mobilenumber_arry = mobilenumber .split(",");
这两个地方都很要命。
字符串相加,会把内存吃光的,谁写的,太有才了。
还有第二行,性能很有问题。数据库设计的方式就有问题。
像这样的大数据量,用分割成数组效率很低的。
23 楼 xzqttt 2010-02-01  
RednaxelaFX 写道
mengke 写道
楼主应该学习了解一下String的类型,这样的频繁拼接应该用StringBuffer

楼主的描述似乎表明这段代码并不是楼主自己写的,而是楼主接手维护的。这种状况让人同情……
而且原本那段代码要表达的意图大概是这样:
List<String> mobilenumber = new ArrayList<String>();
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.add(fileLine);
  }
}
String[] mobilenumber_arry = mobilenumber.toArray(new String[mobilenumber.size()]);

楼主显然也知道这点……只是提醒一下大家说别像顶楼代码那样乱搞而已,吧?


建议楼主看看mobilenumber_arry下面究竟是要做什么用的,当mobilenumber的记录达到一定条数之后,就做一次处理,否则等到字符串都读完了再toArray,可能会OOM!
22 楼 mp_juan 2010-02-01  
如果是要设计多线程的话,建议使用StringBuffer,否则尽量使用StringBuilder
21 楼 sunnymoon 2010-02-01  
没必要用线程安全,StringBuilder效率更高。
20 楼 binlaniua 2010-01-31  
其实
String a = "1" + ..... + ...



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

是一样的..


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

19 楼 vvggsky 2010-01-31  
fish2007 写道
JDK5之后,完全可以用“+”操作符来连接字符串,因为在JDK5之后,java会用StringBuilder来处理


要说清楚开闭区间,Java6(包括)以后才是
18 楼 zcq100 2010-01-31  
试试LinkedList,插入大量数据要比ArrayList来得好
17 楼 495127903 2010-01-31  
RednaxelaFX 写道
mengke 写道
楼主应该学习了解一下String的类型,这样的频繁拼接应该用StringBuffer

楼主的描述似乎表明这段代码并不是楼主自己写的,而是楼主接手维护的。这种状况让人同情……
而且原本那段代码要表达的意图大概是这样:
List<String> mobilenumber = new ArrayList<String>();
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.add(fileLine);
  }
}
String[] mobilenumber_arry = mobilenumber.toArray(new String[mobilenumber.size()]);

楼主显然也知道这点……只是提醒一下大家说别像顶楼代码那样乱搞而已,吧?

用List的话会不会也会出现异常. 我记得有个项目出现过这种情况.. 就是List存放大的数据量. 结果程序都瘫痪了.

小弟刚入门,不懂的事情就问.. 呵呵.. 希望不要拍砖.
16 楼 fish2007 2010-01-31  
JDK5之后,完全可以用“+”操作符来连接字符串,因为在JDK5之后,java会用StringBuilder来处理
15 楼 mazzystar 2010-01-31  
蜗牛创业网 写道
append太多了会溢出。
我的意思是 buffer append太多,在很久以前这件事在我面前发生了

能说说最后怎么解决的吗?
14 楼 raymond2006k 2010-01-30  
用StringBuilder等都没错。

不过,我觉的这种场景的数据量到一定程度,需要重构的主要是设计。

就像分页一样,内存里一次只处理一定数量,例如:200 个,采用stream或异步方式等方式逐步处理。总之,java弱的就是内存,不要做大内存操作。
13 楼 seraphim871211 2010-01-30  
mengke 写道
楼主应该学习了解一下String的类型,这样的频繁拼接应该用StringBuffer



你应该学习认真看别人的帖子,LZ显然知道频繁拼接需要用什么,LZ应该是提醒大家这样的场景根本不需要字符串拼接。
12 楼 javaonejcy 2010-01-30  
毫无疑问这是个设计问题,不论用什么,都不要给系统留下频繁大负载的机会。

相关推荐

    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