不能。原生shape的corners只支持统一半径或全部四角显式声明(如topLeftRadius、topRightRadius等),不支持仅设部分角;android:radius会覆盖单角属性,且单位必须为dp,API 21+也无roundedCorner属性。

shape 中 corners 无法单独设左上/右下等单角半径?
不能。原生 shape 的 corners 标签只支持统一半径(android:radius)或全部四角分别设置(android:topLeftRadius 等),但**不支持“仅设左上和右下,其他为 0”这种混合逻辑**——XML 解析器会静默忽略未声明的属性,导致你以为设了却没生效。
常见错误现象:android:topLeftRadius="8dp" 单独写,其余没写,结果四个角全圆;或者写了 android:radius="0dp" 又额外写某个角,发现那个角也没变圆——因为 android:radius 优先级高于单角属性,一设就覆盖全部。
- 必须同时声明全部四个角,哪怕有些是
0dp,例如:android:topLeftRadius="12dp"、android:topRightRadius="0dp"、android:bottomLeftRadius="0dp"、android:bottomRightRadius="8dp" - 删掉
android:radius这一行,它和单角属性互斥 - 单位必须是
dp(不能用px或无单位),否则在高密度屏上会缩成针尖大小
API 21+ 能用 roundedCorner 吗?
不能。没有这个属性名。Android 原生 shape 从 API 1 到 API 34 都只有 corners 子标签,不存在 roundedCorner、cornerRadii 或数组式写法。网上搜到的所谓“新写法”,基本是混淆了 GradientDrawable 的 Java 构造参数,或误把 Compose 的 RoundedCornerShape 当成了 XML 特性。
如果你真需要动态控制单角,只能放弃 XML,改用代码创建 GradientDrawable 并调用 setCornerRadii(float[]),传入长度为 8 的数组(每对 xy 表示一个角)。
- XML 就是静态的,别指望它有“条件圆角”或“奇数角生效”这种能力
- 第三方库如
MaterialShapeDrawable支持更细粒度控制,但它依赖 Material Components,且仍需 Java/Kotlin 配合,不是纯 XML 方案
为什么设置了 topLeftRadius 却没看到圆角?
大概率是父容器或自身尺寸没撑开。XML shape 是绘制层,不是布局层——如果 View 实际宽高为 wrap_content 且内容为空,它可能只有 0×0,圆角自然不可见;或者父布局加了 clipChildren="true",把超出部分裁掉了。
另一个隐蔽原因是:你用了 layer-list 套多层 drawable,但最外层没设 android:gravity,导致圆角层被居中缩放,视觉上“消失”了。
- 给 View 加个背景色(比如
android:background="#f00")确认它是否真有尺寸 - 检查父布局是否设置了
android:clipChildren="true"(默认是 true) - 如果是
layer-list,确保每一层都明确写了android:gravity="fill",否则圆角层可能被压缩到中心一个小点
替代方案:用 MaterialCardView 真的更省事?
是,但代价是引入整个 Material 组件库,并失去对底层 shape 的直接控制。它的 app:cardCornerRadius 是全局的,而单角控制得靠 ShapeAppearanceModel + 自定义 MaterialShapeDrawable,最终还是要写代码。
如果你只是想让一个按钮左上右下圆、右上左下直,又不想写 Java,最轻量的解法其实是:用两个重叠的 shape(一个画主体矩形,一个用 stroke 或 solid 覆盖掉不需要圆角的边),但这种方式维护成本高,且阴影、点击热区会出问题。
所以真实情况是:XML 圆角本就不该承担“差异化设计”的任务。该用代码的地方,就别硬套 XML。










