精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Java如何實現長圖文生成

開發 后端
很久很久以前,就覺得微博的長圖文實現得非常有意思,將排版直接以最終的圖片輸出,收藏查看分享都很方便,現在則自己動手實現一個簡單版本的,實現一個用于生成微博長圖文樣式的包裝類。

[[200733]]

長圖文生成

很久很久以前,就覺得微博的長圖文實現得非常有意思,將排版直接以最終的圖片輸出,收藏查看分享都很方便,現在則自己動手實現一個簡單版本的

目標

首先定義下我們預期達到的目標:根據文字 + 圖片生成長圖文

目標拆解

  • 支持大段文字生成圖片
  • 支持插入圖片
  • 支持上下左右邊距設置
  • 支持字體選擇
  • 支持字體顏色
  • 支持左對齊,居中,右對齊

預期結果

我們將通過spring-boot搭建一個生成長圖文的http接口,通過傳入參數來指定各種配置信息,下面是一個最終調用的示意圖

 

 

設計&實現

長圖文的生成,采用awt進行文字繪制和圖片繪制

1. 參數選項 ImgCreateOptions

根據我們的預期目標,設定配置參數,基本上會包含以下參數

  1. @Getter 
  2. @Setter 
  3. @ToString 
  4. public class ImgCreateOptions { 
  5.  
  6.     /** 
  7.      * 繪制的背景圖 
  8.      */ 
  9.     private BufferedImage bgImg; 
  10.  
  11.  
  12.     /** 
  13.      * 生成圖片的寬 
  14.      */ 
  15.     private Integer imgW; 
  16.  
  17.  
  18.     private Font font = new Font("宋體", Font.PLAIN, 18); 
  19.  
  20.     /** 
  21.      * 字體色 
  22.      */ 
  23.     private Color fontColor = Color.BLACK; 
  24.  
  25.  
  26.     /** 
  27.      * 兩邊邊距 
  28.      */ 
  29.     private int leftPadding; 
  30.  
  31.     /** 
  32.      * 上邊距 
  33.      */ 
  34.     private int topPadding; 
  35.  
  36.     /** 
  37.      * 底邊距 
  38.      */ 
  39.     private int bottomPadding; 
  40.  
  41.     /** 
  42.      * 行距 
  43.      */ 
  44.     private int linePadding; 
  45.  
  46.  
  47.     private AlignStyle alignStyle; 
  48.  
  49.     /** 
  50.      * 對齊方式 
  51.      */ 
  52.     public enum AlignStyle { 
  53.         LEFT
  54.         CENTER, 
  55.         RIGHT
  56.  
  57.  
  58.         private static Map<String, AlignStyle> map = new HashMap<>(); 
  59.  
  60.         static { 
  61.             for(AlignStyle style: AlignStyle.values()) { 
  62.                 map.put(style.name(), style); 
  63.             } 
  64.         } 
  65.  
  66.  
  67.         public static AlignStyle getStyle(String name) { 
  68.             name = name.toUpperCase(); 
  69.             if (map.containsKey(name)) { 
  70.                 return map.get(name); 
  71.             } 
  72.  
  73.             return LEFT
  74.         } 
  75.     } 

 

2. 封裝類 ImageCreateWrapper

封裝配置參數的設置,繪制文本,繪制圖片的操作方式,輸出樣式等接口

  1. public class ImgCreateWrapper { 
  2.  
  3.  
  4.     public static Builder build() { 
  5.         return new Builder(); 
  6.     } 
  7.  
  8.  
  9.     public static class Builder { 
  10.         /** 
  11.          * 生成的圖片創建參數 
  12.          */ 
  13.         private ImgCreateOptions options = new ImgCreateOptions(); 
  14.  
  15.  
  16.         /** 
  17.          * 輸出的結果 
  18.          */ 
  19.         private BufferedImage result; 
  20.  
  21.  
  22.         private final int addH = 1000; 
  23.  
  24.  
  25.         /** 
  26.          * 實際填充的內容高度 
  27.          */ 
  28.         private int contentH; 
  29.  
  30.  
  31.         private Color bgColor; 
  32.  
  33.         public Builder setBgColor(int color) { 
  34.             return setBgColor(ColorUtil.int2color(color)); 
  35.         } 
  36.  
  37.         /** 
  38.          * 設置背景圖 
  39.          * 
  40.          * @param bgColor 
  41.          * @return 
  42.          */ 
  43.         public Builder setBgColor(Color bgColor) { 
  44.             this.bgColor = bgColor; 
  45.             return this; 
  46.         } 
  47.  
  48.  
  49.         public Builder setBgImg(BufferedImage bgImg) { 
  50.             options.setBgImg(bgImg); 
  51.             return this; 
  52.         } 
  53.  
  54.  
  55.         public Builder setImgW(int w) { 
  56.             options.setImgW(w); 
  57.             return this; 
  58.         } 
  59.  
  60.         public Builder setFont(Font font) { 
  61.             options.setFont(font); 
  62.             return this; 
  63.         } 
  64.  
  65.         public Builder setFontName(String fontName) { 
  66.             Font font = options.getFont(); 
  67.             options.setFont(new Font(fontName, font.getStyle(), font.getSize())); 
  68.             return this; 
  69.         } 
  70.  
  71.  
  72.         public Builder setFontColor(int fontColor) { 
  73.             return setFontColor(ColorUtil.int2color(fontColor)); 
  74.         } 
  75.  
  76.         public Builder setFontColor(Color fontColor) { 
  77.             options.setFontColor(fontColor); 
  78.             return this; 
  79.         } 
  80.  
  81.         public Builder setFontSize(Integer fontSize) { 
  82.             Font font = options.getFont(); 
  83.             options.setFont(new Font(font.getName(), font.getStyle(), fontSize)); 
  84.             return this; 
  85.         } 
  86.  
  87.         public Builder setLeftPadding(int leftPadding) { 
  88.             options.setLeftPadding(leftPadding); 
  89.             return this; 
  90.         } 
  91.  
  92.         public Builder setTopPadding(int topPadding) { 
  93.             options.setTopPadding(topPadding); 
  94.             contentH = topPadding; 
  95.             return this; 
  96.         } 
  97.  
  98.         public Builder setBottomPadding(int bottomPadding) { 
  99.             options.setBottomPadding(bottomPadding); 
  100.             return this; 
  101.         } 
  102.  
  103.         public Builder setLinePadding(int linePadding) { 
  104.             options.setLinePadding(linePadding); 
  105.             return this; 
  106.         } 
  107.  
  108.         public Builder setAlignStyle(String style) { 
  109.             return setAlignStyle(ImgCreateOptions.AlignStyle.getStyle(style)); 
  110.         } 
  111.  
  112.         public Builder setAlignStyle(ImgCreateOptions.AlignStyle alignStyle) { 
  113.             options.setAlignStyle(alignStyle); 
  114.             return this; 
  115.         } 
  116.  
  117.  
  118.         public Builder drawContent(String content) { 
  119.             // xxx 
  120.             return this; 
  121.         } 
  122.  
  123.  
  124.         public Builder drawImage(String img) { 
  125.             BufferedImage bfImg; 
  126.             try { 
  127.                  bfImg = ImageUtil.getImageByPath(img); 
  128.             } catch (IOException e) { 
  129.                 log.error("load draw img error! img: {}, e:{}", img, e); 
  130.                 throw new IllegalStateException("load draw img error! img: " + img, e); 
  131.             } 
  132.  
  133.             return drawImage(bfImg); 
  134.         } 
  135.  
  136.  
  137.         public Builder drawImage(BufferedImage bufferedImage) { 
  138.  
  139.            // xxx 
  140.            return this; 
  141.         } 
  142.  
  143.  
  144.         public BufferedImage asImage() { 
  145.             int realH = contentH + options.getBottomPadding(); 
  146.  
  147.             BufferedImage bf = new BufferedImage(options.getImgW(), realH, BufferedImage.TYPE_INT_ARGB); 
  148.             Graphics2D g2d = bf.createGraphics(); 
  149.  
  150.             if (options.getBgImg() == null) { 
  151.                 g2d.setColor(bgColor == null ? Color.WHITE : bgColor); 
  152.                 g2d.fillRect(0, 0, options.getImgW(), realH); 
  153.             } else { 
  154.                 g2d.drawImage(options.getBgImg(), 0, 0, options.getImgW(), realH, null); 
  155.             } 
  156.  
  157.             g2d.drawImage(result, 0, 0, null); 
  158.             g2d.dispose(); 
  159.             return bf; 
  160.         } 
  161.  
  162.  
  163.         public String asString() throws IOException { 
  164.             BufferedImage img = asImage(); 
  165.             return Base64Util.encode(img, "png"); 
  166.         } 

 

上面具體的文本和圖片繪制實現沒有,后面詳細講解,這里主要關注的是一個參數 contentH, 表示實際繪制的內容高度(包括上邊距),因此最終生成圖片的高度應該是

int realH = contentH + options.getBottomPadding();

其次簡單說一下上面的圖片輸出方法:com.hust.hui.quickmedia.common.image.ImgCreateWrapper.Builder#asImage

  • 計算最終生成圖片的高度(寬度由輸入參數指定)
  • 繪制背景(如果沒有背景圖片,則用純色填充)
  • 繪制實體內容(即繪制的文本,圖片)

3. 內容填充 GraphicUtil

具體的內容填充,區分為文本繪制和圖片繪制

設計

  • 考慮到在填充的過程中,可以自由設置字體,顏色等,所以在我們的繪制方法中,直接實現掉內容的繪制填充,即 drawXXX 方法真正的實現了內容填充,執行完之后,內容已經填充到畫布上了
  • 圖片繪制,考慮到圖片本身大小和最終結果的大小可能有沖突,采用下面的規則
    • 繪制圖片寬度 <=(指定生成圖片寬 - 邊距),全部填充
    • 繪制圖片寬度 >(指定生成圖片寬 - 邊距),等比例縮放繪制圖片
  • 文本繪制,換行的問題
    • 每一行允許的文本長度有限,超過時,需要自動換行處理

文本繪制

考慮基本的文本繪制,流程如下

  • 創建BufferImage對象
  • 獲取Graphic2d對象,操作繪制
  • 設置基本配置信息
  • 文本按換行進行拆分為字符串數組, 循環繪制單行內容
    • 計算當行字符串,實際繪制的行數,然后進行拆分
    • 依次繪制文本(需要注意y坐標的變化)

下面是具體的實現

  1. public static int drawContent(Graphics2D g2d, 
  2.                                   String content, 
  3.                                   int y, 
  4.                                   ImgCreateOptions options) { 
  5.  
  6.     int w = options.getImgW(); 
  7.     int leftPadding = options.getLeftPadding(); 
  8.     int linePadding = options.getLinePadding(); 
  9.     Font font = options.getFont(); 
  10.  
  11.  
  12.     // 一行容納的字符個數 
  13.     int lineNum = (int) Math.floor((w - (leftPadding << 1)) / (double) font.getSize()); 
  14.  
  15.     // 對長串字符串進行分割成多行進行繪制 
  16.     String[] strs = splitStr(content, lineNum); 
  17.  
  18.     g2d.setFont(font); 
  19.  
  20.     g2d.setColor(options.getFontColor()); 
  21.     int index = 0; 
  22.     int x; 
  23.     for (String tmp : strs) { 
  24.         x = calOffsetX(leftPadding, w, tmp.length() * font.getSize(), options.getAlignStyle()); 
  25.         g2d.drawString(tmp, x, y + (linePadding + font.getSize()) * index); 
  26.         index++; 
  27.     } 
  28.  
  29.  
  30.     return y + (linePadding + font.getSize()) * (index); 
  31.  
  32. /** 
  33.  * 計算不同對其方式時,對應的x坐標 
  34.  * 
  35.  * @param padding 左右邊距 
  36.  * @param width   圖片總寬 
  37.  * @param strSize 字符串總長 
  38.  * @param style   對其方式 
  39.  * @return 返回計算后的x坐標 
  40.  */ 
  41. private static int calOffsetX(int padding, 
  42.                               int width, 
  43.                               int strSize, 
  44.                               ImgCreateOptions.AlignStyle style) { 
  45.     if (style == ImgCreateOptions.AlignStyle.LEFT) { 
  46.         return padding; 
  47.     } else if (style == ImgCreateOptions.AlignStyle.RIGHT) { 
  48.         return width - padding - strSize; 
  49.     } else { 
  50.         return (width - strSize) >> 1; 
  51.     } 
  52.  
  53.  
  54. /** 
  55.  * 按照長度對字符串進行分割 
  56.  * <p> 
  57.  * fixme 包含emoj表情時,兼容一把 
  58.  * 
  59.  * @param str      原始字符串 
  60.  * @param splitLen 分割的長度 
  61.  * @return 
  62.  */ 
  63. public static String[] splitStr(String str, int splitLen) { 
  64.     int len = str.length(); 
  65.     int size = (int) Math.ceil(len / (float) splitLen); 
  66.  
  67.     String[] ans = new String[size]; 
  68.     int start = 0; 
  69.     int end = splitLen; 
  70.     for (int i = 0; i < size; i++) { 
  71.         ans[i] = str.substring(start, end > len ? len : end); 
  72.         start = end
  73.         end += splitLen; 
  74.     } 
  75.  
  76.     return ans; 

 

上面的實現比較清晰了,圖片的繪制則更加簡單

圖片繪制

只需要重新計算下待繪制圖片的寬高即可,具體實現如下

  1. /** 
  2.  * 在原圖上繪制圖片 
  3.  * 
  4.  * @param source  原圖 
  5.  * @param dest    待繪制圖片 
  6.  * @param y       待繪制的y坐標 
  7.  * @param options 
  8.  * @return 繪制圖片的高度 
  9.  */ 
  10. public static int drawImage(BufferedImage source, 
  11.                             BufferedImage dest, 
  12.                             int y, 
  13.                             ImgCreateOptions options) { 
  14.     Graphics2D g2d = getG2d(source); 
  15.     int w = Math.min(dest.getWidth(), options.getImgW() - (options.getLeftPadding() << 1)); 
  16.     int h = w * dest.getHeight() / dest.getWidth(); 
  17.  
  18.     int x = calOffsetX(options.getLeftPadding(), 
  19.             options.getImgW(), w, options.getAlignStyle()); 
  20.  
  21.     // 繪制圖片 
  22.     g2d.drawImage(dest, 
  23.             x, 
  24.             y + options.getLinePadding(), 
  25.             w, 
  26.             h, 
  27.             null); 
  28.     g2d.dispose(); 
  29.  
  30.     return h; 
  31.  
  32. public static Graphics2D getG2d(BufferedImage bf) { 
  33.         Graphics2D g2d = bf.createGraphics(); 
  34.  
  35.     g2d.setRenderingHint(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY); 
  36.     g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); 
  37.     g2d.setRenderingHint(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_QUALITY); 
  38.     g2d.setRenderingHint(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE); 
  39.     g2d.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON); 
  40.     g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR); 
  41.     g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY); 
  42.     g2d.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE); 
  43.  
  44.     return g2d; 

 

4. 內容渲染

前面只是給出了單塊內容(如一段文字,一張圖片)的渲染,存在一些問題

  • 繪制的內容超過畫布的高度如何處理
  • 文本繪制要求傳入的文本沒有換行符,否則換行不生效
  • 交叉繪制的場景,如何重新計算y坐標

解決這些問題則是在 ImgCreateWrapper 的具體繪制中進行了實現,先看文本的繪制

  • 根據換行符對字符串進行拆分
  • 計算繪制內容最終轉換為圖片時,所占用的高度
  • 重新生成畫布 BufferedImage result
    • 如果result為空,則直接生成
    • 如果最終生成的高度,超過已有畫布的高度,則生成一個更高的畫布,并將原來的內容繪制上去
  • 迭代繪制單行內容
  1. public Builder drawContent(String content) { 
  2.     String[] strs = StringUtils.split(content, "\n"); 
  3.     if (strs.length == 0) { // empty line 
  4.         strs = new String[1]; 
  5.         strs[0] = " "
  6.     } 
  7.  
  8.     int fontSize = options.getFont().getSize(); 
  9.     int lineNum = calLineNum(strs, options.getImgW(), options.getLeftPadding(), fontSize); 
  10.     // 填寫內容需要占用的高度 
  11.     int height = lineNum * (fontSize + options.getLinePadding()); 
  12.  
  13.     if (result == null) { 
  14.         result = GraphicUtil.createImg(options.getImgW(), 
  15.                 Math.max(height + options.getTopPadding() + options.getBottomPadding(), BASE_ADD_H), 
  16.                 null); 
  17.     } else if (result.getHeight() < contentH + height + options.getBottomPadding()) { 
  18.         // 超過原來圖片高度的上限, 則需要擴充圖片長度 
  19.         result = GraphicUtil.createImg(options.getImgW(), 
  20.                 result.getHeight() + Math.max(height + options.getBottomPadding(), BASE_ADD_H), 
  21.                 result); 
  22.     } 
  23.  
  24.  
  25.     // 繪制文字 
  26.     Graphics2D g2d = GraphicUtil.getG2d(result); 
  27.     int index = 0; 
  28.     for (String str : strs) { 
  29.         GraphicUtil.drawContent(g2d, str, 
  30.                 contentH + (fontSize + options.getLinePadding()) * (++index
  31.                 , options); 
  32.     } 
  33.     g2d.dispose(); 
  34.  
  35.     contentH += height; 
  36.     return this; 
  37.  
  38.  
  39. /** 
  40.  * 計算總行數 
  41.  * 
  42.  * @param strs     字符串列表 
  43.  * @param w        生成圖片的寬 
  44.  * @param padding  渲染內容的左右邊距 
  45.  * @param fontSize 字體大小 
  46.  * @return 
  47.  */ 
  48. private int calLineNum(String[] strs, int w, int padding, int fontSize) { 
  49.     // 每行的字符數 
  50.     double lineFontLen = Math.floor((w - (padding << 1)) / (double) fontSize); 
  51.  
  52.  
  53.     int totalLine = 0; 
  54.     for (String str : strs) { 
  55.         totalLine += Math.ceil(str.length() / lineFontLen); 
  56.     } 
  57.  
  58.     return totalLine; 

 

上面需要注意的是畫布的生成規則,特別是高度超過上限之后,重新計算圖片高度時,需要額外注意新增的高度,應該為基本的增量與(繪制內容高度+下邊距)的較大值

  1. int realAddH = Math.max(bufferedImage.getHeight() + options.getBottomPadding() + options.getTopPadding(), BASE_ADD_H) 

重新生成畫布實現 com.hust.hui.quickmedia.common.util.GraphicUtil#createImg

  1. public static BufferedImage createImg(int w, int h, BufferedImage img) { 
  2.     BufferedImage bf = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB); 
  3.     Graphics2D g2d = bf.createGraphics(); 
  4.  
  5.     if (img != null) { 
  6.         g2d.setComposite(AlphaComposite.Src); 
  7.         g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); 
  8.         g2d.drawImage(img, 0, 0, null); 
  9.     } 
  10.     g2d.dispose(); 
  11.     return bf; 

 

上面理解之后,繪制圖片就比較簡單了,基本上行沒什么差別

  1. public Builder drawImage(String img) { 
  2.     BufferedImage bfImg; 
  3.     try { 
  4.         bfImg = ImageUtil.getImageByPath(img); 
  5.     } catch (IOException e) { 
  6.         log.error("load draw img error! img: {}, e:{}", img, e); 
  7.         throw new IllegalStateException("load draw img error! img: " + img, e); 
  8.     } 
  9.  
  10.     return drawImage(bfImg); 
  11.  
  12.  
  13. public Builder drawImage(BufferedImage bufferedImage) { 
  14.  
  15.     if (result == null) { 
  16.         result = GraphicUtil.createImg(options.getImgW(), 
  17.                 Math.max(bufferedImage.getHeight() + options.getBottomPadding() + options.getTopPadding(), BASE_ADD_H), 
  18.                 null); 
  19.     } else if (result.getHeight() < contentH + bufferedImage.getHeight() + options.getBottomPadding()) { 
  20.         // 超過閥值 
  21.         result = GraphicUtil.createImg(options.getImgW(), 
  22.                 result.getHeight() + Math.max(bufferedImage.getHeight() + options.getBottomPadding() + options.getTopPadding(), BASE_ADD_H), 
  23.                 result); 
  24.     } 
  25.  
  26.     // 更新實際高度 
  27.     int h = GraphicUtil.drawImage(result, 
  28.             bufferedImage, 
  29.             contentH, 
  30.             options); 
  31.     contentH += h + options.getLinePadding(); 
  32.     return this; 

 

5. http接口

上面實現的生成圖片的公共方法,在 quick-media 工程中,利用spring-boot搭建了一個web服務,提供了一個http接口,用于生成長圖文,最終的成果就是我們開頭的那個gif圖的效果,相關代碼就沒啥好說的,有興趣的可以直接查看工程源碼,鏈接看最后

測試驗證

上面基本上完成了我們預期的目標,接下來則是進行驗證,測試代碼比較簡單,先準備一段文本,這里拉了一首詩

招魂酹翁賓旸

鄭起

君之在世帝敕下,君之謝世帝敕回。

魂之為變性原返,氣之為物情本開。

於戲龍兮鳳兮神氣盛,噫嘻鬼兮歸兮大塊埃。

身可朽名不可朽,骨可灰神不可灰。

采石捉月李白非醉,耒陽避水子美非災。

長孫王吉命不夭,玉川老子詩不徘。

新城羅隱在奇特,錢塘潘閬終崔嵬。

陰兮魄兮曷往,陽兮魄兮曷來。

君其歸來,故交寥落更散漫。

君來歸來,帝城絢爛可徘徊。

君其歸來,東西南北不可去。

君其歸來。

春秋霜露令人哀。

花之明吾無與笑,葉之隕吾實若摧。

曉猿嘯吾聞淚墮,宵鶴立吾見心猜。

玉泉其清可鑒,西湖其甘可杯。

孤山暖梅香可嗅,花翁葬薦菊之隈。

君其歸來,可伴逋仙之梅,去此又奚之哉。

測試代碼

  1. @Test 
  2. public void testGenImg() throws IOException { 
  3.     int w = 400; 
  4.     int leftPadding = 10; 
  5.     int topPadding = 40; 
  6.     int bottomPadding = 40; 
  7.     int linePadding = 10; 
  8.     Font font = new Font("宋體", Font.PLAIN, 18); 
  9.  
  10.     ImgCreateWrapper.Builder build = ImgCreateWrapper.build() 
  11.             .setImgW(w) 
  12.             .setLeftPadding(leftPadding) 
  13.             .setTopPadding(topPadding) 
  14.             .setBottomPadding(bottomPadding) 
  15.             .setLinePadding(linePadding) 
  16.             .setFont(font) 
  17.             .setAlignStyle(ImgCreateOptions.AlignStyle.CENTER) 
  18. //                .setBgImg(ImageUtil.getImageByPath("qrbg.jpg")) 
  19.             .setBgColor(0xFFF7EED6) 
  20.             ; 
  21.  
  22.  
  23.     BufferedReader reader = FileReadUtil.createLineRead("text/poem.txt"); 
  24.     String line; 
  25.     int index = 0; 
  26.     while ((line = reader.readLine()) != null) { 
  27.         build.drawContent(line); 
  28.  
  29.         if (++index == 5) { 
  30.             build.drawImage(ImageUtil.getImageByPath("https://static.oschina.net/uploads/img/201708/12175633_sOfz.png")); 
  31.         } 
  32.  
  33.         if (index == 7) { 
  34.             build.setFontSize(25); 
  35.         } 
  36.  
  37.         if (index == 10) { 
  38.             build.setFontSize(20); 
  39.             build.setFontColor(Color.RED); 
  40.         } 
  41.     } 
  42.  
  43.     BufferedImage img = build.asImage(); 
  44.     String out = Base64Util.encode(img, "png"); 
  45.     System.out.println("<img src=\"data:image/png;base64," + out + "\" />"); 

 

輸出圖片

責任編輯:龐桂玉 來源: 六月依的博客
相關推薦

2023-03-13 15:56:00

模型框架

2017-10-12 15:34:17

2025-11-05 08:48:39

2023-11-20 12:49:01

2024-06-14 16:24:42

2025-09-10 02:10:00

AI開發CPT-4

2025-10-13 02:00:00

2012-04-11 15:41:48

JavaNIO

2022-07-18 14:33:05

PythonPDF報告

2024-01-06 16:40:47

視頻模型

2021-02-26 12:37:39

WebSocketOkHttp連接

2013-03-15 10:57:13

AJAXDotNet

2024-05-10 07:58:03

2015-10-26 15:48:51

安裝Ubuntu 15.1Linux

2025-01-17 13:53:11

AI大模型檢測工具

2020-09-15 10:45:06

PythonPyQt5Matplotlib

2023-02-26 10:16:19

JavaPDF文檔

2011-01-20 09:13:18

Postfix

2025-10-21 01:00:00

2011-01-19 17:34:39

Postfix如何接收郵件
點贊
收藏

51CTO技術棧公眾號

亚洲欧美丝袜中文综合| 日本亚洲欧美在线| 国产精品中文| 亚洲风情在线资源站| 久久精品成人一区二区三区蜜臀| 亚洲 欧美 中文字幕| 99久久国产综合精品成人影院| 日韩一区国产二区欧美三区| 亚洲午夜无码av毛片久久| 97在线观看免费观看高清| 国产毛片精品一区| 欧美又大又粗又长| 欧美国产日韩在线观看成人| 久久婷婷国产| 欧美一区中文字幕| 欧美xxxxx在线视频| 超碰在线caoporen| 久久精品一区二区三区不卡 | 黄色激情在线播放| 国产精品色噜噜| 国内成+人亚洲| 国产一区二区自拍视频| 性xx色xx综合久久久xx| 欧美国产亚洲视频| 手机看片国产日韩| 日韩啪啪网站| 日韩欧美国产wwwww| 网站一区二区三区| 欧美gv在线观看| 亚洲综合一二区| 一区二区日本伦理| 美女欧美视频在线观看免费 | 麻豆传媒视频在线| 国产婷婷一区二区| 免费国产在线精品一区二区三区| 99久久亚洲精品日本无码 | 五月天开心婷婷| yy6080久久伦理一区二区| 亚洲韩国一区二区三区| 中文字幕超清在线免费观看| 国产黄色在线播放| 久久品道一品道久久精品| 成人免费在线看片| 精品国产999久久久免费| 久国产精品韩国三级视频| 国产精品igao视频| 中文字幕高清在线免费播放| 亚洲深夜av| 777777777亚洲妇女| 国产极品在线播放| 欧美精品三级| 色综合久久久888| 四虎免费在线视频| 这里只有精品在线| 不卡av在线播放| 超碰手机在线观看| 欧美国产三区| 欧美大片免费观看在线观看网站推荐| 久久国产波多野结衣| 国产精品99久久| 久久精品亚洲94久久精品| 午夜激情福利电影| 91精品亚洲| 欧美另类在线观看| 久久精品国产亚洲AV无码男同| 综合激情一区| 欧美—级高清免费播放| 日本三级网站在线观看| 在线日韩中文| 欧美最猛性xxxx| 国产一卡二卡三卡| 老司机免费视频一区二区三区| 国产综合视频在线观看| 99久久精品无免国产免费| 国产91在线观看丝袜| 国精产品一区二区| 黄色片免费在线| 亚洲少妇最新在线视频| 91免费国产精品| av日韩电影| 欧美日韩亚洲高清一区二区| 91aaa精品| 久久综合五月婷婷| 国产香蕉精品视频一区二区三区 | 在线免费观看视频黄| 小说区图片区亚洲| 精品国产伦理网| 免费污网站在线观看| 日韩av片子| 色综合天天综合网国产成人网 | 欧美高清视频www夜色资源网| www.污网站| 国产精品香蕉| 夜夜嗨av一区二区三区免费区| 欧美日韩午夜视频| 99riav1国产精品视频| 国产精品com| 后入内射欧美99二区视频| 久久奇米777| 一区二区三区四区免费观看| 美女搞黄视频在线观看| 欧美日韩国产免费一区二区| 波多野结衣加勒比| 久久精品亚洲人成影院| 91国内免费在线视频| 国产又黄又大又爽| 91日韩精品一区| 男人天堂成人网| 久久野战av| 精品国产网站在线观看| 亚洲一级二级片| 亚洲一区国产| 91久久国产综合久久蜜月精品| 欧美在线观看在线观看| 亚洲自拍与偷拍| 国产精品视频中文字幕| 欧美一区二区三区红桃小说| 久热在线中文字幕色999舞| 五月天婷婷激情| 粉嫩欧美一区二区三区高清影视| 亚洲欧洲精品一区二区三区波多野1战4 | 黄色av一级片| 波多野洁衣一区| 7777在线视频| 亚洲高清影院| 亚洲欧洲一区二区三区久久| 国产在线拍揄自揄拍无码视频| 久久精品国产一区二区三| 久久久一本精品99久久精品66| 日本中文字幕中出在线| 4438x成人网最大色成网站| 国产一级久久久久毛片精品| 亚洲黄色大片| av一区和二区| 曰本三级在线| 91精品国产综合久久精品app | 不卡的av中文字幕| 沈樵精品国产成av片| 911国产网站尤物在线观看| 亚洲精品国偷拍自产在线观看蜜桃 | 麻豆久久精品| 精品国产乱码久久久久久丨区2区| 在线观看电影av| 日韩午夜av电影| 欧美日韩在线观看免费| 国产激情视频一区二区在线观看| 在线观看成人av电影| 久久69成人| 中文字幕在线精品| 一级黄色片免费看| 中文字幕一区三区| 亚洲午夜激情影院| 偷偷www综合久久久久久久| 成人xxxx视频| 黄网站视频在线观看| 制服.丝袜.亚洲.中文.综合| 五月天激情丁香| 国产精品一区在线| 特大黑人娇小亚洲女mp4| 嫩呦国产一区二区三区av| 久久成人精品一区二区三区| 精品国产999久久久免费| 亚洲午夜三级在线| 中国极品少妇videossexhd| 国产毛片久久| 日韩精品一区二区三区丰满 | 亚洲欧洲日本mm| 久久er99热精品一区二区三区| 丝袜老师在线| 永久免费毛片在线播放不卡| 91精品在线视频观看| 国产精品对白交换视频| a级大片免费看| 1024日韩| 欧美自拍资源在线| av在线亚洲一区| 久久免费视频这里只有精品| 亚洲欧洲精品视频| 欧美性一二三区| 男人操女人的视频网站| www.亚洲人| 欧美少妇性生活视频| 久久久久久久久99精品大| 亚洲伊人第一页| 国产无遮挡裸体视频在线观看| 亚洲欧美日韩视频一区| 国产精品高潮呻吟av| 精品成人乱色一区二区| 91视频免费在观看| 大美女一区二区三区| 免费观看日韩毛片| 一区二区日韩欧美| 欧美精品二区三区四区免费看视频 | 亚洲www视频| 碰碰在线视频| 日韩亚洲第一页| 欧美一区二区三区黄片| 欧美视频在线一区二区三区| 一级aaa毛片| 国产精品伦理一区二区| 精品人妻一区二区三区日产| 蜜桃av一区二区| 免费看欧美一级片| 欧美激情欧美| 久久精彩视频| 影音先锋欧美激情| 国产精品直播网红| 免费v片在线观看| 欧美乱大交xxxxx另类电影| 牛牛澡牛牛爽一区二区| 日韩免费成人网| 一二三四区视频| 狠狠色噜噜狠狠狠狠97| 18岁成人毛片| 中文字幕一区二区三区精华液| 青青草成人免费视频| 国产高清亚洲一区| 亚洲xxxx2d动漫1| 翔田千里一区二区| 老太脱裤让老头玩ⅹxxxx| 图片区亚洲欧美小说区| 午夜精品福利一区二区| 啪啪亚洲精品| 欧美日韩天天操| 国产伦理久久久久久妇女| 91日韩久久| 精品一区二区三区免费看| 国产精品成人国产乱一区| 自拍一区在线观看| 国内精品久久久久伊人av| 18在线观看的| 日韩亚洲一区二区| 午夜在线播放| 在线电影欧美日韩一区二区私密| 美女欧美视频在线观看免费 | 欧美日韩日日夜夜| 国产字幕在线观看| 欧美性生活大片免费观看网址| 国产一级片久久| 亚洲一本大道在线| 国产真实夫妇交换视频| 一区二区三区日韩精品视频| 欧美手机在线观看| 国产精品国产自产拍在线| 日本在线观看网址| 欧美高清在线精品一区| 国产破处视频在线观看| 国产精品免费av| 欧美xxxooo| 亚洲美腿欧美偷拍| 欧美日韩综合一区二区| 亚洲午夜在线视频| 久久高清免费视频| 天天色图综合网| 国产精品suv一区| 在线中文字幕一区二区| 久草热在线观看| 欧美日韩国产一二三| 一级黄色片在线| 日韩精品中午字幕| 男人天堂手机在线观看| 亚洲精品美女久久久久| 邻居大乳一区二区三区| 尤物99国产成人精品视频| 国产视频在线播放| 欧美高清电影在线看| 黄色在线网站噜噜噜| 国产成人精品国内自产拍免费看| 成人黄色图片网站| 91免费视频国产| 盗摄牛牛av影视一区二区| 久久免费一区| 色小子综合网| 日韩精品视频在线观看视频| 亚洲欧美春色| 污污网站免费观看| 国产99久久久国产精品免费看 | 国产婷婷一区二区三区久久| 精品三级在线观看| 男女av在线| 精品国产一区av| 福利成人导航| 国产999在线观看| 国产不卡精品| 精品午夜一区二区| 日韩综合精品| 日本午夜激情视频| 久久国产生活片100| 国产51自产区| 国产精品私人影院| 亚洲精品在线观看av| 欧洲一区二区三区免费视频| 99久久久国产精品无码免费| 亚洲精品在线观看www| 日本高清视频在线播放| 91国内免费在线视频| 99精品国产九九国产精品| 精品国产二区在线| 综合视频在线| 另类小说第一页| 懂色av一区二区三区免费看| 摸摸摸bbb毛毛毛片| 亚洲图片欧美综合| 国产伦一区二区| 亚洲美女动态图120秒| 肉体视频在线| 国产精品日本精品| 亚洲精品亚洲人成在线观看| 好吊色视频988gao在线观看| 日本不卡视频在线| 久久人人妻人人人人妻性色av| 亚洲视频狠狠干| 国产情侣小视频| 日韩精品极品在线观看播放免费视频| 免费在线看黄色| 国产v综合ⅴ日韩v欧美大片| 77成人影视| 伊人网在线免费| 久久精品国产99久久6| 国产精品扒开腿做爽爽| 亚洲成人av电影在线| 国产男女裸体做爰爽爽| 在线精品播放av| 日韩高清不卡| 精品国产免费久久久久久尖叫| 国产一区亚洲| 中文字幕一区二区在线观看视频| 国产欧美视频在线观看| 4438国产精品一区二区| 精品福利在线导航| 手机av免费在线| 91最新在线免费观看| 国产韩国精品一区二区三区| 黄色三级视频在线| 久久久久一区二区三区四区| 日韩欧美不卡视频| 精品美女被调教视频大全网站| 麻豆最新免费在线视频| 国产在线高清精品| 青青草国产成人a∨下载安卓| 已婚少妇美妙人妻系列| 久久综合九色综合97婷婷| 国产精品美女久久久久av爽| 日韩一区二区三区电影在线观看| 在线观看a级片| julia一区二区中文久久94| 欧美片第1页综合| ass极品水嫩小美女ass| 一区二区三区小说| 亚洲欧美另类综合| 欧美精品aaa| 波多野结衣在线一区二区| 国产在线播放观看| 99视频一区二区三区| 天天做天天爱夜夜爽| 亚洲理论在线a中文字幕| 中文在线а√天堂| 欧美日韩精品久久久免费观看| 视频在线观看一区| 欧美巨胸大乳hitomi| 欧美肥妇毛茸茸| 呦呦在线视频| 国产综合av一区二区三区| 香蕉视频成人在线观看| 欧美大波大乳巨大乳| 精品视频一区三区九区| 国产在线激情视频| ts人妖另类在线| 亚洲免费影院| 国产三级黄色片| 日韩午夜精品电影| 一区二区乱码| 亚洲精品国产精品国自产| 国产综合成人久久大片91| 久久久久成人网站| 亚洲人成在线播放| 欧美大片网站| 拔插拔插海外华人免费| 久久一二三国产| 国产视频aaa| 91成人在线播放| 久久国产成人精品| 无码人妻精品一区二区三| 欧美午夜激情在线| 蜜桃视频在线观看www社区 | 91a在线视频| 91欧美国产| 99久久人妻无码中文字幕系列| 91黄色激情网站| 欧美人体视频xxxxx| 日本不卡在线观看| 国产91丝袜在线播放九色| 免费的毛片视频| 欧美国产精品人人做人人爱| 国产探花一区在线观看| 日韩不卡的av| 欧美午夜精品久久久| 91超碰免费在线| 91免费网站视频| 欧美激情一区三区|