为了账号安全,请及时绑定邮箱和手机立即绑定

读《Go并发编程实战》流程控制方式

标签:
Go

       一个网站的用户管理中,一般用户都会起一个昵称,当然用户A和用户B的昵称有可能相同,我们就使用类型为map[string]int的集合来表示,其中string代表用户昵称、int表示昵称相同的人数。

       集合数据内容为:{"chen":20,   "chen长":10,   "春风":5,   "龙ge":13,   "where你are":7}。


§问题1. 从这个集合中找出只包含中文的用户昵称的计数信息

    上面集合中只有用户昵称为“春风”的元素满足条件,其它像“龙ge”、“chen长”、“where你are”都不是只包含中文。像看程序实现:

users := map[string]int{"chen": 20, "chen长": 10, "春风": 5, "龙ge": 13, "where你are": 7}var targetCount map[string]int = make(map[string]int)for name, count := range users {    matched := true    for _, v := range name {        if v < '\u4e00' || v > '\u9fbf' {            matched = false            break        }    }    if matched {        targetCount[name] = count    }}fmt.Println(targetCount)

解释一下:

第一层for循环遍历users集合,得到昵称(name)和计数(count ):

for name, count := range users{

      // ......

}

第二层循环遍历昵称name字符串中的每个字符,并判断是否有非中文字符:

for _, v := range name{

      if v <'\u4e00' || v > '\u9fbf' {  // 该范围下的字符为非中文字符

      }

}

【备注】:

    这里由变量matched充当了标志,当出现非中文字符时,matched被赋为false,并使用break中止第二层循环,然后判断该name是否仅为中文,若是则添加进入targetCountw集合中。


运行结果为:map[春风:5]


§问题2. 从这个集合中找出只包含中文的用户昵称的计数信息,但发现第一个非全中文的用户昵称的时候就停止查找。

    这里涉及到map类型的特性问题,当使用for进行遍历users集合时,顺序会出现不确定性,即第一个昵称有可能是“chen”,也有可能是“春风”,也有可能是“龙ge”,总之是不确定的。

    假设第一个昵称是“chen长”时,程序发现它是非全中文的昵称,这时整个程序需要结束。

for name, count := range users {    matched := true    for _, v := range name {        if v < '\u4e00' || v > '\u9fbf' {            matched = false            break        }    }    if !matched {        break    } else {        targetCount[name] = count    }}fmt.Println(targetCount)

    这个程序实现与上面基本是大同小异,不同之处在于第一层循环中的matched判断。

    假设在第一层for循环中,第一次取到的昵称name为“龙ge”,此时第二层for循环中会逐个字符地查看“龙ge”这个字符串,当程序发现“龙ge”是非全中文时,matched标志被置为false,同时跑出第二层循环,然后在第一层循环中问:“取到的昵称‘龙ge’是否为全中文?”,结果不是,于是按照题目要求就break掉程序并时跳出第一层循环。


    这里如果您多运行一下程序,就会发现结果有可能是不同的,有时为map[],有时为map[春风:5]


§问题3. 不使用辅助标识(如上例中的matched)解决§问题2.

    作者这里是让您学习break Label这个东东,所以这里直接引用作者原话了:“我们之前说过,break语句可以与标记(Label)语句一起配合使用。”

L:    for name, count := range users {        for _, v := range name {            if v < '\u4e00' || v > '\u9fbf' {                 break L            }        }        targetCount[name] = count    }

该程序与§问题2程序功能是相同的,只是把标志matched移除掉了。


    一条标记语句可以成为goto语句、break语句和contiune语句的目标。标记语句中的标记只是一个标识符,它可以被置在任何语句的左边以作为这个语句的标签。标记和被标记的语句之间需要用冒号“:”来分隔,即如下所示:

L:

     for name, count := range users{

          // .....

     }

    需要注意的是,标记L也是一个标识符,那么当它在未被使用的时候同样也会报告编译错误。那么怎么使用呢?一种方法就是让它成为break的目标,即上面示例中的break L。

    标签L代码块包含了第一层for循环,第一层for循环包含了第二层for循环,所以当break L时,它是中止整个标签L代码块,所以这里会中止两层for循环。


 对比“§问题2”和“§问题3”的两个代码实现,“§问题3”的代码更简洁一些。


§问题4.  还是解决问题1的事情,只不过要求使用continue语句,即从这个集合中找出只包含中文的用户昵称的计数信息

由于这里没有更多的知识点,所以引用原文:

    实际上,Go语言中的continue语句只能在for语句中使用。continue语句会使直接包含它的那个for循环直接进入下一次迭代,换言之,本次迭代不会执行该continue语句后面那些语句(它们被跳过了)。

for name, count := range users {    matched := true    for _, v := range name {        if v < '\u4e00' || v > '\u9fbf' {            matched = false            break        }    }    if !matched {       continue    }    targetCount[name] = count}

或者

L:    for name,count := range users{        for _, v := range name{            if r < '\u4e00' || r > '\u9fbf'{                continue L            }        }        targetCount[name] = count    }


OK,在读这个章节时,只是感觉这个例子有点意思,所以分享给大家 :)

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消