登录
首页 >  Golang >  Go问答

golang中常量左移后的奇妙问题

来源:SegmentFault

时间:2023-02-16 15:21:06 119浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《golang中常量左移后的奇妙问题》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下go,希望所有认真读完的童鞋们,都有实质性的提高。

问题内容

package main

import (
    "fmt"
)

const (
    cimax uint8 = 255
)

func main() {
    var i uint8
    var imax uint8 = 255

    i = imax 

是不是常量不能作为左移操作法的操作数?

正确答案

以下是一些不够严谨的分析。

参照文档进行研究,Go必然有这么几个特点:

  1. 编译型语言,编译器的静态求值是自然的。
  2. 必须使用强制类型转换。甚至于对一个uint16变量赋uint8的值都是不允许的。
  3. 整数字面值不包含存储空间的大小,可以直接赋给各种uintX(必须不溢出)。
  4. 运算的结果的类型,以左侧操作数为准。

而从表象上来看:

  1. 静态求值的规则是先不顾代码范围,强制求值,之后再塞回目标的存储空间中
  2. 对变量左移的时候,会直接考虑存储空间大小,将左移的数据丢弃

对于第1条,如下的代码都会出现同样的错误。对于这两个例子,因为

左侧都是uint8,所以最终的结果也是uint8,则得出的结果装不下uint8就会报错,哪怕最后准备了uint16的容器也不行:

const c_i8 uint8 = 255
var i16 uint16
i16 = uint16(c_i8 

对于第2条,如下的代码都可以正确执行:

var i8 uint8
var i8_2 uint8 = 255
i8 = i8_2 

所以并不是说常量不能作为操作数,而是常量左移之后,Go面对溢出的行为并不是截断而是报错。临时对付这个问题,你可能需要先用不定长度的uint类型绕开溢出的限制,再用掩码(

& 0xFF
)运算去约束结果的范围:

const c_i8 uint8 = 255
var i16 uint16
i16 = uint16(uint(c_i8) 

更深入的分析期待其他回答者的补充。

文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《golang中常量左移后的奇妙问题》文章吧,也可关注golang学习网公众号了解相关技术文章。

声明:本文转载于:SegmentFault 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>