Monday, December 27, 2021
Saturday, December 25, 2021
[用Golang寫小遊戲教學] 第一章 Golang開發環境設定
[前言]
工欲善其事,必先利其器。
Golang是個跨作業系統(像是: Windows、Linux、Mac OS等等)與支援多CPU架構(x86、ARM64等等)的程式語言,換句話說,Golang的程式可以透過go command直接在眾多的作業系統與幾種不同的CPU架構下執行。
進一步來說,也可以把Golang程式編譯/或交叉編譯 到不同的作業系統與不同的CPU架構下成為執行檔案,即可直接執行。其內涵不了解沒有關係,這不防礙接下來安裝的步驟。
[安裝Golang]
我這邊只先舉Windows 10的安裝範例,對於Linux或是Mac的作業系統,網路上應該有很多安裝教學可以參考ㄡ~
首先請用點擊下列Golang官方下載網頁:
https://go.dev/dl/
[用Golang寫小遊戲教學] 第二章 基本型別、變數宣告與常數宣告
[前言]
了解"基本型別"與"變數宣告"是開始學習程式語言的第一步。我們其實不需要一次把所有的觀念與知識都學到位,這對於初學者或是小朋友來說太過於複雜,這裡就先介紹比較常見的部分。
這篇的教學還不需要用到本機上安裝的Go來執行,我們會用 The Go Playground (網頁版)來練習。
[變數宣告]
在電腦的世界中,每個程式都需要在記憶體中儲存一些資料,資料被儲存在特定的記憶體位置,變數只是為儲存資料的記憶體位置所取的一個方便的名稱。除了名稱外,每個變數還有關聯的型別。型別用來定義存放的資料是屬於何種型態,像是整數或是字串等等,在後面的段落將說明何為資料型別。
我們先看一個例子: var myScore int = 100,並參考下圖
Thursday, December 23, 2021
[用Golang寫小遊戲教學] 第零章 大綱
[前言]
曾經有位朋友與我討論軟體工程師如何寫好程式(或是開發出好的軟體),我依稀記得是這樣回答:
"木匠製作椅子為例子來說,他(木匠)本身需要具備身為木匠的能力,例如: 鋸木、裁切、刨木、鑽孔、拋光、上漆等等功夫,接下來透過不斷的實作成品的鍛鍊,可以學習到製作越來越複雜的木工成品,當累積一段時間之後,會有自己的風格與設計架構對於擅長的木工項目。
同理可推,在軟體工程師的角度來說,是須要懂得電腦基本概念、了解作業系統的運行、與熟悉某種程式語言,這是基本功。在不斷的實作過程中累積出自己的實力,才有可能寫出好的程式。"
這樣的回答可能沒有完全說服那位朋友,但是基本功對於一般人來說,會是個門檻與障礙。
對於像是不具備相關背景的大朋友們,可能會有心理障礙而不敢學習寫程式,擔心學習門檻很高或是學不會。對於小朋友們來說,過早就進入複雜的背景知識或是學習太多程式語言的細節,不只無法理解其內容,並且會快速遇到瓶頸而備受打擊,或是失去學習程式的動力與興趣。
[教學內容列表]
Friday, December 10, 2021
[用Golang寫小遊戲教學] 第三章 初探Ebiten (A dead simple 2D game library for Go)
[前言]
Ebiten 官方網站 (The official website)
- Source code ( Github )
- API reference
- Tour (WIP)
- Documents
- Examples ( A lot of examples )
- Blog
用Golang寫小遊戲教學這一系列的內容,將會以一個基於Golang語言的 2D Game Library: Ebiten,對於製作小遊戲來說,感覺簡單又易於使用 。Ebiten 套件原作者是日本人,少量遊戲作品可在 wiki 中找到: https://github.com/hajimehoshi/ebiten/wiki/Works
很多使用範例可以學習:
Wednesday, December 1, 2021
[GCC] Using GCC to create static and shared library ( .so / .a )
[轉貼&修改] https://blog.xuite.net/csiewap/cc/23626229-Using+GCC+to+create+static+and+shared+library+.so
Library可分成三種,static、shared與dynamically loaded。
1. Static libraries
Static 程式庫用於靜態連結,簡單講是把一堆object檔用ar(archiver)
包裝集合起來,檔名以 `.a' 結尾。優點是執行效能通常會比後兩者快,
而且因為是靜態連結,所以不易發生執行時找不到library或版本錯置而
無法執行的問題。缺點則是檔案較大,維護度較低;例如library如果發
現bug需要更新,那麼就必須重新連結執行檔。
1.1 編譯
編譯方式很簡單,先例用 `-c' 編出 object 檔,再用 ar 包起來即可。
hello.c
#include
void hello(){ printf("Hello "); }
world.c
#include
void world(){ printf("world."); }
mylib.h
void hello();
void world();
$ gcc -c hello.c world.c /* 編出 hello.o 與 world.o */
$ ar rcs libmylib.a hello.o world.o /* 包成 limylib.a */
這樣就可以建出一個檔名為 libmylib.a 的檔。輸出的檔名其實沒有硬性規定,
但如果想要配合 gcc 的 '-l' 參數來連結,一定要以 `lib' 開頭,中間是你要
的library名稱,然後緊接著 `.a' 結尾。
1.2 使用
main.c
#include "mylib.h"
int main() {
hello();
world();
}
使用上就像與一般的 object 檔連結沒有差別。
$ gcc main.c libmylib.a
也可以配合 gcc 的 `-l' 參數使用
$ gcc main.c -L. -lmylib
`-Ldir' 參數用來指定要搜尋程式庫的目錄,`.' 表示搜尋現在所在的目錄。
通常預設會搜 /usr/lib 或 /lib 等目錄。
`-llibrary' 參數用來指定要連結的程式庫 ,'mylib' 表示要與mylib進行連結
,他會搜尋library名稱前加`lib'後接`.a'的檔案來連結。
$ ./a.out
Hello world.
此時可以用 nm <exe filename> 來顯示symbols (functions, etc.)
(To see which symbols come from static libraries requires running nm against those libraries to get a list of the symbols (functions, etc.) in them, then comparing them to what your list of symbols from nm <exefilename>.)
2. Shared libraries
Shared library 會在程式執行起始時才被自動載入。因為程式庫與執行檔
是分離的,所以維護彈性較好。有兩點要注意,shared library是在程式起始
時就要被載入,而不是執行中用到才載入,而且在連結階段需要有該程式庫
才能進行連結。
首先有一些名詞要弄懂,soname、real name與linker name。
soname 用來表示是一個特定 library 的名稱,像是 libmylib.so.1 。
前面以 `lib' 開頭,接著是該 library 的名稱,然後是 `.so' ,接著
是版號,用來表名他的介面;如果介面改變時,就會增加版號來維護相容度。
real name 是實際放有library程式的檔案名稱,後面會再加上 minor 版號與
release 版號,像是 libmylib.so.1.0.0 。
一般來說,版號的改變規則是(印象中在 APress-Difinitive Guide to GCC中有
提到,但目前手邊沒這本書),最尾碼的release版號用於程式內容的修正,
介面完全沒有改變。中間的minor用於有新增加介面,但相舊介面沒改變,所以
與舊版本相容。最前面的version版號用於原介面有移除或改變,與舊版不相容
時。
linker name是用於連結時的名稱,是不含版號的 soname ,如: libmylib.so。
通常 linker name與 real name是用 ln 指到對應的 real name ,用來提供
彈性與維護性。
2.1 編譯
shared library的製作過程較複雜。
$ gcc -c -fPIC hello.c world.c
編譯時要加上 -fPIC 用來產生 position-independent code。也可以用 -fpic
參數。 (不太清楚差異,只知道 -fPIC 較通用於不同平台,但產生的code較大
,而且編譯速度較慢)。
$ gcc -shared -Wl,-soname,libmylib.so.1 -o libmylib.so.1.0.0 \
hello.o world.o
-shared 表示要編譯成 shared library
-Wl 用於參遞參數給linker,因此-soname與libmylib.so.1會被傳給linker處理。
-soname用來指名 soname 為 limylib.so.1
library會被輸出成libmylib.so.1.0.0 (也就是real name)
若不指定 soname 的話,在編譯結連後的執行檔會以連時的library檔名為
soname,並載入他。否則是載入soname指定的library檔案。
可以利用 objdump 來看 library 的 soname。
$ objdump -p libmylib.so | grep SONAME
SONAME libmylib.so.1
若不指名-soname參數的話,則library不會有這個欄位資料。
在編譯後再用 ln 來建立 soname 與 linker name 兩個檔案。
$ ln -s libmylib.so.1.0.0 libmylib.so
$ ln -s libmylib.so.1.0.0 libmylib.so.1
2.2 使用
與使用 static library 同。
$ gcc main.c libmylib.so
以上直接指定與 libmylib.so 連結。
或用
$ gcc main.c -L. -lmylib
linker會搜尋 libmylib.so 來進行連結。
如果目錄下同時有static與shared library的話,會以shared為主。
使用 -static 參數可以避免使用shared連結。
$ gcc main.c -static -L. -lmylib
此時可以用 ldd 看編譯出的執行檔與shared程式庫的相依性
$ldd a.out
linux-gate.so.1 => (0xffffe000)
libmylib.so.1 => not found
libc.so.6 => /lib/libc.so.6 (0xb7dd6000)
/lib/ld-linux.so.2 (0xb7f07000)
輸出結果顯示出該執行檔需要 libmylib.so.1 這個shared library。
會顯示 not found 因為沒指定該library所在的目錄,所找不到該library。
因為編譯時有指定-soname參數為 libmylib.so.1 的關係,所以該執行檔會
載入libmylib.so.1。否則以libmylib.so連結,執行檔則會變成要求載入
libmylib.so
$ ./a.out
./a.out: error while loading shared libraries: libmylib.so.1:
cannot open shared object file: No such file or directory
因為找不到 libmylib.so.1 所以無法執行程式。
有幾個方式可以處理。
a. 把 libmylib.so.1 安裝到系統的library目錄,如/usr/lib下
b. 設定 /etc/ld.so.conf ,加入一個新的library搜尋目錄,並執行ldconfig
更新快取
c. 設定 LD_LIBRARY_PATH 環境變數來搜尋library
這個例子是加入目前的目錄來搜尋要載作的library
$ LD_LIBRARY_PATH=. ./a.out
Hello world.
3. Dynamically loaded libraries
Dynamicaaly loaded libraries 才是像 windows 所用的 DLL ,在使用到
時才載入,編譯連結時不需要相關的library。動態載入庫常被用於像plug-ins
的應用。
3.1 使用方式
動態載入是透過一套 dl function來處理。
#include
void *dlopen(const char *filename, int flag);
開啟載入 filename 指定的 library。
void *dlsym(void *handle, const char *symbol);
取得 symbol 指定的symbol name在library被載入的記憶體位址。
int dlclose(void *handle);
關閉dlopen開啟的handle。
char *dlerror(void);
傳回最近所發生的錯誤訊息。
dltest.c
#include
#include
#includeint main() {
void *handle;
void (*f)();
char *error;
/* 開啟之前所撰寫的 libmylib.so 程式庫 */
handle = dlopen("./libmylib.so", RTLD_LAZY);
if( !handle ) {
fputs( dlerror(), stderr);
exit(1);
}
/* 取得 hello function 的 address */
f = dlsym(handle, "hello");
if(( error=dlerror())!=NULL) {
fputs(error, stderr);
exit(1);
}
/* 呼叫該 function */
f();
dlclose(handle);
}
編譯時要加上 -ldl 參數來與 dl library 連結
$ gcc dltest.c -ldl
結果會印出 Hello 字串
$ ./a.out
Hello
關於dl的詳細內容請參閱 man dlopen
--
參考資料:
Creating a shared and static library with the gnu compiler [gcc]
Makefile
Thursday, November 25, 2021
[xgo] The first phase of hacking xgo
xgo is a great Go CGO cross compiler that can help users to build the application written in Golang to multi-platforms at the same time.
For more information in details, please check out these URLs:
https://github.com/karalabe/xgo
https://www.jianshu.com/p/a6047d3f976e
But, without any further study, we probably cannot understand how it works well. So here I just give some information about it. I will use my Github repository as the example:
Then, we finish the cross-compilation for my application
And actually, xgo, this tool, is based on its prebuilt docker image: "karalabe/xgo-latest" to complete the cross-compilation task. We can check the dockerhub for it as well: https://hub.docker.com/r/karalabe/xgo-latest/builds
So, we can directly run this Docker Image to cross-compile my application based on my case as follows:
Furthermore, we can directly run xgo's build.sh to cross-compile my application in the host instead of in in the container:
Reference:
Golang 交叉编译
Golang交叉编译各个平台的二进制文件
Fyne Cross Compile
Monday, November 22, 2021
[Fun] Detecting the license plate and replacing with a given image
Propably two years ago, I studied a ECCV 2018 paper "License Plate Detection and Recognition in Unconstrained Scenarios", and its GitHub (https://github.com/sergiomsilva/alpr-unconstrained) repository contains the author's implementation. During that time, I had an idea to leveraged its implementation for replacing the license plate with my image. After tryed a couple days, I successed and provided a Python script to detect and replace cars' license plate in the video with a specified image.
I recently commited my modification code and pushed it to my forked repository as follows:
https://github.com/teyenliu/alpr-unconstrained
In this repository, it contains 2 samples which are convenient for users to test.
The way to process a video which will replace any car license plate in the video with a given image content.
$ python license-plate-replacement.py --video cars_mountain_short.mov --replace src/hello.PNG --output output1.avi
$ python license-plate-replacement.py --video cars_mountain.mp4 --replace src/hello.PNG --output output2.avi
The result looks like this:
Friday, November 19, 2021
[Golang] Golang with Cgo 動態連結(dynamic linking) 與 靜態連結(static linking)函式庫範例與筆記 in Windows environment
This post is focusd on how to do dynamic/static linking using Golang with cgo on Windows environment. If you want to see it on Linux environment, please check out another post:
[Golang] Golang with Cgo 動態連結(dynamic linking) 函式庫範例與筆記 in Linux environment
First of all, thanks to someone who had posted several useful articles describing how to do dynamic linking and static linking using Golang(Go) with Cgo in Windows environment as follows:
P.S: For using this example on Linux environment, you should deal with the build process for the static and dynamic libraries.
在 Windows 環境建置動態連結函式庫 (Dynamic-link library),使用 MinGW gcc/g++ 以及 CodeBlock
Go with Cgo 靜態連結 (static linking) 函式庫建置範例與筆記
Go with Cgo 動態連結 (dynamic linking) 函式庫建置範例與筆記
Wednesday, November 3, 2021
[tmux] The most common used tmux commands
This post can help you quickly learn how to use tmux in just a few minutes.
Get tmux session ready to start (new session)
# list sessions
$ tmux ls
Split a tmux window into two panes
P.S: If you want to close one of the pane you created,
For instance: split a tmux window into two panes vertically
Adjust a splitted pane size
You also are able to ajust the splitted pane vertically or horizontally.
For instance: There are 5 panes in the window and I can adjust the currently focused pane.
Attach tmux session or detach it
# attach a background session
Delete tmux session
Thursday, October 28, 2021
[Golang] The example function of converting Struct or Map data to Map[string]interface{} Type
I recently encountered a use case that I need to convert my struct or map value to the type of map[string]interface{}. After surveying and studying for a while, I figure out how to deal with this kind of task and the way to do it.
The following function ConvertStructToMap() is the example of converting Struct or Map data to Map[string]interface{} Type. It receives the argument as interface type and uses reflect function to check the receiver type is correct and what we want. But, it has a constrain, which if you give a map, and Map's key type must be string.
How to use it? Here we go! You can try it on the Go Playground:
https://play.golang.org/p/TLKBIbSxIQj
or just take a look at the sample as follows:
Run Result==>
map[JsonName:john smith MsgScore:30]
map[Score1:100 Score2:60]
Monday, October 18, 2021
Python matplotlib 中文亂碼解決 in Linux or Windows
大家是否有遇過當使用matplotlib時,遇到顯示不出中文或是中文為亂碼的問題。無論是在Anaconda內使用matplotlib或是直接在Python安裝環境內使用matplotlib,基本解決方式是一樣的。本文將以簡單與條列方式描述在Linux 或是 Windows 解決方法。
1. 下載字型的來源
2. 字型安裝
執行python command, 並以下列方式找到matplotlib放置ttf字型檔位置與設定檔
根據下列不同環境的執行結果內放置ttf字型檔
Windows:
放到: C:\Users\User\AppData\Local\Packages\PythonSoftwareFoundation.Python.3.9_qbz5n2kfra8p0\LocalCache\local-packages\Python39\site-packages\matplotlib\mpl-data\fonts\ttf
修改設定檔 C:\Users\User\AppData\Local\Packages\PythonSoftwareFoundation.Python.3.9_qbz5n2kfra8p0\LocalCache\local-packages\Python39\site-packages\matplotlib\mpl-data\matplotlibrc
Linux:
放到: /usr/local/lib/python2.7/dist-packages/matplotlib/mpl-data/fonts/ttf修改設定檔 /usr/local/lib/python2.7/dist-packages/matplotlib/mpl-data/matplotlibrc
Anaconda:
放到: /opt/conda/lib/python3.7/site-packages/matplotlib/mpl-data/fonts/ttf修改設定檔 /opt/conda/lib/python3.7/site-packages/matplotlib/mpl-data/matplotlibrc
範例如下:
Wednesday, October 6, 2021
個人常用的 Git Commands
0. 相關初始設定
# 查看到我們名下的遠端項目
#獲取upstream的最新版本
git fetch upstream
#將upstream merge到我們當前分支
git merge upstream/master
Trouble Shooting
1. 先刪除分支然後上傳Tag:
git branch -D testtag
或是
2. 先刪除tag然後上傳分支:
git tag -d testtag
Monday, September 27, 2021
簡介 Fluentd 寫入 InfluxDB 透過 Fluent::Plugin::InfluxDB
fluent-logger-golang (fluent client-go API)
目前使用此fluent client-go API,將資料傳入Fluentd,
會使用其Async模式來避免後續的latency event被block住, 當Fluentd server有問題時:// Use "Async" to enable asynchronous I/O (connect and write)
// for sending events to Fluentd without blocking
setting.FluentdLogger.Logger, errF = fluent.New(
fluent.Config{Async: true, FluentHost: fluenthost, FluentPort: intFluentPort})
if errF != nil {
fmt.Println("[ERROR]:", errF)
setting.FluentdLogger.Enabled = false
}
P.S: When Fluentd server has a problem, the events which are going to send will be buffered on the memory. The default size is 8192
BufferLimit
The example of using fluent-logger-golang to send data:Sets the number of events buffered on the memory.
tag := "apm.latency"
var data = map[string]string{
"mytimestamp": strconv.FormatInt(time.Now().Unix(), 10),
"mydata": "hoge",
"myjob": "apm",
"myvalue": "55.55",
}
error := logger.Post(tag, data)Fluentd
Fluentd
Fluentd安裝,請參考官網
https://docs.fluentd.org/installation/install-by-deb
Fluentd操作
# 啟動/關閉/查看服務
sudo systemctl start td-agent.service
sudo systemctl stop td-agent.service
sudo systemctl status td-agent.service
sudo systemctl restart td-agent.service
#修改Fluentd設定
vi /etc/td-agent/td-agent.conf
#查看Fluentd logs
cat /var/log/td-agent/td-agent.log
InfluxDB 2.X
2.0 以上的InfluxDB 會需要token,Client端才有權限讀寫,其中一種方式查出Token是用InfluxDB自己的Web http://<your ip address>:8086,登入後點選"Data"
例如:
選擇GO
可以看到 token的值
// You can generate a Token from the "Tokens Tab" in the UI const token = "Iiq0TiIpL9lXn2GATwh3WeZBkLq-SEul6C0yrKLjq4T4WZ9b0BKVAsFeNs8q0Is93SMbhF0l63s4DwJja4MSbw=="
Fluent::Plugin::InfluxDB (This plugin is for using with InfluxDB 2.x 目前使用此Plugin)
This repository contains the reference Fluentd plugin for the InfluxDB 2.0.Fluentd需要此Plugin來對應寫資料到InfluxDB2
我目前用的: The configuration of /etc/td-agent/td-agent.conf
#<match apm.**>
# @type stdout
#</match>
<match apm.**>
@type copy
<store>
@type influxdb2
url <https://localhost:8086>
token Iiq0TiIpL9lXn2GATwh3WeZBkLq-SEul6C0yrKLjq4TXXXX
use_ssl false
bucket apm
org com
time_precision s
tag_keys ["mytimestamp","mydata"]
field_keys ["myvalue"]
</store>
<store>
@type stdout
</store>
</match>
The result in InfluxDB
安裝此plugin:
sudo td-agent-gem install fluent-plugin-influxdb-v2
sudo td-agent-gem uninstall fluent-plugin-influxdb-v2
influxdb-plugin-fluent (只適用於InfluxDB 1.x instances)
fluent-plugin-influxdb is a buffered output plugin for fluentd and influxDB.
Configuration Example
<match apm.**>
@type influxdb
host localhost
port 8086
dbname apm
user danny
password xxxxxxxxx
use_ssl false
time_precision s
tag_keys ["timestamp", "data"]
sequence_tag _seq
</match>
安裝 plugin:
sudo td-agent-gem install fluent-plugin-influxdb
sudo td-agent-gem uninstall fluent-plugin-influxdb
Reference
遇過的Error Message
[warn]: #0 failed to flush the buffer. retry_time=7 next_retry_seconds=2021-04-22 14:28:33 14429497340994114529/137438953472000000000 +0800 chunk="5c089be4bdfb9a28fb4b5ac58228a90 7" error_class=InfluxDB2::InfluxError error="failure writing points to database: partial write: points beyond retention policy dropped=1" 2021-04-22 14:27:22 +0800 [warn]: #0 suppressed same stacktrace
https://stackoverflow.com/questions/54359348/unable-to-insert-data-in-influxdb