百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

Rust编程思想(七)—— loop循环表达式与while/for条件循环语句

wptr33 2025-09-06 13:55 8 浏览

在Rust中,除了while、for实现循环控制,增加了一个新的循环方式loop,这才是名符其实的循环语句。类似C/C++、Java、JavaScript、C#语言一样,使用关键字continue和break控制执行流程。Continue语句表示本次循环内,后面的语句不再执行,直接进入下一轮循环。break语句表示跳出循环,不再继续,而loop关键表示一个无限死循环。loopwhile true的区别在于前者是表达式,而后者是语句,这是二者区别的核心。

其实loop就是一个while(true)的语法糖,既然死循环都出来了,又想起来为啥不支持 :?三元运算符?

1. loop

loop 关键字,在Rust中,使用loop表示一个无限死循环。示例如下:

//loop死循环
use std::time::Duration;
use std::thread::sleep;
fn loop_test1(){ 
    println!("***** loop_test1 start *****");

    let mut count = 0u32;
    loop{
        count +=1;
        println!("print count={}",count);    
        sleep(Duration::from_millis(100)); //睡眠 100 ms
        // if count>10{
        //     break; //如果没有break则一直死循环。
        // }
    }
    println!("***** loop_test1 end *****\n");
}

fn main() {

    loop_test1();
}

这个死循环有一个非常明确的使用场景,就是在嵌入式系统中的主函数,直接看到loop就可以认为是主函数了,或者是某个任务的主体,这个设计还是很好的,至少在可读性上又增加了一层。

break语句和continue语句还可以在多重循环中选择跳出到哪一层的循环。

//loop循环 + 跳出标签所指的那层循环。
fn loop_test2()
{
    let mut count1 = 0u32;
    let mut count2 = 0u32;
    'loop1:loop{
        count1 +=1;

        'loop2:loop{
            count2+=1;
            if count2 == 2 {
                println!("count2=={},continue 'loop2!",count2);  
                continue 'loop2;
            }
            
            if count2 >= 4 {
                count2 = 0;
                break ;
            }
            
            if count1 == 5{
                println!("count1=={},break: out of 'loop1!",count1);
                break 'loop1;
            }
            
            println!("print count1={} -- count2={}",count1,count2);
        }
    }
}

loop的break返回值
loop是表达式而while/for是语句,这是loop与while/for循环不一样的地方。
在loop内部break的后面可以跟一个表达式,这个表达式就是最终的loop表达式的值。如果一个loop永远不返回,那么它的类型就是“发散类型”。

当使用 loop循环时,通过形如 break EXPR 或 break 'label EXPR 来返回,其中 EXPR 是一个表达式,它的结果被从 loop循环中返回。 例如:

#![allow(unused)]
fn main() {
    let (mut a, mut b) = (1, 1);
    let result = loop {
        if b > 10 {
            break b;
        }
        let c = a + b;
        a = b;
        b = c;
    };
    // 斐波那契数列中第一个大于10的值:
    assert_eq!(result, 13);
}

2. while

while 关键字,while语句是带条件判断的循环语句。其语法是while关键字后跟条件判断语句,最后是结果语句块。如果条件满足,则持续循环执行结果语句块。示例如下:

//while语句是带条件判断的循环语句,同样可有continue和break来控制循环流程。。
fn while_test1(){ 
    let mut while_count = 0;
    while while_count < 10 {
        while_count+=1;
        println!("print while_count={}",while_count);

        if while_count == 3 {
            println!("continue: while while_count= {}",while_count);
            continue;
        }

        if while_count == 5 {
            println!("reak: out of while while_count = {}",while_count);
            break;
        }
    }
}
//while实现死/无限循环:
use std::time::Duration;
use std::thread::sleep;
fn while_test2(){ 
    let mut while_count = 0;
    #[warn(while_true)]{
        while true {
            while_count+=1;
            println!("print while_count={}",while_count);
            sleep(Duration::from_millis(500)); //睡眠 500 ms

            // if while_count == 10 {
            //     break; //如果没有break则一直循环。
            // }
        } 
    }
}

会警告提示用关键字“loop”:

warning: denote infinite loops with `loop { ... }`
   --> src/main.rs:108:9
    |
108 |         while true {
    |         ^^^^^^^^^^ help: use `loop`

注意:loop和while true的死/无限循环区别:
//loop和while true无限循环区别。
fn loop_while_infinitecycle_test3(){ 
    let x;
    loop { x = 1; break; }
    println!("{}", x);//没问题。

    let x;
    while true { x = 1; break; }
    println!("{}", x);//报错,提示x不确定被赋初值,却被使用。
}

原因在于对编译器来说loop是一定会进入循环体去执行,但是while是需要先判断条件为真才进入循环体执行。

while let谓词模式循环
while let循环在语义上类似于 while循环,但它用 let关键字后紧跟着一个模式、一个 =、一个检验对象(scrutinee)表达式和一个块表达式,来替代原来的条件表达式。 如果检验对象表达式的值与模式匹配,则执行循环体块,然后控制流再返回到模式匹配语句。如果不匹配,则 while表达式执行完成。

#![allow(unused)]
fn main() {
    let mut x = vec![1, 2, 3];

    while let Some(y) = x.pop() {
        println!("y = {}", y);
    }

    while let _ = 5 {
        println!("不可反驳模式总是会匹配成功");
        break;
    }
}

可以使用操作符 | 指定多个模式。 这与匹配(match)表达式中的 | 具有相同的语义:

#![allow(unused)]
fn main() {
    let mut vals = vec![2, 3, 1, 2, 2];
    while let Some(v @ 1) | Some(v @ 2) = vals.pop() {
        // 打印 2, 2, 然后 1
        println!("{}", v);
    }
}

3. for循环

Rust中的for关键字类似其他语言中的for-each循环,但不是C/C++的三段式for循环语句。for循环的主要用处是利用迭代器对包含同样类型的多个元素的容器执行遍历。for循环内部也可以使用continuebreak控制执行流程。举例如下:

//for 循环。continue和break控制执行流程
fn for_test1(){ 
    let mut for_count: i32 ;
    for n in 1..10{
        for_count = n;
        
        if for_count == 3 {
            continue;
        }
        if for_count == 6 {
            break;
        }
        println!("print for_count={}",for_count);
    }
}
//for 循环。for + match 的用法
fn for_test2(){     
    /* 数值匹配 */
    for n in 0..=20{
        if n == 16 {
            println!("continue: for n= {}",n);
            continue;
        }
        match n {
            0 => println!("match 0 n={}",n),
            0|2|4|6|8 => println!("match n is even number:{}",n),
            1|3|5|7|9 => println!("match n is odd number:{}",n),
            10..=15 => println!("match n is '10'<=n<='19' : {}",n),
            _ => println!("useless number {}",n),
        }
    }

    /* 字符串匹配 */
    let for_name = vec!["s1","s2","s3"];
    for name in for_name.iter(){
        match name{
            &"s1" => println!("match \"s1\" name ={}",name),
            &"s2" => println!("match \"s2\"  name ={}",name),
            _ => println!("current name={}, isn't = \"s1\" or \"s2\"!",name),
        }
    }
}

for + while 循环混用:

fn for_test3(){ 
    println!("******************** for_test3 start ********************");

    'outer: for x in 1..5 {
        let mut i = 0;
        while i < x {
            i += 1;
            if i == 3 {
                println!("###break: 'outer. for: x={}, while:i={}",x,i);
                break 'outer;
            }
            println!("for: x={}, while:i={}",x,i);
        }
    }
}

for+iter(), iter()函数获取数组中所有元素的值:

fn my_fn()->u32{
    let arr:[i32;4]=[10,20,30,40];
    println!("array is {:?}",arr);
    println!("array size is :{}",arr.len());
    
    for val in arr.iter(){
      println!("value is :{}",val);
    }
    return 0;
}

//for 引用传递
fn my_fn(arr:&mut [i32;3])
{
    let n=arr.len();
   for i in 0..n {
      arr[i]=0;
   }
}

fn main() {
    let mut arr:[i32;3]=[1,2,3];
    println!("arr: {:?}", arr);
    my_fn(&mut arr);
    println!("arr after uptade: {:?}", arr);
}

注意
for i in 0...10 { }中的i是直接使用,是不用单独声明的,例如:
let mut i:u32 = 0;警告未使用的变量:warning: unused variable: i

for i in 1..10 { //这个for循环的i不是上面声明的变量i.
    println!("i={ }",i);
}

4. 基于rust循环控制实现的相关算法

  1. 计算考拉兹序列的长度。
    (若n为偶数): n→n/2.(若n为奇数): n→n*3+1.
    从13开始,可以迭代生成如下的序列:13→40→20→10→5→16→8→4→2→1,代码示例如下:
fn collatz_length(mut n: i32) -> u32 {
    let mut len = 1;
    while n > 1 {
        println!("len={}:{}",len,n);
        n = if n % 2 == 0 { n  / 2 } else { 3 * n + 1 };
        len += 1;
    }
        println!("len={}:{}",len,n);
    return len;
}
  1. 求斐波那契数列
    斐波那契函数的变化规律:F1=1,F2=1,F(n)=F(n-1)+F(n-2) (n>=3)。
fn fib(mut n: u32) -> u32 {
    let mut fn1:u32 = 1;
    let mut fn2:u32 = 1;
    let mut fn_value:u32 = 0;
    
    while n > 2{
        println!("fn_value={},fn1={},fn2={}",fn_value,fn1,fn2);
        /* fn_value = fn-1 + fn-2 */
        fn_value=fn1+fn2;
        
        fn2=fn1;
        fn1=fn_value;
        n-=1;
    }
    println!("fn1={},fn2={}",fn1,fn2);
    return fn_value;
}
  1. 猴子吃桃问题
    一只猴子第一天摘了n个桃,当即吃了一半又多吃了一个,第二天又吃了一半再多吃一个,直到第十天早上想吃时只剩下1个,问第一天摘了多少桃?
fn peach() -> u32 {
    let mut peach_num:u32 = 1;//in day 10(no eat), only remained 1 peach.
    for i in 1..10{
        println!("day={},peach_num={}",i,peach_num);
        peach_num=2*(peach_num+1);
    }
    println!("day={},peach_num={}",10,peach_num);
    return peach_num;
}
  1. 实现冒泡排序
fn my_fn( mem:&mut Vec<i32>)
{
    for i in 0..mem.len() {
        for x in 0..mem.len()-1{
            if mem[x] > mem[x+1]{
                mem.swap(x,x+1);
            }
        }
    }
}

fn main() {
    let mut mem1 =vec![1,6,5,3,9,4,8,2,7,0];
    println!("mem1: {:?}", mem1);
    my_fn(&mut mem1);
    println!("mem1 after update: {:?}", mem1);
}

建议
在实际使用中,对于数组的遍历,建议用for,在线程的主函数体中使用loop,在其它情况下条件循环使用while,形成模式以后使代码可读性更强。

相关推荐

MySQL进阶五之自动读写分离mysql-proxy

自动读写分离目前,大量现网用户的业务场景中存在读多写少、业务负载无法预测等情况,在有大量读请求的应用场景下,单个实例可能无法承受读取压力,甚至会对业务产生影响。为了实现读取能力的弹性扩展,分担数据库压...

Postgres vs MySQL_vs2022连接mysql数据库

...

3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?

引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...

一文由浅入深带你完全掌握MySQL的锁机制原理与应用

本文将跟大家聊聊InnoDB的锁。本文比较长,包括一条SQL是如何加锁的,一些加锁规则、如何分析和解决死锁问题等内容,建议耐心读完,肯定对大家有帮助的。为什么需要加锁呢?...

验证Mysql中联合索引的最左匹配原则

后端面试中一定是必问mysql的,在以往的面试中好几个面试官都反馈我Mysql基础不行,今天来着重复习一下自己的弱点知识。在Mysql调优中索引优化又是非常重要的方法,不管公司的大小只要后端项目中用到...

MySQL索引解析(联合索引/最左前缀/覆盖索引/索引下推)

目录1.索引基础...

你会看 MySQL 的执行计划(EXPLAIN)吗?

SQL执行太慢怎么办?我们通常会使用EXPLAIN命令来查看SQL的执行计划,然后根据执行计划找出问题所在并进行优化。用法简介...

MySQL 从入门到精通(四)之索引结构

索引概述索引(index),是帮助MySQL高效获取数据的数据结构(有序),在数据之外,数据库系统还维护者满足特定查询算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构...

mysql总结——面试中最常问到的知识点

mysql作为开源数据库中的榜一大哥,一直是面试官们考察的重中之重。今天,我们来总结一下mysql的知识点,供大家复习参照,看完这些知识点,再加上一些边角细节,基本上能够应付大多mysql相关面试了(...

mysql总结——面试中最常问到的知识点(2)

首先我们回顾一下上篇内容,主要复习了索引,事务,锁,以及SQL优化的工具。本篇文章接着写后面的内容。性能优化索引优化,SQL中索引的相关优化主要有以下几个方面:最好是全匹配。如果是联合索引的话,遵循最...

MySQL基础全知全解!超详细无废话!轻松上手~

本期内容提醒:全篇2300+字,篇幅较长,可搭配饭菜一同“食”用,全篇无废话(除了这句),干货满满,可收藏供后期反复观看。注:MySQL中语法不区分大小写,本篇中...

深入剖析 MySQL 中的锁机制原理_mysql 锁详解

在互联网软件开发领域,MySQL作为一款广泛应用的关系型数据库管理系统,其锁机制在保障数据一致性和实现并发控制方面扮演着举足轻重的角色。对于互联网软件开发人员而言,深入理解MySQL的锁机制原理...

Java 与 MySQL 性能优化:MySQL分区表设计与性能优化全解析

引言在数据库管理领域,随着数据量的不断增长,如何高效地管理和操作数据成为了一个关键问题。MySQL分区表作为一种有效的数据管理技术,能够将大型表划分为多个更小、更易管理的分区,从而提升数据库的性能和可...

MySQL基础篇:DQL数据查询操作_mysql 查

一、基础查询DQL基础查询语法SELECT字段列表FROM表名列表WHERE条件列表GROUPBY分组字段列表HAVING分组后条件列表ORDERBY排序字段列表LIMIT...

MySql:索引的基本使用_mysql索引的使用和原理

一、索引基础概念1.什么是索引?索引是数据库表的特殊数据结构(通常是B+树),用于...