欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺析Go語言如何在終端里實現(xiàn)倒計時

 更新時間:2025年03月06日 08:34:27   作者:apocelipes  
這篇文章主要為大家詳細介紹了Go語言中是如何在終端里實現(xiàn)倒計時的,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下

最近在更新系統(tǒng)的時候發(fā)現(xiàn)pacman的命令行界面變了,我有很久沒更新過設備上的Linux系統(tǒng)了,所以啥時候變的不好說。但這一變化成功勾起了我的好奇心。新版的更新進度界面如下:

新的更新進度界面能同時顯示多個進度條,而且并沒有依靠ncurses這個傳統(tǒng)的TUI庫。為啥我能斷定沒有用ncurses呢,因為用過這個庫的人都會發(fā)現(xiàn)程序在繪制界面的時候會用背景色清屏,且退出后終端的內容會恢復成運行程序前的樣子,而上述表現(xiàn)都不存在。

不借助專用的庫卻又能繪制出比較生動的效果,這難道不吸引人嗎?

所以帶著好奇心,我簡單探索了實現(xiàn)的原理,并且用相同的原理做了個新東西:

這是一個在終端中顯示倒計時的小玩具,原理和pacman的進度條是一樣的,我并沒有一比一去復現(xiàn)pacman的效果,那樣其實和對著范本寫作文一樣略顯無聊,所以我選擇活用知識做個新玩具。

好了,我們先來復習下單個終端命令行的進度條是怎么實現(xiàn)的。

單個進度條的原理其實很簡單,幾乎所有的終端和終端模擬器都支持一些特殊的控制字符,比如\n表示新加一個空白行并把光標移動到這個新行的最左側也就是開頭處;\r則是將光標移動到當前行的開頭處。

所以單個進度條的繪制過程一共只要兩步:

  • 根據(jù)進度計算出當前進度條的樣子,然后用打印函數(shù)輸出,注意不能輸出換行符\n;
  • 輸出\r讓光標回到行首,等待一段時間,重復步驟1,新的輸出內容會覆蓋掉老的。
  • 進度到了100%之后就可以輸出一個換行符\n結束進度條的打印了。

最關鍵的地方也只有一處,新的輸出內容的長度要大于或者等于老內容,否則老內容會殘留在終端里。

人眼的要求很低,所以你甚至可以不必做到每秒xx次刷新,只要在一秒或幾秒里更新幾次就能讓人覺得你的進度條動起來了。

所以一個最簡單的例子可以是這樣的:

package main
 
import (
	"bytes"
	"fmt"
	"time"
)
 
const width = 50
 
func main() {
	bar := bytes.Repeat([]byte{' '}, width)
	fmt.Println()
	for i := range 50 {
		bar[i] = '='
		fmt.Printf("[%s] % 3d%%\r", bar, (i+1)*2)
		time.Sleep(100 * time.Millisecond)
	}
    fmt.Println()
    fmt.Println("end")
}

這是效果:

\r有個缺點,它只能回溯當前行,而且這個“行”是以終端顯示為準的——即使你的輸出并沒有包含換行符但它的長度超過了終端顯示的寬度導致需要“折行”,那么新折行出來的那行在終端顯示中會被認為是一個新行,\r只會將光標放到這個新行的開頭。

其實我最開始想利用折行加\r字符實現(xiàn)多行進度條,但很快就發(fā)現(xiàn)這條路是走不通的。顯然pacman并沒有使用\r或者說它還利用了一些其他的東西。

看源代碼是最快的,而且簡單搜索一下“progressbar”很快就能找到答案。我就不賣關子了,pacman實現(xiàn)多行進度條效果是利用了ASNI轉義序列。

ANSI轉義序列(ANSI escape sequences)是一種帶內信號的轉義序列標準,用于控制視頻文本終端上的光標位置、顏色和其他選項。在文本中嵌入確定的字節(jié)序列,大部分以ESC轉義字符和"["字符開始,終端會把這些字節(jié)序列解釋為相應的指令,而不是普通的字符編碼。

簡單的說,轉義序列就像一些命令,可以控制光標和終端的各種行為。

具體格式是:轉義序列開始字符參數(shù)1;參數(shù)2;...;參數(shù)N命令。我們最常見的轉義序列是顏色控制,讓終端里的文字變成紅色:\033[0;31m。其中\033[是轉義序列的開始標志,0;31是命令m的兩個參數(shù),參數(shù)之間用空格分隔,最后一個參數(shù)緊貼著命令。

轉義序列的支持程度要看終端和終端模擬器,好消息是我們需要用到的轉義序列的被廣泛支持的,我們要用它們來在行與行之間移動光標并繪制內容。

轉義序列支持光標上下左右移動還支持直接清除整行的內容,這使得我們可以將終端當成一個畫布:每個字符的位置相當于畫布上的一個像素點(因此使用等寬字體效果顯示會更好),坐標原點是程序運行開始后光標所在的位置,根據(jù)這個原點可以簡單構建出一個平面坐標系,我們可以用一些特殊字符模擬點和線來繪制簡單的圖形。

我們要用的轉義序列是這些:

  • \033[nF,將光標向上移動n行
  • \033[nE,將光標向下移動n行
  • \033[nC,將光標向后(右)移動n個字符
  • \033[2K,清除光標所在行的整個內容(2以外的參數(shù)可以選擇只清除光標前/后的內容)
  • 轉義字符之間可以組合使用,比如\033[nE\033[mC表示光標先向下移動n行然后再向右移動m個字符。

現(xiàn)在你應該明白那個倒計時是怎么畫出來的了,核心技術點就是找到個合適的數(shù)字asciiart,然后根據(jù)每秒更新的內容在正確的位置上用上面的轉義序列像畫像素點一樣把數(shù)字和分隔符畫出來就行了。

說說其實一句話的事情,但做起來還是比較麻煩的,因為轉義序列用的都是相對坐標,稍微算錯一點相對位置顯示效果就整個完蛋了,我也是調試了三四回才做到正確繪制的:

func (ar *ASCIIArtCharRender) RenderContent(duration time.Duration) {
	if len(ar.chars) > 0 {
		ar.chars = ar.chars[:0]
	}
	ar.chars = char.ConvertToChars(duration, char.ASCIIArtChars, ar.chars)
	for i := 0; i < char.MaxASCIIArtCharHeight(); i++ {
		util.CursorEraseEntireLine()
		fmt.Print(ar.chars[0][i])
		fmt.Print(" ")
		fmt.Print(ar.chars[1][i])
		fmt.Print("  ")
		fmt.Print(char.ASCIIArtChars[char.ASCIIArtColonIdx][i])
		fmt.Print("  ")
		fmt.Print(ar.chars[2][i])
		fmt.Print(" ")
		fmt.Print(ar.chars[3][i])
		fmt.Print("  ")
		fmt.Print(char.ASCIIArtChars[char.ASCIIArtColonIdx][i])
		fmt.Print("  ")
		fmt.Print(ar.chars[4][i])
		fmt.Print(" ")
		fmt.Print(ar.chars[5][i])
		fmt.Print("\n")
	}
}
 
func (ar *ASCIIArtCharRender) RenderFlashing() {
	util.CursorDownForward(1, 3+len(ar.chars[0][0])+1+len(ar.chars[1][0]))
	fmt.Print(" ")
	util.CursorForward(3 + len(ar.chars[2][0]) + 1 + len(ar.chars[3][0]) + 3)
	fmt.Print(" ")
	util.CursorDownForward(1, 2+len(ar.chars[0][0])+1+len(ar.chars[1][0]))
	fmt.Print("   ")
	util.CursorForward(2 + len(ar.chars[2][0]) + 1 + len(ar.chars[3][0]) + 2)
	fmt.Print("   ")
 
	util.CursorDownForward(2, 3+len(ar.chars[0][0])+1+len(ar.chars[1][0]))
	fmt.Print(" ")
	util.CursorForward(3 + len(ar.chars[2][0]) + 1 + len(ar.chars[3][0]) + 3)
	fmt.Print(" ")
	util.CursorDownForward(1, 2+len(ar.chars[0][0])+1+len(ar.chars[1][0]))
	fmt.Print("   ")
	util.CursorForward(2 + len(ar.chars[2][0]) + 1 + len(ar.chars[3][0]) + 2)
	fmt.Print("   ")
	// move to bottom
	util.CursorDown(1)
}

第一個函數(shù)是繪制時間用的數(shù)字的,為了簡單我已經(jīng)提前把數(shù)字的asciiart保存進了二維數(shù)組并且做到了等高,這樣畫的時候只要知道需要什么數(shù)字就行,剩下的就是逐行輸出“像素點”。

第二個函數(shù)是用來繪制電子時鐘數(shù)字分隔符的閃爍效果的,這個看上去就更亂了,因為需要在終端畫布上大范圍移動。

所以會者不難,純體力活。

完整的代碼可以在這找到:https://github.com/apocelipes/ascii-count-down,歡迎各位大佬的改進或者功能增強。

總結

TUI還是挺有意思的,好玩能學到東西而且很能消磨無聊的時間。

另外我覺得在之間看源碼對答案之前,可以先自己思考一下并動手做做試驗比如像我那樣最先異想天開用折行去實現(xiàn)多行進度條。這樣雖然浪費了點時間,但可以加深自己對新知識的理解和記憶。

到此這篇關于淺析Go語言如何在終端里實現(xiàn)倒計時的文章就介紹到這了,更多相關Go終端實現(xiàn)倒計時內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • go語言中的defer關鍵字

    go語言中的defer關鍵字

    這篇文章介紹了go語言中的defer關鍵字,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-07-07
  • Golang基于內存的鍵值存儲緩存庫go-cache

    Golang基于內存的鍵值存儲緩存庫go-cache

    go-cache是一個內存中的key:value store/cache庫,適用于單機應用程序,本文主要介紹了Golang基于內存的鍵值存儲緩存庫go-cache,具有一定的參考價值,感興趣的可以了解一下
    2025-03-03
  • Go語言sync.Cond基本使用及原理示例詳解

    Go語言sync.Cond基本使用及原理示例詳解

    這篇文章主要為大家介紹了Go語言sync.Cond基本使用及原理示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-03-03
  • 源碼剖析Golang中singleflight的應用

    源碼剖析Golang中singleflight的應用

    這篇文章主要為大家詳細介紹了如何利用singleflight來避免緩存擊穿,并剖析singleflight包的源碼實現(xiàn)和工作原理,感興趣的可以了解下
    2024-03-03
  • golang中包無法引入問題解決

    golang中包無法引入問題解決

    本文主要介紹了golang中包無法引入問題解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-03-03
  • Go 值傳遞與引用傳遞的方法

    Go 值傳遞與引用傳遞的方法

    這篇文章主要介紹了Go 值傳遞與引用傳遞的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-03-03
  • 詳解如何使用Go語言進行文件監(jiān)控和通知

    詳解如何使用Go語言進行文件監(jiān)控和通知

    在Go語言中,文件監(jiān)控通常涉及到文件系統(tǒng)事件的監(jiān)聽,文件或目錄的狀態(tài)發(fā)生變化(如創(chuàng)建、刪除、修改等)時,你的程序需要得到通知,所以本文給大家介紹了如何使用Go語言進行文件監(jiān)控和通知,需要的朋友可以參考下
    2024-06-06
  • Golang使用gofumpt進行代碼格式化

    Golang使用gofumpt進行代碼格式化

    這篇文章主要為大家詳細介紹了Golang如何使用gofumpt進行代碼格式化,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2024-02-02
  • Go語言中的map擴容機制

    Go語言中的map擴容機制

    Go語言中的map是一種高效的數(shù)據(jù)結構,其擴容機制確保了在大數(shù)據(jù)量情況下的性能,本文介紹了包括擴容觸發(fā)條件、擴容過程和漸進式擴容,感興趣的可以了解一下
    2024-12-12
  • 為什么不建議在go項目中使用init()

    為什么不建議在go項目中使用init()

    這篇文章主要介紹了為什么不建議在go項目中使用init(),本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04

最新評論