使用docker過程當中遇過的最詭異的問題,服務在本地環境中,經過在IDEA裏面運行,或者使用java -jar ***.war運行,獲取驗證碼圖片都沒有問題,可是運行在docker中,圖片正常返回,可是上面的漢字卻沒法顯示。java
(正常返回)docker
(非正常返回)bash
將漢字寫入圖片的代碼如:app
private String drawRandomNum(Graphics2D g) { StringBuffer sb = new StringBuffer(); // 設置顏色 g.setColor(Color.YELLOW); // 設置字體 g.setFont(new Font("宋體", Font.ITALIC, 20)); // 準備經常使用漢字集 String base = "\u7684\u4e00\u4e86\u662f\u6211\u4e0d\u5728\u4eba\u4eec\u6709\u6765"; int x = 5; // 控制字數 for (int i = 0; i < 4; i++) { // 設置字體旋轉角度 int degree = new Random().nextInt() % 30; // 截取漢字 String ch = base.charAt(new Random().nextInt(base.length())) + ""; sb.append(ch); // 正向角度 g.rotate(degree * Math.PI / 180, x, 20); g.drawString(ch, x, 20); // 反向角度 g.rotate(-degree * Math.PI / 180, x, 20); x += 30; } System.out.println(sb.toString()); return sb.toString(); }
基本邏輯是從unicode編碼的漢字字符集中隨機抽取4個漢字,調用Graphics2D的drawString方法畫入。dom
排查問題的第一步,將漢字字符集轉化爲英語字符集,方法調用沒有問題,英語字母能夠正常顯示,問題彷佛落在docker環境對中文字符的支持上。maven
經過參考,對container中locale進行檢查和重置(修改container中locale的方法):字體
C.UTF-8原本就已經支持中文,因此問題也不在locale。編碼
由於代碼中使用了Graphics2D進行畫圖,試着用關鍵字」Graphics2D 畫漢字 docker 亂碼「百度一下,果真搜出了好東西。.net
問題的根源在於:container環境中缺少對中文字體的支持,須要添加字體文件simsun.ttf 到/usr/share/fonts中,具體步驟如:3d
1)進入container:
docker exec -it 40ebba024c0b bash
2)打開/usr/share/fonts:
cd /usr/share/fonts
3)下載字體文件:
wget https://dlc2.pconline.com.cn/filedown_367689_7048847/3BswZ0YQ/simsun.zip
結果如:
4)解壓simsun.zip:
unzip simsun.zip
結果如:
5)拷貝一份simsun.ttf,並重命名爲simsun.tt
cp simsun.ttf simsun.tt
6)退出並重啓container:
問題解決!!
結論:Docker環境和本地環境是存在差別的,在解決具體問題的過程當中,須要將這種差別添加到打包的過程當中,如本例中須要將字體文件進行拷貝,能夠在Dockerfile中添加
COPY simsun.ttf /usr/share/fonts/simsun.tt
(因爲使用了docker-maven-plugin進行打包,這一步並未實踐)