shape drawable 未声明 size 时宽高为 -1,无固有尺寸,导致 imageview 等依赖 intrinsic size 的组件不渲染或异常;必须显式用 设置 dp 或 px 值。

shape drawable 没设 size 时宽高是多少?
它没有默认宽高——size 标签完全不写,就等于没声明尺寸约束。系统不会给它填 0dp 或 wrap_content 这类“默认值”,而是让父容器或使用处决定最终大小;但多数情况下(比如用在 android:background),会触发拉伸、模糊或渲染异常。
常见错误现象:shape 作为 Button 背景时看起来被横向撑满、圆角失效、渐变错位;或者用在 ImageView 的 src 里直接不显示。
-
size不写 ≠ 尺寸为 0,而是“未定义”,行为取决于宿主控件的测量逻辑 - 只有显式写了
<size android:width="24dp" android:height="24dp"></size>,才能确保该 Drawable 有稳定固有尺寸 - 如果用在
android:src(非 background),没size几乎必然不渲染——因为ImageView默认按 intrinsic size 测量,而无size的 shape intrinsic size 是 -1
为什么 size 必须写在 shape 根标签里,不能靠父 layout 控制?
因为 shape 是一个 Drawable,不是 View;它的尺寸语义属于资源自身,和 layout XML 的 layout_width 无关。父 layout 只能影响“怎么展示它”,不能赋予它“本来多大”。
使用场景举例:你把同一个 @drawable/rounded_bg 同时用在 TextView 的 background 和 ImageView 的 src 上——前者可能靠 padding 勉强撑开,后者大概率空白,除非你在 shape 里写了 size。
-
size是ShapeDrawable自身的 intrinsic size,影响getIntrinsicWidth()/getIntrinsicHeight()返回值 - 省略
size后,这两个方法都返回 -1,很多系统组件(如ImageView)遇到 -1 就跳过绘制 - 即使加了
android:layout_width="50dp",也只控制 ImageView 容器大小,不改变 Drawable 内部的尺寸认知
size 的单位只能用 dp 吗?px、sp、wrap_content 行不行?
单位只能是 dp 或 px;sp 不支持(字体缩放不适用于 shape),wrap_content 和 match_parent 更是语法错误——XML 解析直接失败,报错:Binary XML file line #X: Invalid value for android:width。
性能影响很小,但兼容性要注意:低版本 Android(API dp 解析更严格,若用 px 在高密度屏上会显得特别小。
- 推荐统一用
dp,例如android:width="32dp",和设计稿对齐最稳 - 避免用
px,除非你明确知道目标设备像素密度且不做适配 - 别尝试写
android:width="wrap_content"——编译不过,IDE 会红标,运行时报Resources$NotFoundException
用 inset 或 layer-list 包一层能替代 size 吗?
不能。包裹操作只是新增一层 Drawable 容器,底层 shape 本身 still has no intrinsic size。整个 layer-list 的 intrinsic size 依然由子项决定;如果所有子项都没 size,结果还是 -1。
容易踩的坑:有人以为给 layer-list 加 android:padding 或套个 inset 就能“撑开” shape,其实只是视觉上偏移或留白,Drawable 本身的测量值没变,ImageView 仍不画。
-
inset的android:insetLeft等属性只影响绘制区域偏移,不影响 intrinsic size -
layer-list中任意一项有size,整体会取最大那个 intrinsic size;但如果你只靠shape+inset,依然无效 - 真要动态控制尺寸,得用代码:创建
ShapeDrawable实例后调用setIntrinsicWidth(),但 XML 里做不到
最常被忽略的一点:设计师给的切图尺寸(比如 24×24 px)不能直接当 size 值用——必须换算成 dp,并确认是否对应基准密度(mdpi)。否则在不同设备上,shape 的实际物理尺寸会漂移。这不是 bug,是 density-aware 的正常表现。










