Go提供了两种size的浮点数,float32和float64。它们的算术规范是由IEEE754国际标准定义,现代CPU都实现了这个规范。

浮点数能够表示的范围可以从很小到很巨大,这个极限值范围可以在math包中获取,math.MaxFloat32表示float32的最大值,大约是3.4e38,math.MaxFloat64大约是1.8e308,两个类型最小的非负值大约是1.4e-45和4.9e-324。

float32大约可以提供小数点后6位的精度,作为对比,float64可以提供小数点后15位的精度。通常情况应该优先选择float64,因此float32的精确度较低,在累积计算时误差扩散很快,而且float32能精确表达的最小正整数并不大,因为浮点数和整数的底层解释方式完全不同,具体见IEEE754详解。

var f float32 = 16777216 // 1 << 24
fmt.Println(f == f+1)    // "true"!

浮点数字面量可以使用十进制数字表示:

const e = 2.71828 // (非精确值)

小数点前面或者后面的数字都可以省略,例如:.707 , 1.   ,对于那种很小或者很大的数值最好用科学计数法,在指数前加上e或者E:

const Avogadro = 6.02214129e23  // 阿伏伽德罗常数
const Planck   = 6.62606957e-34 // 普朗克常数

fmt打印浮点数时,若使用%g参数,会采用更高的精度更紧凑的表现形式进行打印,但是在打印表格数据时,%e(指数)或者%f(非指数的)的形式可能更合适,上面三个参数都可以控制打印的宽度和精度:

for x := 0; x < 8; x++ {fmt.Printf("x = %d e^x = %8.3f\n", x, math.Exp(float64(x)))
}

上面的代码使用了小数点后3位的精度进行打印,打印宽度是8个字符:

x = 0   ex =    1.000
x = 1   ex =    2.718
x = 2   ex =    7.389
x = 3   ex =   20.086
x = 4   ex =   54.598
x = 5   ex =  148.413
x = 6   ex =  403.429
x = 7   ex = 1096.633

math包不仅包含了大量的数学函数,还包含了IEEE754规范下特殊浮点数的创建和查看:正无穷,表明数字太大溢出的情况;负无穷,表示被0除的结果;NaN(不是一个数值),用来表示无效运算的结果,例如 0 / 0, math.Sqrt(-1)。

var z float64
fmt.Println(z, -z, 1/z, -1/z, z/z) //  "0 -0 +Inf -Inf NaN"

函数math.IsNaN测试一个数值是否是NaN,math.NaN会返回一个NaN值。虽然可以在数值计算中用NaN做为一个哨兵值,但是测试一个计算的结果是否等于NaN是很危险的,因为任何值跟NaN比较的结果都是false:

nan := math.NaN()
fmt.Println(nan == nan, nan < nan, nan > nan) // "false false false"

如果一个返回浮点数的函数可能失败,那最好还是单独的报告失败:

func compute() (value float64, ok bool) {// ...if failed {return 0, false}return result, true
}

下面的程序演示了通过浮点数计算来生成图形,使用了z = f(x,y)来进行三维建模,使用了SVG格式做图像输出,SVG是一个用于绘制矢量线的XML标准。下图展示了sin(r)/r函数生成的图形,r = sqrt(x*x + y*y):

// Surface computes an SVG rendering of a 3-D surface function.
package mainimport ("fmt""math"
)const (width, height = 600, 320            // canvas size in pixelscells         = 100                 // number of grid cellsxyrange       = 30.0                // axis ranges (-xyrange..+xyrange)xyscale       = width / 2 / xyrange // pixels per x or y unitzscale        = height * 0.4        // pixels per z unitangle         = math.Pi / 6         // angle of x, y axes (=30°)
)var sin30, cos30 = math.Sin(angle), math.Cos(angle) // sin(30°), cos(30°)func main() {fmt.Printf("<svg xmlns='http://www.w3.org/2000/svg' "+"style='stroke: grey; fill: white; stroke-width: 0.7' "+"width='%d' height='%d'>", width, height)for i := 0; i < cells; i++ {for j := 0; j < cells; j++ {ax, ay := corner(i+1, j)bx, by := corner(i, j)cx, cy := corner(i, j+1)dx, dy := corner(i+1, j+1)fmt.Printf("<polygon points='%g,%g %g,%g %g,%g %g,%g'/>\n",ax, ay, bx, by, cx, cy, dx, dy)}}fmt.Println("</svg>")
}func corner(i, j int) (float64, float64) {// Find point (x,y) at corner of cell (i,j).x := xyrange * (float64(i)/cells - 0.5)y := xyrange * (float64(j)/cells - 0.5)// Compute surface height z.z := f(x, y)// Project (x,y,z) isometrically onto 2-D SVG canvas (sx,sy).sx := width/2 + (x-y)*cos30*xyscalesy := height/2 + (x+y)*sin30*xyscale - z*zscalereturn sx, sy
}func f(x, y float64) float64 {r := math.Hypot(x, y) // distance from (0,0)return math.Sin(r) / r
}

corner函数返回两个值,分别是网格顶点的x,y坐标。
如果要深入解释图像生成的原理,我们还需要一些几何学知识。但是这里会跳过这些几何学原理,毕竟这个程序主要是为了演示浮点数的运算。程序本质上是三个坐标系间的映射,如下图所示,第一个是100*100的二维网格,每个单元格都有坐标(i,j),从坐标系原点(0,0)开始延伸。绘制时是从远处开始绘制,因此远处先绘制的多边形可能被后绘制的多边形覆盖。

第二个坐标系是三维网格组成的,坐标(x,y,z),其中x和y是i和j的线性函数,通过坐标转换把原点变为中心点,然后通过xyrange进行缩放。高度z是f(x,y)的值。

第三个坐标系是一个二维的画布,起点(0,0)在左上角。画布上任意点的坐标(sx,sy),我们使用等角投影将三维点(x,y,z)投影到二维的画布中。画布上的点离右边越远,x和y值越大,z值越小。x和y的垂直缩放系数是30度角的sin值,水平缩放系统是30度角的cos值。z的缩放系数0.4是一个任意的值。

对于二维网格中的每一个网格单元,main函数会计算该单元在画布上对应的多边形ABCD的顶点,B对应顶点(i,j),A、C、D是B的邻接点,然后输出SVG的绘制指令。

文章所有权:Golang隐修会 联系人:孙飞,CTO@188.com!

Go语言核心之美 2.2-浮点数相关推荐

  1. Go语言核心之美-必读

    Go语言核心之美开篇了!,无论你是新手还是一代高人,在这个系列文章中,总能找到你想要的! 博主是计算机领域资深专家并且是英语专8水平,翻译标准只有三个:精确.专业.不晦涩,为此每篇文章可能都要耗费数个 ...

  2. Go语言核心之美 2.6-常量

    在Go语言中,常量表达式是在编译期求值的,因此在程序运行时是没有性能损耗的.常量的底层类型是前面提过的基本类型:布尔值,字符串,数值变量. 常量的声明方式和变量很相似,但是常量的值是不可变的,因此在运 ...

  3. Go语言核心之美 3.4-Struct结构体

    struct(结构体)也是一种聚合的数据类型,struct可以包含多个任意类型的值,这些值被称为struct的字段.用来演示struct的一个经典案例就是雇员信息,每条雇员信息包含:员工编号,姓名,住 ...

  4. Go语言核心之美 3.3-Map

    哈希表是一种非常好用.适用面很广的数据结构,是key-value对的无序集合.它的key是唯一的,通过key可以在常数复杂度时间内进行查询.更新或删除,无论哈希表有多大. Go语言的map类型就是对哈 ...

  5. Go语言核心之美 2.1-整数

    第二章 序 在计算机底层,一切都是比特位.然而计算机一般操作的都是固定大小的值,称之为字(word).字会被解释为整数.浮点数.比特位数组.内存地址等,这些字又可以进一步聚合成数据包(packet). ...

  6. Go语言核心之美 2.5-字符串

    字符串是不可变的字节序列,虽然可以包含任意数据,包括0这个字节,不过字符串通常是用来包含可读性较强的文本.文本字符串通常采用UTF-8编码,由Unicode码点(rune)组成. 内置的len函数会返 ...

  7. Go语言核心之美 1.5-作用域

    变量的作用域是指程序代码中可以有效使用这个变量的范围.不要将作用域和生命期混在一起.作用域是代码中的一块区域,是一个编译期的属性:生命期是程序运行期间变量存活的时间段,在此时间段内,变量可以被程序的其 ...

  8. Go语言核心之美 1.4-包和文件

    一.Package Go语言中的包(Package)就像其它语言的库(Library)或模块(Module)一样,支持模块化,封装性,可重用性,单独编译等特点.包的源码是由数个.go文件组成,这些文件 ...

  9. Go语言核心之美 1.2-变量及声明篇

    变量 1.声明变量 使用var关键字可以创建一个指定类型的变量: var i int = 0 var i = 0 var i int 以上三个表达式均是合法的,第三个表达式会将i初始化为int类型的零 ...

  10. Go语言核心之美 2.3-复数

    Go提供了两种大小的复数类型:complex64和complex128,分别由float32和float64组成.内置函数complex从指定的实部和虚部构建复数,内置函数real和imag用来获取复 ...

最新文章

  1. 封装了一下我佛山人4.0 (支持vs2005)asp.net 页面验证
  2. 没想到我提前56年感受了赛博朋克
  3. 山东华为服务器安装系统,山东云服务器平台安装
  4. vivado使用自带IP核和创建自己定义的IP核
  5. 只想安安静静的做个程序员
  6. 【字符串】最长回文子串 ( 动态规划算法 ) ★
  7. javascript +new Date()
  8. Gradle善良:仅添加包装用于战争
  9. 【Android】命令行jarsigner签字和解决找不到证书链错误
  10. 在XML drawable中引用自定义颜色
  11. python——画伪彩图、设置彩色图例
  12. 安装WIN10系统时“谁将会使用这台电脑”输入卡死状态的问题
  13. 算法题目打卡:Ques20201007
  14. P9 力荐!阿里巴巴最新出品 776 页 JDK 源码 + 并发核心原理解析小册
  15. 《输赢》精彩段落总结
  16. Windows 8安装软件时遇到运行时错误‘339’ MSCOMCTL.ocx或其附件之一未注册
  17. OpenCV图像旋转(cv::rotate)与镜像(cv::flip)
  18. 在ftp服务器上打开wOrd文件空白,ftp服务器word文件打开是空白
  19. 小娜 api android,Android版小娜截图曝光:联通一切
  20. ElasticSearch经纬度相关查询

热门文章

  1. RestTemplate源码解读
  2. 初级java程序员遇到的面试题
  3. 荷兰国旗算法设计C语言,2.7 荷兰国旗
  4. Linux下设备驱动
  5. 如何查看计算机的硬盘序列号,电脑上的硬盘序列号如何查
  6. 金蝶K3-航天信息税控发票开票软件接口程序
  7. DELL XPS M1530安装MAC OS X Lion 10.7.3经验分享!
  8. 如何运用创客匠人微信小程序实现引流拓客?
  9. 瑞星杀毒软件linux序列号,瑞星杀毒软件2007光盘零售版是面向 Linux的?
  10. 【Mark工具】一些好用的图片标注工具