2021程序效率

上传人:一** 文档编号:21034400 上传时间:2021-04-22 格式:DOCX 页数:16 大小:22.29KB
收藏 版权申诉 举报 下载
2021程序效率_第1页
第1页 / 共16页
2021程序效率_第2页
第2页 / 共16页
2021程序效率_第3页
第3页 / 共16页
资源描述:

《2021程序效率》由会员分享,可在线阅读,更多相关《2021程序效率(16页珍藏版)》请在装配图网上搜索。

1、程序效率通过使用一些辅助性工具来找到程序中的瓶颈,然后就可以对瓶颈部分的代码进行优化。一般有两种方案:即优化代码或更改设计方法。我们一般会选择后者,因为不去调用以下代码要比调用一些优化的代码更能提高程序的性能。而一个设计良好的程序能够精简代码,从而提高性能。下面将提供一些在JAVA程序的设计和编码中,为了能够提高JAVA程序的性能,而经常采用的一些方法和技巧。1对象的生成和大小的调整。JAVA程序设计中一个普遍的问题就是没有好好的利用JAVA语言本身提供的函数,从而常常会生成大量的对象(或实例)。由于系统不仅要花时间生成对象,以后可能还需花时间对这些对象进行垃圾回收和处理。因此,生成过多的对象

2、将会给程序的性能带来很大的影响。例1:关于String ,StringBuffer,+和appendJAVA语言提供了对于String类型变量的操作。但如果使用不当,会给程序的性能带来影响。如下面的语句:String name=new String(HuangWeiFeng);System.out.println(name+is my name);看似已经很精简了,其实并非如此。为了生成二进制的代码,要进行如下的步骤和操作:(1) 生成新的字符串new String(STR_1);(2) 复制该字符串;(3) 加载字符串常量HuangWeiFeng(STR_2);(4) 调用字符串的构架器(C

3、onstructor);(5) 保存该字符串到数组中(从位置0开始);(6) 从java.io.PrintStream类中得到静态的out变量;(7) 生成新的字符串缓冲变量new StringBuffer(STR_BUF_1);(8) 复制该字符串缓冲变量;(9) 调用字符串缓冲的构架器(Constructor);(10) 保存该字符串缓冲到数组中(从位置1开始);(11) 以STR_1为参数,调用字符串缓冲(StringBuffer)类中的append方法;(12) 加载字符串常量is my name(STR_3);(13) 以STR_3为参数,调用字符串缓冲(StringBuffer)类

4、中的append方法;(14) 对于STR_BUF_1执行toString命令;(15) 调用out变量中的println方法,输出结果。由此可以看出,这两行简单的代码,就生成了STR_1,STR_2,STR_3,STR_4和STR_BUF_1五个对象变量。这些生成的类的实例一般都存放在堆中。堆要对所有类的超类,类的实例进行初始化,同时还要调用类极其每个超类的构架器。而这些操作都是非常消耗系统资源的。因此,对对象的生成进行限制,是完全有必要的。经修改,上面的代码可以用如下的代码来替换。StringBuffer name=new StringBuffer(HuangWeiFeng);System

5、.out.println(name.append(is my name.).toString();系统将进行如下的操作:(1) 生成新的字符串缓冲变量new StringBuffer(STR_BUF_1);(2) 复制该字符串缓冲变量;(3) 加载字符串常量HuangWeiFeng(STR_1);(4) 调用字符串缓冲的构架器(Constructor);(5) 保存该字符串缓冲到数组中(从位置1开始);(6) 从java.io.PrintStream类中得到静态的out变量;(7) 加载STR_BUF_1;(8) 加载字符串常量is my name(STR_2);(9) 以STR_2为参数,调

6、用字符串缓冲(StringBuffer)实例中的append方法;(10) 对于STR_BUF_1执行toString命令(STR_3);(11)调用out变量中的println方法,输出结果。由此可以看出,经过改进后的代码只生成了四个对象变量:STR_1,STR_2,STR_3和STR_BUF_1.你可能觉得少生成一个对象不会对程序的性能有很大的提高。但下面的代码段2的执行速度将是代码段1的2倍。因为代码段1生成了八个对象,而代码段2只生成了四个对象。代码段1:String name= new StringBuffer(HuangWeiFeng);name+=is my;name+=name

7、;代码段2:StringBuffer name=new StringBuffer(HuangWeiFeng);name.append(is my);name.append(name.).toString();因此,充分的利用JAVA提供的库函数来优化程序,对提高JAVA程序的性能时非常重要的.其注意点主要有如下几方面;(1) 尽可能的使用静态变量(Static Class Variables)如果类中的变量不会随他的实例而变化,就可以定义为静态变量,从而使他所有的实例都共享这个变量。例:public class fooSomeObject so=new SomeObject();就可以定义为:

8、public class foostatic SomeObject so=new SomeObject();(2) 不要对已生成的对象作过多的改变。对于一些类(如:String类)来讲,宁愿在重新生成一个新的对象实例,而不应该修改已经生成的对象实例。例:String name=Huang;name=Wei;name=Feng;上述代码生成了三个String类型的对象实例。而前两个马上就需要系统进行垃圾回收处理。如果要对字符串进行连接的操作,性能将得更差,因为系统将不得为此生成更多得临时变量,如上例1所示。(3) 生成对象时,要分配给它合理的空间和大小JAVA中的很多类都有它的默认的空间分配大小

9、。对于StringBuffer类来讲,默认的分配空间大小是16个字符。如果在程序中使用StringBuffer的空间大小不是16个字符,那么就必须进行正确的初始化。(4) 避免生成不太使用或生命周期短的对象或变量。对于这种情况,因该定义一个对象缓冲池。以为管理一个对象缓冲池的开销要比频繁的生成和回收对象的开销小的多。(5) 只在对象作用范围内进行初始化。JAVA允许在代码的任何地方定义和初始化对象。这样,就可以只在对象作用的范围内进行初始化。从而节约系统的开销。例:SomeObject so=new SomeObject();If(x=1) thenFoo=so.getXX();可以修改为:i

10、f(x=1) thenSomeObject so=new SomeObject();Foo=so.getXX();2异常(Exceptions)JAVA语言中提供了try/catch来发方便用户捕捉异常,进行异常的处理。但是如果使用不当,也会给JAVA程序的性能带来影响。因此,要注意以下两点:(1) 避免对应用程序的逻辑使用try/catch如果可以用if,while等逻辑语句来处理,那么就尽可能的不用try/catch语句。(2) 重用异常在必须要进行异常的处理时,要尽可能的重用已经存在的异常对象。以为在异常的处理中,生成一个异常对象要消耗掉大部分的时间。3. 线程(Threading)一个

11、高性能的应用程序中一般都会用到线程。因为线程能充分利用系统的资源。在其他线程因为等待硬盘或网络读写而时,程序能继续处理和运行。但是对线程运用不当,也会影响程序的性能。例2:正确使用Vector类Vector主要用来保存各种类型的对象(包括相同类型和不同类型的对象)。但是在一些情况下使用会给程序带来性能上的影响。这主要是由Vector类的两个特点所决定的。第一,Vector提供了线程的安全保护功能。即使Vector类中的许多方法同步。但是如果你已经确认你的应用程序是单线程,这些方法的同步就完全不必要了。第二,在Vector查找存储的各种对象时,常常要花很多的时间进行类型的匹配。而当这些对象都是同

12、一类型时,这些匹配就完全不必要了。因此,有必要设计一个单线程的,保存特定类型对象的类或集合来替代Vector 类.用来替换的程序如下(StringVector.java):public class StringVectorprivate String data;private int count;public StringVector()this(10); / default size is 10public StringVector(int initialSize)data = new StringinitialSize;public void add(String str)/ ignore

13、 null stringsif(str = null) return; ensureCapacity(count + 1);datacount+ = str;private void ensureCapacity(int minCapacity)int oldCapacity = data.length;if (minCapacity oldCapacity)String oldData = data;int newCapacity = oldCapacity * 2;data = new StringnewCapacity;System.arraycopy(oldData, 0, data,

14、 0, count);public void remove(String str)if(str = null) return; / ignore null str for(int i = 0; i / check for a matchif(datai.equals(str)System.arraycopy(data,i+1,data,i,count-1); / copy data/ allow previously valid array element be gcddata-count = null;return;public final String getStringAt(int in

15、dex)if(index else if(index count) return null; / index is # strings else return dataindex; / index is good 因此,代码:Vector Strings=new Vector();Strings.add(One);Strings.add(Two);String Second=(String)Strings.elementAt(1);可以用如下的代码替换:StringVector Strings=new StringVector();Strings.add(One);Strings.add(Tw

16、o);String Second=Strings.getStringAt(1);这样就可以通过优化线程来提高JAVA程序的性能。用于测试的程序如下(TestCollection.java):import java.util.Vector;public class TestCollectionpublic static void main(String args )TestCollection collect = new TestCollection();if(args.length = 0)System.out.println(Usage: java TestCollection vector

17、 | stringvector );System.exit(1);if(args0.equals(vector)Vector store = new Vector();long start = System.currentTimeMillis();for(int i = 0; i store.addElement(string);long finish = System.currentTimeMillis();System.out.println(finish-start);start = System.currentTimeMillis();for(int i = 0; i String r

18、esult = (String)store.elementAt(i);finish = System.currentTimeMillis();System.out.println(finish-start);else if(args0.equals(stringvector)StringVector store = new StringVector();long start = System.currentTimeMillis();for(int i = 0; i long finish = System.currentTimeMillis();System.out.println(finis

19、h-start);start = System.currentTimeMillis();for(int i = 0; i String result = store.getStringAt(i);finish = System.currentTimeMillis();System.out.println(finish-start);关于线程的操作,要注意如下几个方面:(1) 防止过多的同步如上所示,不必要的同步常常会造成程序性能的下降。因此,如果程序是单线程,则一定不要使用同步。(2) 同步方法而不要同步整个代码段对某个方法或函数进行同步比对整个代码段进行同步的性能要好。(3) 对每个对象使用

20、多”锁”的机制来增大并发。一般每个对象都只有一个”锁”,这就表明如果两个线程执行一个对象的两个不同的同步方法时,会发生”死锁”。即使这两个方法并不共享任何资源。为了避免这个问题,可以对一个对象实行”多锁”的机制。如下所示:class fooprivate static int var1;private static Object lock1=new Object();private static int var2;private static Object lock2=new Object();public static void increment1()synchronized(lock1)

21、var1+;public static void increment2()synchronized(lock2)var2+;4输入和输出(I/O)输入和输出包括很多方面,但涉及最多的是对硬盘,网络或数据库的读写操作。对于读写操作,又分为有缓存和没有缓存的;对于数据库的操作,又可以有多种类型的JDBC驱动器可以选择。但无论怎样,都会给程序的性能带来影响。因此,需要注意如下几点:(1) 使用输入输出缓冲尽可能的多使用缓存。但如果要经常对缓存进行刷新(flush),则建议不要使用缓存。(2) 输出流(Output Stream)和Unicode字符串当时用Output Stream和Unicode字

22、符串时,Write类的开销比较大。因为它要实现Unicode到字节(byte)的转换.因此,如果可能的话,在使用Write类之前就实现转换或用OutputStream类代替Writer类来使用。(3) 当需序列化时使用transient当序列化一个类或对象时,对于那些原子类型(atomic)或可以重建的原素要表识为transient类型。这样就不用每一次都进行序列化。如果这些序列化的对象要在网络上传输,这一小小的改变对性能会有很大的提高。(4) 使用高速缓存(Cache)对于那些经常要使用而又不大变化的对象或数据,可以把它存储在高速缓存中。这样就可以提高访问的速度。这一点对于从数据库中返回的结果集尤其重要。(5) 使用速度快的JDBC驱动器(Driver)JAVA对访问数据库提供了四种方法。这其中有两种是JDBC驱动器。一种是用JAVA外包的本地驱动器;另一种是完全的JAVA驱动器。具体要使用哪一种得根据JAVA布署的环境和应用程序本身来定。5.一些其他的经验和技巧(1) 使用局部变量。(2) 避免在同一个类中动过调用函数或方法(get或set)来设置或调用变量。(3) 避免在循环中生成同一个变量或调用同一个函数(参数变量也一样)。(4) 尽可能的使用static,final,private等关键字。(5) 当复制大量数据时,使用System.arraycopy()命令。

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!