登录
首页 >  文章 >  java教程

JavaCollections.swap使用技巧分享

时间:2026-03-23 10:06:48 331浏览 收藏

Java的Collections.swap看似简单,实则暗藏多重陷阱:它无法用于Arrays.asList返回的固定大小列表、不校验索引合法性、在泛型擦除下易引发类型安全问题,且与不可变列表或高开销并发列表(如CopyOnWriteArrayList)兼容性差;真正关键的不是“怎么调用swap”,而是先确认你的List是否真正支持set操作——多数bug根源在于上游构造方式不当,而非swap本身写错。

Java里的Collections.swap交换列表中两个元素的位置_索引操作技巧

为什么 Collections.swap 不能交换 Arrays.asList 返回的列表?

因为 Arrays.asList 返回的是一个固定大小的、不支持结构性修改的列表实现(Arrays$ArrayList),而 Collections.swap 内部会尝试调用 set 方法——这在该实现中会抛出 UnsupportedOperationException

  • 常见错误现象:Exception in thread "main" java.lang.UnsupportedOperationException
  • 使用场景:你用 String[] arr = {"a","b","c"}; 转成 List 后直接 swap,就踩坑了
  • 解决办法:包装一层可变列表,比如 new ArrayList(Arrays.asList(...))
  • 性能影响:多一次构造,但只在初始化时发生,无运行时开销

Collections.swap 的索引越界行为和安全边界

它不会做任何索引检查,传入负数或超出 list.size() 的索引,会在内部调用 getset 时直接抛出 IndexOutOfBoundsException —— 和你手动 get/set 一样。

  • 容易踩的坑:误以为 swap 会自动校验或静默忽略非法索引
  • 建议做法:调用前自己判断 i >= 0 && i = 0 && j
  • 参数差异:两个索引顺序无关,swap(list, 0, 2)swap(list, 2, 0) 效果完全相同

替代方案:手写 swap 时要注意泛型擦除陷阱

如果你绕过 Collections.swap 自己写,比如用临时变量 + set,得小心泛型类型安全问题。尤其在方法签名里用了通配符或原始类型时,编译器可能放行但运行时报错。

  • 典型错误代码:List> list = new ArrayList(); swap(list, 0, 1); —— 编译失败,因为 ? 不允许 set
  • 正确写法:确保列表是具体泛型类型,如 List;或用 Object 临时变量(但丢失类型信息)
  • 示例安全写法:
    public static <T> void swap(List<T> list, int i, int j) {<br>    T tmp = list.get(i);<br>    list.set(i, list.get(j));<br>    list.set(j, tmp);<br>}

不可变列表、并发列表等特殊实现的兼容性

Collections.swap 本质依赖 get + set,所以只要列表实现了这两个方法且允许修改,它就能工作;否则就会失败。

  • 明确不支持:所有 Collections.unmodifiable* 系列返回的视图、List.of()(Java 9+)、Guava 的 ImmutableList
  • 需注意并发安全:对 CopyOnWriteArrayList 可以用,但每次 set 都触发复制,高频 swap 会显著拖慢性能
  • 推荐判断方式:看文档是否声明 “supports the set operation” —— 这是 JDK Javadoc 中的明确判定依据
实际用的时候,别光盯着 swap 函数本身,先确认你的 list 是不是真能被改。很多问题其实出在上游构造方式上,而不是 swap 写错了。

以上就是《JavaCollections.swap使用技巧分享》的详细内容,更多关于的资料请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>