Java項目計算程式執行時間方法

来源:https://www.cnblogs.com/damudaran/p/18022411
-Advertisement-
Play Games

一、總結 1.1、使用System.currentTimeMillis();計算程式執行毫秒數 // 開始時間1 long startTime1 = System.currentTimeMillis(); Thread.sleep(100); // 結束時間1 long endTime1 = Sys ...


一、總結

1.1、使用System.currentTimeMillis();計算程式執行毫秒數

		// 開始時間1
		long startTime1 = System.currentTimeMillis();
		Thread.sleep(100);
		// 結束時間1
		long endTime1 = System.currentTimeMillis();

		// 開始時間2
		long startTime2 = System.currentTimeMillis();
		Thread.sleep(200);
		// 結束時間2
		long endTime2 = System.currentTimeMillis();
		System.out.println("邏輯1執行時間:"+ (endTime1 - startTime1));
		System.out.println("邏輯2執行時間:"+ (endTime2 -startTime2));

1.2、使用org.springframework.util包下的一個工具類StopWatch計算執行時間

		StopWatch testTask = new StopWatch("TestTask");
		// 記錄開始時間點
		testTask.start("task1");
		Thread.sleep(100);
		// 記錄結束時間點
		testTask.stop();
		// 記錄開始時間點
		testTask.start("task2");
		Thread.sleep(200);
		// 記錄結束時間點
		testTask.stop();
		// 輸出執行時間
		System.out.println("==任務執行時間==");
		System.out.println(testTask.prettyPrint());
		System.out.println("執行任務的毫秒數:"+testTask.getTotalTimeMillis());

1.3兩個案例的完整代碼、執行結果

package time.stopwatch;

import org.springframework.util.StopWatch;

public class StopWatchTest {
		public static void main(String[] args) throws InterruptedException {
		// 計算執行時間
		calculateExecuteTime1();
		// 計算執行時間
		calculateExecuteTime2();
	}

	public static void calculateExecuteTime1() throws InterruptedException {
		// 開始時間1
		long startTime1 = System.currentTimeMillis();
		Thread.sleep(100);
		// 結束時間1
		long endTime1 = System.currentTimeMillis();

		// 開始時間2
		long startTime2 = System.currentTimeMillis();
		Thread.sleep(200);
		// 結束時間2
		long endTime2 = System.currentTimeMillis();
		System.out.println("邏輯1執行時間:"+ (endTime1 - startTime1));
		System.out.println("邏輯2執行時間:"+ (endTime2 -startTime2));
	}

	public static void calculateExecuteTime2() throws InterruptedException {
		StopWatch testTask = new StopWatch("TestTask");
		// 記錄開始時間點
		testTask.start("task1");
		Thread.sleep(100);
		// 記錄結束時間點
		testTask.stop();
		// 記錄開始時間點
		testTask.start("task2");
		Thread.sleep(200);
		// 記錄結束時間點
		testTask.stop();
		// 輸出執行時間
		System.out.println("==任務執行時間==");
		System.out.println(testTask.prettyPrint());
		System.out.println("執行任務的毫秒數:"+testTask.getTotalTimeMillis());
	}
}

執行結果:

邏輯1執行時間:109
邏輯2執行時間:203
==任務執行時間==
StopWatch 'TestTask': running time = 319076700 ns
---------------------------------------------
ns         %     Task name
---------------------------------------------
114748000  036%  task1
204328700  064%  task2

執行任務的毫秒數:319

1.4 StopWatch優缺點:

優點:
1、spring自帶工具類,可直接使用
2、代碼實現簡單,使用更簡單
3、統一歸納,展示每項任務耗時與占用總時間的百分比,展示結果直觀
4、性能消耗相對較小,並且最大程度的保證了start與stop之間的時間記錄的準確性
5、可在start時直接指定任務名字,從而更加直觀的顯示記錄結果
缺點:
1、一個StopWatch實例一次只能開啟一個task,不能同時start多個task,並且在該task未stop之前不能start一個新的task,必須在該task stop之後才能開啟新的task,若要一次開啟多個,需要new不同的StopWatch實例
2、代碼侵入式使用,需要改動多處代碼

1.5、spring中StopWatch源碼實現如下:

import java.text.NumberFormat;
import java.util.LinkedList;
import java.util.List;

public class StopWatch {
	private final String id;
	private boolean keepTaskList = true;
	private final List<TaskInfo> taskList = new LinkedList();
	private long startTimeMillis;
	private boolean running;
	private String currentTaskName;
	private StopWatch.TaskInfo lastTaskInfo;
	private int taskCount;
	private long totalTimeMillis;

	public StopWatch() {
		this.id = "";
	}

	public StopWatch(String id) {
		this.id = id;
	}

	public void setKeepTaskList(boolean keepTaskList) {
		this.keepTaskList = keepTaskList;
	}

	public void start() throws IllegalStateException {
		this.start("");
	}

	public void start(String taskName) throws IllegalStateException {
		if (this.running) {
			throw new IllegalStateException("Can't start StopWatch: it's already running");
		} else {
			this.startTimeMillis = System.currentTimeMillis();
			this.running = true;
			this.currentTaskName = taskName;
		}
	}

	public void stop() throws IllegalStateException {
		if (!this.running) {
			throw new IllegalStateException("Can't stop StopWatch: it's not running");
		} else {
			long lastTime = System.currentTimeMillis() - this.startTimeMillis;
			this.totalTimeMillis += lastTime;
			this.lastTaskInfo = new StopWatch.TaskInfo(this.currentTaskName, lastTime);
			if (this.keepTaskList) {
				this.taskList.add(this.lastTaskInfo);
			}

			++this.taskCount;
			this.running = false;
			this.currentTaskName = null;
		}
	}

	public boolean isRunning() {
		return this.running;
	}

	public long getLastTaskTimeMillis() throws IllegalStateException {
		if (this.lastTaskInfo == null) {
			throw new IllegalStateException("No tasks run: can't get last task interval");
		} else {
			return this.lastTaskInfo.getTimeMillis();
		}
	}

	public String getLastTaskName() throws IllegalStateException {
		if (this.lastTaskInfo == null) {
			throw new IllegalStateException("No tasks run: can't get last task name");
		} else {
			return this.lastTaskInfo.getTaskName();
		}
	}

	public StopWatch.TaskInfo getLastTaskInfo() throws IllegalStateException {
		if (this.lastTaskInfo == null) {
			throw new IllegalStateException("No tasks run: can't get last task info");
		} else {
			return this.lastTaskInfo;
		}
	}

	public long getTotalTimeMillis() {
		return this.totalTimeMillis;
	}

	public double getTotalTimeSeconds() {
		return (double) this.totalTimeMillis / 1000.0D;
	}

	public int getTaskCount() {
		return this.taskCount;
	}

	public StopWatch.TaskInfo[] getTaskInfo() {
		if (!this.keepTaskList) {
			throw new UnsupportedOperationException("Task info is not being kept!");
		} else {
			return (StopWatch.TaskInfo[]) this.taskList.toArray(new StopWatch.TaskInfo[this.taskList.size()]);
		}
	}

	public String shortSummary() {
		return "StopWatch '" + this.id + "': running time (millis) = " + this.getTotalTimeMillis();
	}

	public String prettyPrint() {
		StringBuilder sb = new StringBuilder(this.shortSummary());
		sb.append('\n');
		if (!this.keepTaskList) {
			sb.append("No task info kept");
		} else {
			sb.append("-----------------------------------------\n");
			sb.append("ms     %     Task name\n");
			sb.append("-----------------------------------------\n");
			NumberFormat nf = NumberFormat.getNumberInstance();
			nf.setMinimumIntegerDigits(5);
			nf.setGroupingUsed(false);
			NumberFormat pf = NumberFormat.getPercentInstance();
			pf.setMinimumIntegerDigits(3);
			pf.setGroupingUsed(false);
			StopWatch.TaskInfo[] var7;
			int var6 = (var7 = this.getTaskInfo()).length;

			for (int var5 = 0; var5 < var6; ++var5) {
				StopWatch.TaskInfo task = var7[var5];
				sb.append(nf.format(task.getTimeMillis())).append("  ");
				sb.append(pf.format(task.getTimeSeconds() / this.getTotalTimeSeconds())).append("  ");
				sb.append(task.getTaskName()).append("\n");
			}
		}

		return sb.toString();
	}

	@Override
	public String toString() {
		StringBuilder sb = new StringBuilder(this.shortSummary());
		if (this.keepTaskList) {
			StopWatch.TaskInfo[] var5;
			int var4 = (var5 = this.getTaskInfo()).length;

			for (int var3 = 0; var3 < var4; ++var3) {
				StopWatch.TaskInfo task = var5[var3];
				sb.append("; [").append(task.getTaskName()).append("] took ").append(task.getTimeMillis());
				long percent = Math.round(100.0D * task.getTimeSeconds() / this.getTotalTimeSeconds());
				sb.append(" = ").append(percent).append("%");
			}
		} else {
			sb.append("; no task info kept");
		}

		return sb.toString();
	}

	public static final class TaskInfo {
		private final String taskName;
		private final long timeMillis;

		TaskInfo(String taskName, long timeMillis) {
			this.taskName = taskName;
			this.timeMillis = timeMillis;
		}

		public String getTaskName() {
			return this.taskName;
		}

		public long getTimeMillis() {
			return this.timeMillis;
		}

		public double getTimeSeconds() {
			return (double) this.timeMillis / 1000.0D;
		}
	}

}

原文摘錄至:https://blog.csdn.net/gxs1688/article/details/87185030
致敬原作者:一個不二,侵刪。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 痞子衡嵌入式半月刊: 第 92 期 這裡分享嵌入式領域有用有趣的項目/工具以及一些熱點新聞,農曆年分二十四節氣,希望在每個交節之日準時發佈一期。 本期刊是開源項目(GitHub: JayHeng/pzh-mcu-bi-weekly),歡迎提交 issue,投稿或推薦你知道的嵌入式那些事兒。 上期回顧 ...
  • 索引:為了提高數據查詢的效率,就像書的目錄一樣。 索引的常見模型 哈希表 圖中,User2 和 User4 根據身份證號算出來的值都是 N,後面還跟了一個鏈表。假設,這時候你要查 ID_card_n2 對應的名字是什麼,處理步驟就是:首先,將 ID_card_n2 通過哈希函數算出 N;然後,按順序 ...
  • 03 事務隔離 事務:保證一組資料庫操作,要麼全部成功,要麼全部失敗。在 MySQL 中,事務支持是在引擎層實現的。 事務ACID(Atomicity、Consistency、Isolation、Durability,即原子性、一致性、隔離性、持久性)。 建議你儘量不要使用長事務。**** 讀未提交 ...
  • 引言 在實際業務開發中,隨著業務的變化,數據的複雜性和多樣性不斷增加。傳統的關係型資料庫模型在這種情況下會顯得受限,因為它們需要預先定義嚴格的數據模式,並且通常只能存儲具有相同結構的數據。而面對非結構化或半結構化數據的存儲和處理需求,選擇使用非關係型資料庫或者創建子表存儲這些變化的結構可能會變得複雜 ...
  • 在項目開發中需要添加webview,載入內置的html文件,代碼寫完後ios運行沒有問題,運行安卓時報錯,錯誤提示如下: FAILURE: Build failed with an exception. * What went wrong: Execution failed for task ':a ...
  • 玩轉 CMS2 上篇研究了樣式、請求、evn、mock,感覺對效率的提升沒有太明顯作用。 比如某個工作需要2天,現在1天可以幹完,這就是很大的提升。 提高效率的方法有代碼復用、模塊化、低代碼工具。 目前可以考慮從代碼復用方面下手,即使最低級的代碼複製也可以。 要快速提高效率,需要對本地項目中的一些關 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 ref 和 reactive 是 Vue3 中實現響應式數據的核心 API。ref 用於包裝基本數據類型,而 reactive 用於處理對象和數組。儘管 reactive 似乎更適合處理對象,但 Vue3 官方文檔更推薦使用 ref。 我 ...
  • 簡介 簡單工廠模式又稱為靜態工廠模式,屬於創建型模式,但不屬於GOF23設計模式。由一個工廠對象決定創建出哪一種產品類的實例。簡單工廠模式的實質是由一個工廠類根據傳入的參數,動態決定應該創建哪一個產品類。 簡單工廠適用場景:工廠類負責創建的對象比較少;客戶端只需要知道傳入工廠類的參數,對於如何創建對 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...