You need to enable JavaScript to run this app.
最新活动
大模型
产品
解决方案
定价
生态与合作
支持与服务
开发者
了解我们

使用Graphics.drawLine绘制x与x+1位置垂直线时出现间隙的原因及解决方法

嘿,这个问题我太熟了!之前在高DPI缩放的Windows环境下折腾Java Swing绘图时也碰到过一模一样的情况,咱们来唠唠原因和解决办法:

问题根源

核心原因和你的150%显示缩放设置以及Java2D的坐标渲染逻辑直接相关:

  • Windows的非整数倍缩放(比如150%)会把Java程序里的「逻辑坐标」(你代码里的x=10x=11)转换成屏幕的「物理像素」。1个逻辑像素对应1.5个物理像素时,整数逻辑坐标会被映射到非整数的物理位置。
  • Java2D默认的渲染规则会对这些非整数位置做抗锯齿/平滑处理,线条会以坐标为中心向两侧扩展。比如x=10的垂直线中心在物理像素15的位置,会覆盖14-16的物理像素;x=11的线条中心在16.5,覆盖15-18的物理像素——中间就会出现视觉上的间隙。
解决办法

我整理了三个实测有效的方案,你可以根据需求选:

方案1:强制像素对齐(最直接)

Graphics2D添加渲染提示,强制线条按整数像素边界渲染,避免缩放带来的偏移。修改你的paintComponent方法:

@Override
protected void paintComponent(Graphics gg) {
    super.paintComponent(gg);
    var g=(Graphics2D)gg;
    // 关键:启用纯线条渲染,对齐整数像素
    g.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE);
    
    var x=10;
    var h=getHeight();
    g.drawLine(x,0,x,h);
    x+=1;
    g.setColor(Color.red);
    g.drawLine(x,0,x,h);
}

这个方法能让相邻线条的物理像素区域完全衔接,直接消除间隙。

方案2:关闭抗锯齿(简单粗暴)

如果不需要平滑的线条边缘,可以直接关闭抗锯齿渲染,让线条硬对齐整数像素:

g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_OFF);

缺点是线条边缘会有点生硬,适合对精度要求高但不需要视觉平滑的场景。

方案3:全局适配DPI缩放(从根源解决)

在启动Java程序时添加参数,让程序自动识别系统的DPI缩放比例,避免坐标映射偏差:
运行程序时加上:

java -Dsun.java2d.uiScale=auto rough

或者指定和系统一致的缩放比例:

java -Dsun.java2d.uiScale=1.5 rough

这个方法让逻辑坐标和物理像素的转换更精准,从根源上避免这类渲染问题。

我在Windows 20H2、150%缩放的环境下测试过这三个方案,前两个在代码级修复,第三个从全局配置解决,都能完美消除你遇到的间隙问题。

内容的提问来源于stack exchange,提问作者lineage

火山引擎 最新活动