java 精确到微妙_如何在Java中以微秒精度测量时间?
我在Internet上看到应该使用System.nanoTime(),但这对我不起作用-它为我提供了毫秒级的时间。我只需要函数执行前后的微秒,就可以知道需要多长时间。我正在使用Windows XP。
基本上,我有这段代码,例如,在Java链表中进行了1百万个到1000万个插入。问题是我无法正确测量精度;有时,花费较少的时间将所有内容插入较小的列表中。
这是一个例子:
class test
{
public static void main(String args[])
{
for(int k=1000000; k<=10000000; k+=1000000)
{
System.out.println(k);
LinkedList aux = new LinkedList();
//need something here to see the start time
for(int i=0; i
aux.addFirst(10000);
//need something here to see the end time
//print here the difference between both times
}
}
}
我做了很多次-每个k有20次外部循环-但是结果不好。有时,进行一千万次插入所需的时间少于一百万次,因为使用当前使用的时间我无法获得正确的测量时间(System.nanoTime())
编辑2:是的,我正在使用Sun JVM。
编辑3:我可能在代码中做错了什么,我会看看更改是否符合我的要求。
编辑4:我的错误,似乎System.nanoTime()有效。唷。
什么版本的Java?在具有JDK1.6.0_06的XP机器上,nanoTime()的精度一直到最后一位小数。我假设您的版本返回的值始终以三个零结尾。
Sun JVM是不够的,它是哪个VERSION? (您的回答是正确的,我没有提出正确的问题。我假设每个人都使用Sun JVM)
cmd中的Java -version给我:Java版本" 1.7.0-ea"和另外2行。无论哪种方式,这都是我的代码中的错误。抱歉,即使搜索了一段时间我也找不到错误。
勇敢的灵魂,1.7仍然很早
自OT开始添加评论。您应该阅读如何编写JVM的微基准测试,例如concentric.net/~ttwang/tech/microbench.htm。此外,您可能希望将10000个整数放在循环外。
要求微秒精度的基准不会非常准确。如果网络上的另一台计算机恰巧对您的设备执行ping操作,或者移动鼠标等,您可以在微秒级的时间测量中增加噪声,而CPU负担最少。
这不能回答您的特定问题,但是使用它可能使它变得不必要:code.google.com/p/caliper
我的猜测是,由于System.nanoTime()使用了"最精确的可用系统计时器",该计时器显然在系统上只有毫秒精度,因此您无法得到更好的结果。
对我来说,尚不清楚您要进行基准测试的确切信息,但总的来说,任何需要花费如此短时间才能运行且精度低于50 ms的测试都将很容易受到其他干扰。
我通常会尝试使基准测试至少运行10秒钟。我目前正在编写的框架将猜测要运行多少次迭代,因此将花费30秒。这意味着您不会因为某些其他进程窃取了CPU几毫秒而获得根本不同的结果。
与尝试以更细粒度的精度进行测量相比,长时间运行几乎总是一种更好的方法。
System.nanoTime()在CPU中使用一个计数器,在Windows XP和Linux中通常精确到大约1微秒。
注意:Windows XP在多cpu机器上通常精度较差,因为它不能补偿具有不同计数器的不同CPU。 Linux确实如此。
注意2:它会相对于System.currentTimeMillis()漂移,因为它基于CPU时钟的准确性(一段时间内不需要那么精确),而不是您拥有的时钟来获取时间(每天漂移较少,但粒度较小)
在基准测试中,您基本上正在测试创建新对象的速度。毫不奇怪,根据您的GC设置以及最近执行一次GC的结果,您的结果将发生巨大变化。
尝试使用以下选项运行测试,您将看到非常不同的结果。
-verbosegc -XX:NewSize=128m -mx256m
使用java.time
FYI,Java 9和更高版本具有Clock的全新实现,可以以高达纳秒的分辨率捕获当前时刻。
Instant类表示UTC时间轴上的时刻,其分辨率为纳秒(最多十进制的九(9)位数字)。
调用Instant.now捕获当前时刻。
在Java 9和更高版本中,当前时刻的分辨率最高为纳秒。
在Java 8中,当前时刻最多只能捕获到毫秒级的分辨率(您确实可以保留纳秒级的值,但只能捕获到毫秒级的当前时刻)。
Instant Instant = Instant.now();
用Duration类表示未附加到时间轴的时间跨度。保留时间量(以秒和纳秒为单位)。
Duration d = Duration.between( instantThen , Instant.now() );
明确地说,问题中要求的微秒分辨率介于毫秒和纳秒之间。十进制小数位数:毫秒为3(0.123),微米为6(0.123456),纳米为9(0.123456789)。
警告
Java依赖于计算机的硬件时钟。正如其他人警告的那样,硬件几乎可以肯定会以比纳秒低得多的精度和更低的分辨率捕获时间。
如此细粒度的基准测试存在很多问题,通常不建议这样做。
并提防过早的优化。
有人建议在JEP 230:Microbenchmark Suite中向Java平台添加微基准测试功能。基于Java Microbenchmark Harness(JMH)。
关于java.time
java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧旧式日期时间类,例如java.util.Date,Calendar和SimpleDateFormat。
现在处于维护模式的Joda-Time项目建议迁移到java.time类。
要了解更多信息,请参见Oracle教程。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310。
在哪里获取java.time类?
Java SE 8和SE 9及更高版本
内置。
标准Java API的一部分,具有捆绑的实现。
Java 9添加了一些次要功能和修复。
Java SE 6和SE 7
java.time的许多功能在ThreeTen-Backport中都被反向移植到Java 6和7。
Android的
ThreeTenABP项目专门针对Android改编了ThreeTen-Backport(如上所述)。
请参阅如何使用ThreeTenABP…。
ThreeTen-Extra项目使用其他类扩展了java.time。该项目为将来可能在java.time中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如Interval,YearWeek,YearQuarter等。
如果想要可靠的结果,请使用分析器。我建议使用VisualVM,它易于安装,并且从1.6.0_07版本开始与JDK捆绑在一起。
这是一个易于使用的可视化工具,集成了多个命令行JDK工具和轻量级分析功能。
您必须重复测试数千次。有很多事情会影响您的测量,例如垃圾收集,I / O,交换/交换,就绪队列线程的大小等。
这很奇怪。 System.nanoTime()应该可以工作。您正在使用Sun JVM吗?
您是否可以重复操作1000次,然后将时间除以1000,以找出需要了解的内容?
是的,System.nanoTime的准确性和精确度通常比System.currentTimeMillis更好,但是并不能保证:在最坏的情况下,它也同样会变得很糟糕。
ThreadMXBean.getCurrentThreadCpuTime倾向于产生更短的时间,但是其分辨率尚不清楚,并且还具有其他缺点(您是否真的需要CPU时间?平台依赖的语义,平台是否支持?)。
用这三种技术测量时间也有一些成本,即需要时间本身,这会使测量值失真。成本与平台高度相关,但通常成本(System.currentTimeMillis)<
关于一般的微基准测试,请参阅
如何在Java中编写正确的微基准测试?
使用Java创建快速/可靠的基准测试?
底层操作系统可能不提供具有纳秒级精度的计时器。
还有一个较旧的帖子。
对于我们最近的分析,我发现ThreadMXBean.getCurrentThreadCpuTime()和选项-XX:+UseLinuxPosixThreadCPUClocks满足了我们的需求。
有关更多详细信息,请参见http://bugs.java.com/view_bug.do?bug_id=6888526
我最终使用的"快速而肮脏的"解决方案:
TimeUnit.NANOSECONDS.toMicros(System.nanoTime());
更新:
我最初使用System.nanoTime,但是后来我发现它仅应用于经过的时间,最终我更改了代码以使其可以使用毫秒或在某些地方使用:
TimeUnit.MILLISECONDS.toMicros(System.currentTimeMillis());
但这只会在值的末尾添加零(微秒=毫秒* 1000)
如果有人想到nanoTime,则将此答案留在此处作为"警告信号" :)
这种依赖短时间间隔的基准会给您带来不可靠的结果。由于I / O,交换,进程切换,高速缓存,垃圾回收等外部因素,您总是会得到不同的结果。此外,JVM优化了您的调用,因此最先进行测量的事情可能会比以后的调用慢。 JVM越来越多地开始优化您执行的命令。
另外,类似System.nanoTime()的方法取决于基础系统的计时器。他们可能(而且很可能会)没有粒度来测量该准确性。引用API:
This method provides nanosecond
precision, but not necessarily
nanosecond accuracy. No guarantees are
made about how frequently values
change.
要真正进行高精度测量,您需要使用保证精度的外部计时硬件。
为了使基准更加稳定,您需要执行多次,并且测量的时间间隔要大于毫秒。
java 精确到微妙_如何在Java中以微秒精度测量时间?相关推荐
- java整数的因式分解_如何在Java中找到整数的质数-因式分解
java整数的因式分解 编程课程中的常见家庭作业/任务之一是关于Prime Factorization. 要求您编写一个程序以找到给定整数的素因子 . 一个数字的素数因子是将精确地除以给定数字的所有素 ...
- java中转json字符串_如何在Java中转义JSON字符串-Eclipse IDE技巧
java中转json字符串 在Java应用程序中工作或进行JSON解析时,通常很常见的做法是从某些资源(例如RESTful Web服务)中复制粘贴JSON字符串,然后使用Jackson库解析JSON. ...
- java swing 列表框_如何在Swing中使用列表框?
下面的示例展示了如何在Java Swing应用程序中使用标准列表框. 使用以下API - JList - 创建标准列表. JList.setSelectedIndex(index); - 选择项目. ...
- Java poi 列移动_如何在java xssf中移動列poi
如何將現有列數據和格式化到Apache POI中的下一列並將下一列移到右側.如何在java xssf中移動列poi 我試過了. 讓說我的代碼是這樣... XSSFCell oldCell = work ...
- java 方法当参数_如何在Java中将函数作为参数传递?
如何在Java中将函数作为参数传递? 这个问题在这里已有答案: Java Pass方法作为参数 14个答案 是否可以将方法作为参 ...
- java adt怎么使用_如何在eclipse中添加android ADT ADT插件的安装图文教程
对于程序开发的学者来说,eclipse并不陌生,它为我们提供了一个非常广阔的平台来开发程序.同样我们也可以用它来开发android程序.但是在eclipse中并不能直接开发android程序,需要我们 ...
- java 千位分隔符_如何在Java中设置千位分隔符?
问题 如何在Java中设置千位分隔符?我有BigDecimal的String表示,我想设置千位分隔符并返回String. #1 热门回答(180 赞) 你可以使用格式功能","; ...
- java 判断是否换行_如何在java中检测换行符
是的,我已经阅读了一些关于换行符的其他帖子,但他们没有帮助我.如何在java中检测换行符 我的Java程序应该读取一个.PHP文件,并将源代码分成逐行格式.问题是我似乎无法区分一个换行符和另一个换行符 ...
- java如何实例化集合_如何在java中实例化一个Queue对象?
Queue是一个接口,这意味着你不能直接构造一个Queue . 最好的select是构造一个已经实现Queue接口的类,如下所示: AbstractQueue , ArrayBlockingQueue ...
最新文章
- hdu3635 Dragon Balls(带权并查集)
- Druid:数据库连接池实现技术
- 假如生活欺骗了你!——Leo网上答疑(14)
- java 自动转 golang_基于dubbo-go的golang与java通信解决方案实践——伍:golang代码的生成...
- 2017.10.10 Perm 排列计数 失败总结
- html5的常用标签,HTML5常用标签
- 通向财务自由之路02_成功的决定因素:你
- HackPorts – Mac OS X 渗透测试框架与工具
- 与 Netcraft 携手为 GlobalSign 的客户提供先进的保护措施以防止网站遭受恶意入侵和钓鱼攻击...
- lammps教程:restart重启计算命令用法详解
- C++ 输入输出加速挂(ACM竞赛常用)
- 超链接 qq群一键添加
- 微信网页版打不开怎么办?这里有官方解决办法!
- 代码实现堆溢出、栈溢出、永久代溢出、直接内存溢出
- 可决系数、相关系数、均方误差
- 跟着Vam一起学习Typescript(第一期)
- 毕业设计 - 题目:基于机器视觉的图像矫正 (以车牌识别为例) - 图像畸变校正
- 科技云报道:元宇宙,会是云计算的下一个新战场吗?
- 时间序列分析教程(四):AR与MA模型详细分析(公式推导,慎入)
- 深度学习数据标注_Lableme及标注文件的使用(以YOLO v3为例)
热门文章
- CET-6--2018.12--1
- 摩托罗拉ap6521恢复出厂_moto AP6521 配置命令
- php artisan command,artisan command 小技巧
- 2-2 第17次课 高项之人力资源管理
- python实现懒人听书
- 自己用的一些觉得不错的软件
- Vue3.x 深度选择器(样式穿刺)>>> 和 /deep/ 和 ::v-deep 被弃用desprecated
- 一个tomcat服务单独控制多个项目启停
- PCtoLCD2002 生成字模转换为字库 .h文件
- wcp知识库系统的安装