Java如何实现长图文生成的示例代码

很久很久以前,就觉得微博的长图文实现得非常有意思,将排版直接以最终的图片输出,收藏查看分享都很方便,现在则自己动手实现一个简单版本的

目标

首先定义下我们预期达到的目标:根据文字 + 图片生成长图文

目标拆解

  • 支持大段文字生成图片
  • 支持插入图片
  • 支持上下左右边距设置
  • 支持字体选择
  • 支持字体颜色
  • 支持左对齐,居中,右对齐

预期结果

我们将通过spring-boot搭建一个生成长图文的http接口,通过传入参数来指定各种配置信息,下面是一个最终调用的示意图

设计&实现

长图文的生成,采用awt进行文字绘制和图片绘制

1. 参数选项 ImgCreateOptions

根据我们的预期目标,设定配置参数,基本上会包含以下参数

@Getter
@Setter
@ToString
public class ImgCreateOptions { 

  /**
   * 绘制的背景图
   */
  private BufferedImage bgImg; 

  /**
   * 生成图片的宽
   */
  private Integer imgW; 

  private Font font = new Font("宋体", Font.PLAIN, 18); 

  /**
   * 字体色
   */
  private Color fontColor = Color.BLACK; 

  /**
   * 两边边距
   */
  private int leftPadding; 

  /**
   * 上边距
   */
  private int topPadding; 

  /**
   * 底边距
   */
  private int bottomPadding; 

  /**
   * 行距
   */
  private int linePadding; 

  private AlignStyle alignStyle; 

  /**
   * 对齐方式
   */
  public enum AlignStyle {
    LEFT,
    CENTER,
    RIGHT; 

    private static Map<String, AlignStyle> map = new HashMap<>(); 

    static {
      for(AlignStyle style: AlignStyle.values()) {
        map.put(style.name(), style);
      }
    } 

    public static AlignStyle getStyle(String name) {
      name = name.toUpperCase();
      if (map.containsKey(name)) {
        return map.get(name);
      } 

      return LEFT;
    }
  }
}

2. 封装类 ImageCreateWrapper

封装配置参数的设置,绘制文本,绘制图片的操作方式,输出样式等接口

public class ImgCreateWrapper { 

  public static Builder build() {
    return new Builder();
  } 

  public static class Builder {
    /**
     * 生成的图片创建参数
     */
    private ImgCreateOptions options = new ImgCreateOptions(); 

    /**
     * 输出的结果
     */
    private BufferedImage result; 

    private final int addH = 1000; 

    /**
     * 实际填充的内容高度
     */
    private int contentH; 

    private Color bgColor; 

    public Builder setBgColor(int color) {
      return setBgColor(ColorUtil.int2color(color));
    } 

    /**
     * 设置背景图
     *
     * @param bgColor
     * @return
     */
    public Builder setBgColor(Color bgColor) {
      this.bgColor = bgColor;
      return this;
    } 

    public Builder setBgImg(BufferedImage bgImg) {
      options.setBgImg(bgImg);
      return this;
    } 

    public Builder setImgW(int w) {
      options.setImgW(w);
      return this;
    } 

    public Builder setFont(Font font) {
      options.setFont(font);
      return this;
    } 

    public Builder setFontName(String fontName) {
      Font font = options.getFont();
      options.setFont(new Font(fontName, font.getStyle(), font.getSize()));
      return this;
    } 

    public Builder setFontColor(int fontColor) {
      return setFontColor(ColorUtil.int2color(fontColor));
    } 

    public Builder setFontColor(Color fontColor) {
      options.setFontColor(fontColor);
      return this;
    } 

    public Builder setFontSize(Integer fontSize) {
      Font font = options.getFont();
      options.setFont(new Font(font.getName(), font.getStyle(), fontSize));
      return this;
    } 

    public Builder setLeftPadding(int leftPadding) {
      options.setLeftPadding(leftPadding);
      return this;
    } 

    public Builder setTopPadding(int topPadding) {
      options.setTopPadding(topPadding);
      contentH = topPadding;
      return this;
    } 

    public Builder setBottomPadding(int bottomPadding) {
      options.setBottomPadding(bottomPadding);
      return this;
    } 

    public Builder setLinePadding(int linePadding) {
      options.setLinePadding(linePadding);
      return this;
    } 

    public Builder setAlignStyle(String style) {
      return setAlignStyle(ImgCreateOptions.AlignStyle.getStyle(style));
    } 

    public Builder setAlignStyle(ImgCreateOptions.AlignStyle alignStyle) {
      options.setAlignStyle(alignStyle);
      return this;
    } 

    public Builder drawContent(String content) {
      // xxx
      return this;
    } 

    public Builder drawImage(String img) {
      BufferedImage bfImg;
      try {
         bfImg = ImageUtil.getImageByPath(img);
      } catch (IOException e) {
        log.error("load draw img error! img: {}, e:{}", img, e);
        throw new IllegalStateException("load draw img error! img: " + img, e);
      } 

      return drawImage(bfImg);
    } 

    public Builder drawImage(BufferedImage bufferedImage) { 

      // xxx
      return this;
    } 

    public BufferedImage asImage() {
      int realH = contentH + options.getBottomPadding(); 

      BufferedImage bf = new BufferedImage(options.getImgW(), realH, BufferedImage.TYPE_INT_ARGB);
      Graphics2D g2d = bf.createGraphics(); 

      if (options.getBgImg() == null) {
        g2d.setColor(bgColor == null ? Color.WHITE : bgColor);
        g2d.fillRect(0, 0, options.getImgW(), realH);
      } else {
        g2d.drawImage(options.getBgImg(), 0, 0, options.getImgW(), realH, null);
      } 

      g2d.drawImage(result, 0, 0, null);
      g2d.dispose();
      return bf;
    } 

    public String asString() throws IOException {
      BufferedImage img = asImage();
      return Base64Util.encode(img, "png");
    }
}

上面具体的文本和图片绘制实现没有,后面详细讲解,这里主要关注的是一个参数 contentH, 表示实际绘制的内容高度(包括上边距),因此最终生成图片的高度应该是

int realH = contentH + options.getBottomPadding();

其次简单说一下上面的图片输出方法:com.hust.hui.quickmedia.common.image.ImgCreateWrapper.Builder#asImage

  • 计算最终生成图片的高度(宽度由输入参数指定)
  • 绘制背景(如果没有背景图片,则用纯色填充)
  • 绘制实体内容(即绘制的文本,图片)

3. 内容填充 GraphicUtil

具体的内容填充,区分为文本绘制和图片绘制

设计

考虑到在填充的过程中,可以自由设置字体,颜色等,所以在我们的绘制方法中,直接实现掉内容的绘制填充,即 drawXXX 方法真正的实现了内容填充,执行完之后,内容已经填充到画布上了

图片绘制,考虑到图片本身大小和最终结果的大小可能有冲突,采用下面的规则

  • 绘制图片宽度 <=(指定生成图片宽 - 边距),全部填充
  • 绘制图片宽度 >(指定生成图片宽 - 边距),等比例缩放绘制图片

文本绘制,换行的问题

  • 每一行允许的文本长度有限,超过时,需要自动换行处理

文本绘制

考虑基本的文本绘制,流程如下

1、创建BufferImage对象

2、获取Graphic2d对象,操作绘制

3、设置基本配置信息

4、文本按换行进行拆分为字符串数组, 循环绘制单行内容

  • 计算当行字符串,实际绘制的行数,然后进行拆分
  • 依次绘制文本(需要注意y坐标的变化)

下面是具体的实现

public static int drawContent(Graphics2D g2d,
                 String content,
                 int y,
                 ImgCreateOptions options) { 

  int w = options.getImgW();
  int leftPadding = options.getLeftPadding();
  int linePadding = options.getLinePadding();
  Font font = options.getFont(); 

  // 一行容纳的字符个数
  int lineNum = (int) Math.floor((w - (leftPadding << 1)) / (double) font.getSize()); 

  // 对长串字符串进行分割成多行进行绘制
  String[] strs = splitStr(content, lineNum); 

  g2d.setFont(font); 

  g2d.setColor(options.getFontColor());
  int index = 0;
  int x;
  for (String tmp : strs) {
    x = calOffsetX(leftPadding, w, tmp.length() * font.getSize(), options.getAlignStyle());
    g2d.drawString(tmp, x, y + (linePadding + font.getSize()) * index);
    index++;
  } 

  return y + (linePadding + font.getSize()) * (index);
} 

/**
 * 计算不同对其方式时,对应的x坐标
 *
 * @param padding 左右边距
 * @param width  图片总宽
 * @param strSize 字符串总长
 * @param style  对其方式
 * @return 返回计算后的x坐标
 */
private static int calOffsetX(int padding,
               int width,
               int strSize,
               ImgCreateOptions.AlignStyle style) {
  if (style == ImgCreateOptions.AlignStyle.LEFT) {
    return padding;
  } else if (style == ImgCreateOptions.AlignStyle.RIGHT) {
    return width - padding - strSize;
  } else {
    return (width - strSize) >> 1;
  }
} 

/**
 * 按照长度对字符串进行分割
 * <p>
 * fixme 包含emoj表情时,兼容一把
 *
 * @param str   原始字符串
 * @param splitLen 分割的长度
 * @return
 */
public static String[] splitStr(String str, int splitLen) {
  int len = str.length();
  int size = (int) Math.ceil(len / (float) splitLen); 

  String[] ans = new String[size];
  int start = 0;
  int end = splitLen;
  for (int i = 0; i < size; i++) {
    ans[i] = str.substring(start, end > len ? len : end);
    start = end;
    end += splitLen;
  } 

  return ans;
}

上面的实现比较清晰了,图片的绘制则更加简单

图片绘制

只需要重新计算下待绘制图片的宽高即可,具体实现如下

/**
 * 在原图上绘制图片
 *
 * @param source 原图
 * @param dest  待绘制图片
 * @param y    待绘制的y坐标
 * @param options
 * @return 绘制图片的高度
 */
public static int drawImage(BufferedImage source,
              BufferedImage dest,
              int y,
              ImgCreateOptions options) {
  Graphics2D g2d = getG2d(source);
  int w = Math.min(dest.getWidth(), options.getImgW() - (options.getLeftPadding() << 1));
  int h = w * dest.getHeight() / dest.getWidth(); 

  int x = calOffsetX(options.getLeftPadding(),
      options.getImgW(), w, options.getAlignStyle()); 

  // 绘制图片
  g2d.drawImage(dest,
      x,
      y + options.getLinePadding(),
      w,
      h,
      null);
  g2d.dispose(); 

  return h;
} 

public static Graphics2D getG2d(BufferedImage bf) {
    Graphics2D g2d = bf.createGraphics(); 

  g2d.setRenderingHint(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY);
  g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  g2d.setRenderingHint(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_QUALITY);
  g2d.setRenderingHint(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE);
  g2d.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
  g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);
  g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
  g2d.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE); 

  return g2d;
}

4. 内容渲染

前面只是给出了单块内容(如一段文字,一张图片)的渲染,存在一些问题

  • 绘制的内容超过画布的高度如何处理
  • 文本绘制要求传入的文本没有换行符,否则换行不生效
  • 交叉绘制的场景,如何重新计算y坐标

解决这些问题则是在 ImgCreateWrapper 的具体绘制中进行了实现,先看文本的绘制

根据换行符对字符串进行拆分

计算绘制内容最终转换为图片时,所占用的高度

重新生成画布 BufferedImage result

  • 如果result为空,则直接生成
  • 如果最终生成的高度,超过已有画布的高度,则生成一个更高的画布,并将原来的内容绘制上去

迭代绘制单行内容

public Builder drawContent(String content) {
  String[] strs = StringUtils.split(content, "\n");
  if (strs.length == 0) { // empty line
    strs = new String[1];
    strs[0] = " ";
  } 

  int fontSize = options.getFont().getSize();
  int lineNum = calLineNum(strs, options.getImgW(), options.getLeftPadding(), fontSize);
  // 填写内容需要占用的高度
  int height = lineNum * (fontSize + options.getLinePadding()); 

  if (result == null) {
    result = GraphicUtil.createImg(options.getImgW(),
        Math.max(height + options.getTopPadding() + options.getBottomPadding(), BASE_ADD_H),
        null);
  } else if (result.getHeight() < contentH + height + options.getBottomPadding()) {
    // 超过原来图片高度的上限, 则需要扩充图片长度
    result = GraphicUtil.createImg(options.getImgW(),
        result.getHeight() + Math.max(height + options.getBottomPadding(), BASE_ADD_H),
        result);
  } 

  // 绘制文字
  Graphics2D g2d = GraphicUtil.getG2d(result);
  int index = 0;
  for (String str : strs) {
    GraphicUtil.drawContent(g2d, str,
        contentH + (fontSize + options.getLinePadding()) * (++index)
        , options);
  }
  g2d.dispose(); 

  contentH += height;
  return this;
} 

/**
 * 计算总行数
 *
 * @param strs   字符串列表
 * @param w    生成图片的宽
 * @param padding 渲染内容的左右边距
 * @param fontSize 字体大小
 * @return
 */
private int calLineNum(String[] strs, int w, int padding, int fontSize) {
  // 每行的字符数
  double lineFontLen = Math.floor((w - (padding << 1)) / (double) fontSize); 

  int totalLine = 0;
  for (String str : strs) {
    totalLine += Math.ceil(str.length() / lineFontLen);
  } 

  return totalLine;
}

上面需要注意的是画布的生成规则,特别是高度超过上限之后,重新计算图片高度时,需要额外注意新增的高度,应该为基本的增量与(绘制内容高度+下边距)的较大值

代码如下:

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

重新生成画布实现 com.hust.hui.quickmedia.common.util.GraphicUtil#createImg

public static BufferedImage createImg(int w, int h, BufferedImage img) {
  BufferedImage bf = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB);
  Graphics2D g2d = bf.createGraphics(); 

  if (img != null) {
    g2d.setComposite(AlphaComposite.Src);
    g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
    g2d.drawImage(img, 0, 0, null);
  }
  g2d.dispose();
  return bf;
}

上面理解之后,绘制图片就比较简单了,基本上行没什么差别

public Builder drawImage(String img) {
  BufferedImage bfImg;
  try {
    bfImg = ImageUtil.getImageByPath(img);
  } catch (IOException e) {
    log.error("load draw img error! img: {}, e:{}", img, e);
    throw new IllegalStateException("load draw img error! img: " + img, e);
  } 

  return drawImage(bfImg);
} 

public Builder drawImage(BufferedImage bufferedImage) { 

  if (result == null) {
    result = GraphicUtil.createImg(options.getImgW(),
        Math.max(bufferedImage.getHeight() + options.getBottomPadding() + options.getTopPadding(), BASE_ADD_H),
        null);
  } else if (result.getHeight() < contentH + bufferedImage.getHeight() + options.getBottomPadding()) {
    // 超过阀值
    result = GraphicUtil.createImg(options.getImgW(),
        result.getHeight() + Math.max(bufferedImage.getHeight() + options.getBottomPadding() + options.getTopPadding(), BASE_ADD_H),
        result);
  } 

  // 更新实际高度
  int h = GraphicUtil.drawImage(result,
      bufferedImage,
      contentH,
      options);
  contentH += h + options.getLinePadding();
  return this;
}

5. http接口

上面实现的生成图片的公共方法,在 quick-media 工程中,利用spring-boot搭建了一个web服务,提供了一个http接口,用于生成长图文,最终的成果就是我们开头的那个gif图的效果,相关代码就没啥好说的,有兴趣的可以直接查看工程源码,链接看最后

测试验证

上面基本上完成了我们预期的目标,接下来则是进行验证,测试代码比较简单,先准备一段文本,这里拉了一首诗

招魂酹翁宾旸

郑起

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

魂之为变性原返,气之为物情本开。

於戏龙兮凤兮神气盛,噫嘻鬼兮归兮大块埃。

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

采石捉月李白非醉,耒阳避水子美非灾。

长孙王吉命不夭,玉川老子诗不徘。

新城罗隐在奇特,钱塘潘阆终崔嵬。

阴兮魄兮曷往,阳兮魄兮曷来。

君其归来,故交寥落更散漫。

君来归来,帝城绚烂可徘徊。

君其归来,东西南北不可去。

君其归来。

春秋霜露令人哀。

花之明吾无与笑,叶之陨吾实若摧。

晓猿啸吾闻泪堕,宵鹤立吾见心猜。

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

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

君其归来,可伴逋仙之梅,去此又奚之哉。

测试代码

@Test
public void testGenImg() throws IOException {
  int w = 400;
  int leftPadding = 10;
  int topPadding = 40;
  int bottomPadding = 40;
  int linePadding = 10;
  Font font = new Font("宋体", Font.PLAIN, 18); 

  ImgCreateWrapper.Builder build = ImgCreateWrapper.build()
      .setImgW(w)
      .setLeftPadding(leftPadding)
      .setTopPadding(topPadding)
      .setBottomPadding(bottomPadding)
      .setLinePadding(linePadding)
      .setFont(font)
      .setAlignStyle(ImgCreateOptions.AlignStyle.CENTER)
//        .setBgImg(ImageUtil.getImageByPath("qrbg.jpg"))
      .setBgColor(0xFFF7EED6)
      ; 

  BufferedReader reader = FileReadUtil.createLineRead("text/poem.txt");
  String line;
  int index = 0;
  while ((line = reader.readLine()) != null) {
    build.drawContent(line); 

    if (++index == 5) {
      build.drawImage(ImageUtil.getImageByPath("https://static.oschina.net/uploads/img/201708/12175633_sOfz.png"));
    } 

    if (index == 7) {
      build.setFontSize(25);
    } 

    if (index == 10) {
      build.setFontSize(20);
      build.setFontColor(Color.RED);
    }
  } 

  BufferedImage img = build.asImage();
  String out = Base64Util.encode(img, "png");
  System.out.println("<img src=\"data:image/png;base64," + out + "\" />");
}

输出图片

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Java如何实现长图文生成的示例代码

    很久很久以前,就觉得微博的长图文实现得非常有意思,将排版直接以最终的图片输出,收藏查看分享都很方便,现在则自己动手实现一个简单版本的 目标 首先定义下我们预期达到的目标:根据文字 + 图片生成长图文 目标拆解 支持大段文字生成图片 支持插入图片 支持上下左右边距设置 支持字体选择 支持字体颜色 支持左对齐,居中,右对齐 预期结果 我们将通过spring-boot搭建一个生成长图文的http接口,通过传入参数来指定各种配置信息,下面是一个最终调用的示意图 设计&实现 长图文的生成,采用awt进行文

  • Java实现一个简单的长轮询的示例代码

    目录 分析一下长轮询的实现方式 长轮询与短轮询 配置中心长轮询设计 配置中心长轮询实现 客户端实现 服务端实现 分析一下长轮询的实现方式 现在各大中间件都使用了长轮询的数据交互方式,目前比较流行的例如Nacos的配置中心,RocketMQ Pull(拉模式)消息等,它们都是采用了长轮询方的式实现.就例如Nacos的配置中心,如何做到服务端感知配置变化实时推送给客户端的呢? 长轮询与短轮询 说到长轮询,肯定存在和它相对立的,我们暂且叫它短轮询吧,我们简单介绍一下短轮询: 短轮询也是拉模式.是指不管

  • Java调用微信支付功能的方法示例代码

    Java 使用微信支付 前言百度搜了一下微信支付,都描述的不太好,于是乎打算自己写一个案例,希望以后拿来直接改造使用. 因为涉及二维码的前端显示,所以有前端的内容 一. 准备工作 所需微信公众号信息配置 APPID:绑定支付的APPID(必须配置) MCHID:商户号(必须配置) KEY:商户支付密钥,参考开户邮件设置(必须配置) APPSECRET:公众帐号secert(仅JSAPI支付的时候需要配置) 我这个案例用的是尚硅谷一位老师提供的,这里不方便提供出来,需要大家自己找,或者公司提供 二

  • Java实现经典游戏黄金矿工的示例代码

    目录 前言 主要设计 功能截图 代码实现 游戏核心类 钩子类 总结 前言 <黄金矿工>游戏是一个经典的抓金子小游戏,它可以锻炼人的反应能力..该游戏中,可以通过“挖矿”获得积分,游戏道具:有3瓶药水,在回收绳子抓金子的时候速度稍快点. 主要设计 设计游戏界面,用swing实现 随机创建金块算法 随机创建石块算法 计时系统设计 积分系统设计 设置鼠标事件,鼠标左键出钩子:鼠标右键开始游戏,确认吃药水等功能. 功能截图 游戏开始: 抓金子 代码实现 游戏核心类 public class GameW

  • Java实现鼠标随机移动效果的示例代码

    目录 前言 实现代码 效果图 前言 疫情,需要远程办公,为了更好的远程办(划)公(水).而我们公司因为没有想到会有大批量的远程办公,从而导致连接的人过多,需要抢占连接才能登录,而且好不容易抢到了,去上个厕所,然后就长时间未操作断开了,防止这种事情的发生,特地写了这个脚本 实现代码 import javax.swing.*; import java.awt.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListe

  • Java实现无损Word转PDF的示例代码

    目录 前言 word转pdf实现思路 项目远程仓库 Maven项目pom文件依赖 核心代码实现 结果分析 前言 本来想写word转pdf和pdf转word的代码呢,没想到word转pdf就写了很多很多行代码才实现,为了方便大家消化理解,先写了word转pdf方法实现作为一篇文章. word转pdf实现思路 代码实现主要依赖两个第三方jar包,一个是pdfbox,一个是aspose-words.pdfbox包完全开源免费,aspose-words免费版生成有水印,且生成数量有限制.单纯用pdfbo

  • Java实现手写线程池的示例代码

    目录 前言 线程池给我们提供的功能 工具介绍 Worker设计 线程池设计 总结 前言 在我们的日常的编程当中,并发是始终离不开的主题,而在并发多线程当中,线程池又是一个不可规避的问题.多线程可以提高我们并发程序的效率,可以让我们不去频繁的申请和释放线程,这是一个很大的花销,而在线程池当中就不需要去频繁的申请线程,他的主要原理是申请完线程之后并不中断,而是不断的去队列当中领取任务,然后执行,反复这样的操作.在本篇文章当中我们主要是介绍线程池的原理,因此我们会自己写一个非常非常简单的线程池,主要帮

  • SpringBoot整合screw实现数据库文档自动生成的示例代码

    有时候数据库文档需要整理,可是只能手动的复制粘贴,心中一万只草泥马奔腾而过... screw 简洁好用的数据库表结构文档生成工具. 1. 创建项目 1.1 pom.xml <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency>

  • tk.mybatis实现uuid主键生成的示例代码

    引入依赖 <dependency> <groupId>tk.mybatis</groupId> <artifactId>mapper-spring-boot-starter</artifactId> <version>2.0.2</version> </dependency> 1.创建一个GenId的实现类 package com.xiaobu.base.entity; import tk.mybatis.ma

  • Java基于LoadingCache实现本地缓存的示例代码

    目录 一. 添加 maven 依赖 二.CacheBuilder 方法说明 三.创建 CacheLoader 四.工具类 五.guava Cache数据移除 一. 添加 maven 依赖 <dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId> <version>27.1-jre</version> </depend

随机推荐