for循环优化方式有哪些?

前言

经常使用一些循环,进行耗时计算的操作,特别是 for 循环,它是一种重复计算的操作,如果处理不好,耗时就比较大,如果处理书写得当,将大大提高效率,下面总结几条 for 循环的常见优化方式。

首先初始化一个集合 list,如下:

代码语言:javascript

Cloud Studio代码运行

ini 复制代码
List<String> list = new ArrayList<>();

方式一:最常规的不加思考的写法

代码语言:javascript

Cloud Studio代码运行

css 复制代码
for (int i = 0; i < list.size(); i++) {
    System.out.println(list.get(i));
}
  • 优点:较常见,易于理解
  • 缺点:每次都要计算 list.size()

方式二:数组长度提取出来

代码语言:javascript

Cloud Studio代码运行

ini 复制代码
int size = list.size();

for (int i = 0; i < size; i++) {
    System.out.println(list.get(i));
}
  • 优点:不必每次都计算 list.size()

  • 缺点:

    • size 的作用域不够小,违反了最小作用域原则
    • 不能在 for 循环中操作 list 的大小,比如除去或新加一个元素

方法三:数组长度提取出来

代码语言:javascript

Cloud Studio代码运行

css 复制代码
for (int i = 0, size = list.size(); i < size; i++) {
    System.out.println(list.get(i));
}
  • 优点:不必每次都计算,变量的作用域遵循最小范围原则

  • 缺点:

    • size 的作用域不够小,违反了最小作用域原则
    • 不能在 for 循环中操作 list 的大小,比如除去或新加一个元素

方法四:采用倒序的写法

代码语言:javascript

Cloud Studio代码运行

css 复制代码
for (int i = list.size() - 1; i >= 0; i--) {
    System.out.println(list.get(i));
}
  • 优点:不必每次都计算,变量的作用域遵循最小范围原则

  • 缺点:

    • 1、结果的顺序会反
    • 2、看起来不习惯,不易读懂
  • 适用场合:与显示结果顺序无关的地方:比如保存之前数据的校验

方法五:Iterator 遍历

代码语言:javascript

代码运行次数:0

Cloud Studio代码运行

ini 复制代码
for (Iterator<String> it = list.iterator(); it.hasNext(); ) {
    System.out.println(it.next());
}
  • 优点:简洁

方法六:jdk 1.5 后的写法

代码语言:javascript

代码运行次数:0

Cloud Studio代码运行

scss 复制代码
for (String o : list) {
    System.out.println(o);
}
  • 优点:简洁, 结合泛型使用更简洁
  • 缺点:jdk 1.4 向下不兼容(也就是 < 1.5 版本就不能这么写)

方法七:循环嵌套外小内大原则

代码语言:javascript

代码运行次数:0

Cloud Studio代码运行

ini 复制代码
for (int i = 0; i < 10; i++) {
    for (int j = 0; j < 10000; j++) {
    }
}

原因

分支优化规则

引入流水线工作机制以后,为了配合流水线工作,处理器增加了一个分支目标缓冲器(Branch Target Buffer)。在流水线工作模式下,如果遇到分支结构,就可以利用分支目标缓冲器预测并读取指令的目标地址。分支目标缓冲器在程序运行时将动态记录和调整转移指令的目标地址,可以记录多个地址,对其进行表格化管理。当发生转移时,如果分支目标缓冲器中有记录,下一条指令在取指令阶段就会将其作为目标地址。如果记录地址等于实际目标地址,则并行成功;如果记录地址不等于实际目标地址,则流水线被冲洗。同一个分支,多次预测失败,则更新记录的目标地址。因此,分支预测属于 "经验主义" 或 "机会主义",会存在一定的误测。

基于上述原因,大家以后在编写多重循环时应该把大循环放到内层,这样可以增加分支预测的准确度,如下面的示例所示:

代码语言:javascript

Cloud Studio代码运行

ini 复制代码
int[][] a = new int[10][10000];

for (int i = 0; i < 10; i++) {
    // 下面每次循环会预测成功9999次
    // 第1次没有预测,最后退出循环时预测失败1次这样的
    // 过程重复10次
    for (int j = 0; j < 10000; j++) {
        a[i][j]++;
    }
}

for (int j = 0; j < 10000; j++) {
    // 下面每次循环会预测成功9次
    // 第1次没有预测,最后退出循环时预测失败1次
    // 这样的过程重复10000次
    for (int i = 0; i < 10; i++) {
        a[i][j]++;
    }
}

方法八:循环嵌套提取不需要循环的逻辑

代码语言:javascript

Cloud Studio代码运行

ini 复制代码
// 前:
int a = 10, b = 11;
for (int i = 0; i < 10; i++) {
    i = i * a * b;
}

// 后:
int c = a * b;
for (int i = 0; i < 10; i++) {
    i = i * c;
}

方法九:异常处理写在循环外面

反例:

代码语言:javascript

Cloud Studio代码运行

php 复制代码
for (int i = 0; i < 10; i++) {
    try {

    } catch (Exception e) {

    }
}

正例:

代码语言:javascript

Cloud Studio代码运行

php 复制代码
try {
    for (int i = 0; i < 10; i++) {
    }
} catch (Exception e) {

}
相关推荐
于小汐在咯1 小时前
深入浅出:增强现实(AR)技术全解析
后端·ar·restful
爱上妖精的尾巴2 小时前
5-27 WPS JS宏数组元素添加删除应用2
后端·restful·wps·js宏
努力的小郑2 小时前
与产品经理的“模糊”对决:Elasticsearch实现MySQL LIKE '%xxx%' 的奇幻之旅
后端·elasticsearch·搜索引擎
陌上花开缓缓归以2 小时前
linux系统移植过程中挂死问题分析
性能优化
工藤学编程2 小时前
深入Rust:Tokio多线程调度架构的原理、实践与性能优化
性能优化·架构·rust
一 乐2 小时前
物业管理系统|小区物业管理|基于SprinBoot+vue的小区物业管理系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·后端
稚辉君.MCA_P8_Java2 小时前
RocketMQ 是什么?它的架构是怎么样的?和 Kafka 又有什么区别?
后端·架构·kafka·kubernetes·rocketmq
yolo_Yang3 小时前
【Spring Boot】Spring Boot解决循环依赖
java·spring boot·后端
wdfk_prog3 小时前
结合QBoot与HPatchLite实现高效差分升级(FOTA)
java·后端·struts
舒一笑3 小时前
用数据照亮成长之路:PandaCoder Git 统计工具窗口
git·后端·intellij idea