GO语言的进阶之路-面向对象编程
作者:尹正杰
版权声明:原创作品,谢绝转载!否则将追究法律责任。
当你看完这篇文章之时,我可以说你的Golang算是入门了,何为入门?就是你去看Docker 源码能看懂60%的语法结构,因为涉及一些unix的代码可能没有Linux运维基础的同学在学习的时候会很吃力,看起来也会带来一定的难度,如果有时间的话我会给大家解析Docker部门精辟的源码。好了,回归正题吧,我们今天要学习的内容是什么呢?即面向对象编程。当然,不要用屌丝的心态来说:“那要是没对象的还咋编程呢?”,哈哈~正杰要告诉你的是:“此对象非彼对象”,没有“对象”照样编程啊!那么问题来了,到底什么事面向对象编程呢?
一.什么是面向对象编程;
在Golang的对象可以用一句话总结:“面向对象就是将要处理的数据跟函数进行绑定的方法”。
如果你从来没有学过Python的话就是可以理解是class,如果在学习Golang之前从来没有接触过其他语言(比如说我)的话,那么你可以这样理解:“它是一种编程风格,就是把一切东西看成一个个对象,比如人,车,面包,等等,然后把这些对象拥有的属性变量,比如年龄,民族,工作地点,变质期,寿命,还有操作这些属性变量的函数打包成一个类来表示,这个类的一个抽象就是一个对象,比如人这个类包含一些属性,比如年龄,名字,住址等,他还有一些对别人告诉这些属性的功能,比如:说,看,走等!!”。这就是的面向对象的特点!!!
二.为什么要有面向对象编程;
说到面向对象编程,就不得不说一下面向过程编程,我上次跟大家分享过面向过程编程的方法,也就是定义一些函数,减少了代码的重复性,增加了代码的扩展性和易读性等等。而且当大家“啪啪啪”代码敲的起劲的时候突然冒出个面向对象,对它的出现不免会有所疑惑,面向过程已经如此好了,干嘛还要面向对象呢?其实,我们举个例子来说明一下你就知道了。比如让你写一个人的模型,用函数写你要如果实现呢?比如现在要让你写关于:“刘德华,范冰冰,蔡依林”他们三个人的特点,没错,你可以面向过程式编程用函数将他们的特点定义 出来,那么问题来了,如果我想在外部调用“刘德华”或是“范冰冰”的特点该如果还实现呢?用函数可能非常难以实现。因为面向过程式编程虽然不印象实现的功能,但是其复杂度很低,所以,在实现一些功能上可能欠缺点火候。
这个时候,面向对象就思想就出来啦,面向对象就可能很轻松的实现在外部调用“刘德华”或是“范冰冰”的特点。想要了解更多关于面向对象的发展是可以问我的大师兄“百度”,当然也可以问下我的二师兄“谷歌”,在这里不扯太多历史了,大家多去敲一些代码就会体现面向对象的好处,因为Go是一个完全面向对象的语言。例如,它允许基于我们定义的类型的方法,而没有像其他语言一样的装箱/拆箱操作。其实接触面向对象编程起初大家都是拒绝的,都有抵触心理。但是时间一长你就知道哪个才是你想要的!就好像你整天和你的男性朋友玩的很愉快,突然有一天一个长得非常哇塞的小姐姐出现在你面前,你在你的男性朋友面向说着:“这女孩也就那样,长得还行吧”,然后备注兄弟私下找机会和这个妹子约会,吃饭,看电影是一个道理。
三.如果定义一个对象;
当你看到这里的时候,恭喜你成功被洗脑了,如果你现在还有抵触心理学习面向对象编程的话,建议关闭该网页,因为内心的抵触你在学习这篇博客的内容会很吃力,可能看着看着你就不懂了,忘记之前的面向过程编程,让我们重新学习一种新的编程风格吧。当然,我们也可以对比一下两者的不同,这样学习起来也方便记忆。
下面的案例是:在二维空间中,求两点之间的距离。
1.用函数实现求两点的距离;
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "math" 12 "fmt" 13 ) 14 15 type Point struct { 16 X,Y float64 17 } 18 19 func Distence(p, q Point) float64 { 20 return math.Hypot(q.X-p.X,q.Y-p.Y) //"Hypot"是计算两点之间点距离 21 } 22 23 func main() { 24 p := Point{1,2} 25 q := Point{4,6} 26 fmt.Println(Distence(p,q)) //函数的调用方式,即传值的方式调用。 27 } 28 29 30 31 #以上代码输出结果如下: 32 5
2.用对象实现求两点的距离;
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "math" 12 "fmt" 13 ) 14 15 type Point struct { //定义一个结构题体,你可以理解是是Python中的class 16 X,Y float64 17 } 18 19 func (p Point)Distence(q Point) float64 { //给p对象定义一个Distence的方法,你可以理解绑定了一个Distence的方法。 20 return math.Hypot(q.X-p.X,q.Y-p.Y) 21 } 22 23 func main() { 24 p := Point{1,2} 25 q := Point{4,6} 26 fmt.Println((p.Distence(q))) //类的调用方式,注意,如果定义就要如何调用!(这里是调用p的Distence方法。) 27 } 28 29 30 31 #以上代码输出结果如下: 32 5
当你看了这两段代码你现在可能会反问,实现的效果都是一样的啊,即他们的运行结果都相同,只是在定义和调用的方式不同而已。并不能明显体现出他们的差异性。也比较不出来他们的差别。我只是想说好戏还在后头,我们慢慢来体会它的奥妙之处,现在,我们就知道如何定义了一个对象了吧。那求多个点的长度又该如果定义呢?
3.小试牛刀;
计算多个点的连线的总长度,实现代码如下:
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "math" 12 "fmt" 13 ) 14 15 type Point struct { 16 X,Y float64 17 } 18 19 20 func (p Point)Distence(q Point) float64 { 21 return math.Hypot(q.X-p.X,q.Y-p.Y) 22 } 23 24 func Distence(path []Point) float64 { //定义一个path变了,path其是包含Point类型的数组切片, Slice可以理解为动态增长的数组. 25 var s float64 26 for i := 0;i <len(path) - 1 ; i++ { 27 s += path[i].Distence(path[i+1]) 28 } 29 return s 30 } 31 32 func main() { 33 path := []Point{{10,20},{30,40},{50,60}} 34 fmt.Println(Distence(path)) 35 } 36 37 38 39 #以上代码输出结果如下: 40 56.568542494923804
四.给对象定义一个别名;
玩过linux的朋友可能知道:“alias”这个命令,没错,就是起别名的意思,目的是为了方便用户操作,在Golang里也有可以用type设置别名,这种方法在Go里面随处可见,因为Golang就是一门面向对象编程的语言。
1.Time模块的使用;
也许,您在看官网的时候,会发现time模块,起本质的实现就是基于type的起别名的方法来实现新的功能,将原本的“float64”起别名为“Duration”类型,最后给“Duration”类型绑定特有的方法,这样time也就有了很多中方法,比如Now,String,Second等方法,下面我们来看看time模块常用的方法。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "time" 12 "fmt" 13 ) 14 15 func main() { 16 var n time.Duration //其实"Duration"就是利用别名来实现的。 17 n = 3 * time.Hour + 30 * time.Second //表示3小时又30分钟的时间。 18 fmt.Println(int64(n)) 19 fmt.Println(n.String()) //可读性最高,这是"Duration"特有的方法。 20 fmt.Println(n.Seconds()) 21 fmt.Println(n.Minutes()) 22 23 24 t := time.Now() 25 26 t1 := t.Add(time.Hour) 27 t2 := t.Add(-time.Hour) 28 29 fmt.Println(t1) 30 fmt.Println(t2) 31 fmt.Println(t1.Sub(t2)) //计算时间长度 32 } 33 34 35 36 #以上代码输出结果如下: 37 10830000000000 38 3h0m30s 39 10830 40 180.5 41 2017-07-09 10:49:45.8808815 +0800 CST 42 2017-07-09 08:49:45.8808815 +0800 CST 43 2h0m0s
2.定义一个新类型;
由于Golang的默认编码是中文编码,所以我们才存变了的时候可以用中文来当做变量名,在python3.x版本默认的编码也是utf-8,这一点大家很喜欢。但是我要在这里说的是,可以这么干,但是最好不要这么干,因为在涉及到对象的属性的时候你会遇到一些坑,先不要着急,我在后面的博客中也会分享到关于Golang的公有属性和私有属性。大家可以一起看看我用中文定义的变量名称,大家不要这么干,我这里就是为了方便说明如何定义对象,如果给定义的对象起别名,以及如何调用对象等等。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 12 type 车的属性 struct { 13 名称 string 14 描述 string 15 是否需要备用 bool 16 } 17 18 type 车的特性 []车的属性 //“车的特性”类型是包含结构体“车的属性”类型的数组切片,你可以理解是起了一个别名。 19 20 func (零件 车的特性) 备件方法()(车的信息 车的特性) { //给“车的特性”绑定一个叫“备件”的方法起名为“零件”。 21 for _,part := range 零件{ 22 if part.是否需要备用 { //只将有备胎的车追加到“车的信息”这个空切片中。 23 车的信息 = append(车的信息,part) 24 } 25 } 26 return 车的信息 27 } 28 29 type 汽车 struct { //“汽车”由“车身大小”组成 30 车身大小 string 31 车的特性 //没有给“车的特性”指定一个名称,我们是要保证实现“内嵌”。这样可以提供自动的委托,不需特殊的声明, 32 // 例如“汽车.备件方法()”和“汽车.车的特性.备件方法()”是等同的。 33 } 34 35 36 var ( 37 特斯拉 = 车的特性{ 38 {"Tesla_90D(加速时间)", "100km/2.9s", true}, 39 {"车身大小", "109.47万元", false}, 40 {"颜色", "red", false}, 41 } 42 43 宝马 = 车的特性{ 44 {"BMW M4敞篷轿跑车(加速时间)", "100km/4.4s", true}, 45 {"价格", "1,098,000美元", true}, 46 {"倍耐力轮胎", "兰博基尼Huracan LP580-2前轮原配", true}, 47 {"夏季冰丝汽车坐垫", "1088.00", true}, 48 } 49 50 兰博基尼 = 车的特性{ 51 {"Avetador(加速时间)", "100km/2.8s", true}, 52 {"价格", "648.80-801.15万", true}, 53 {"颜色", "黑色", false}, 54 {"夏季冰丝汽车坐垫", "1088.00", true}, 55 } 56 ) 57 58 59 60 func main() { 61 roadBike := 汽车{车身大小: "5037×2070×mm", 车的特性: 特斯拉} 62 mountainBike := 汽车{车身大小: "1678*1870*1398", 车的特性: 宝马} 63 recumbentBike := 汽车{车身大小: "4780*2030*1136", 车的特性: 兰博基尼} 64 fmt.Println(roadBike.备件方法()) 65 fmt.Println(mountainBike.备件方法()) 66 fmt.Println(recumbentBike.备件方法()) 67 comboParts := 车的特性{} 68 comboParts = append(comboParts, mountainBike.车的特性...) 69 comboParts = append(comboParts, roadBike.车的特性...) 70 comboParts = append(comboParts, recumbentBike.车的特性...) 71 72 fmt.Println(len(comboParts), comboParts[9:]) 73 fmt.Println(comboParts.备件方法()) 74 } 75 76 77 #以上代码执行结果如下: 78 [{Tesla_90D(加速时间) 100km/2.9s true}] 79 [{BMW M4敞篷轿跑车(加速时间) 100km/4.4s true} {价格 1,098,000美元 true} {倍耐力轮胎 兰博基尼Huracan LP580-2前轮原配 true} {夏季冰丝汽车坐垫 1088.00 true}] 80 [{Avetador(加速时间) 100km/2.8s true} {价格 648.80-801.15万 true} {夏季冰丝汽车坐垫 1088.00 true}] 81 11 [{颜色 黑色 false} {夏季冰丝汽车坐垫 1088.00 true}] 82 [{BMW M4敞篷轿跑车(加速时间) 100km/4.4s true} {价格 1,098,000美元 true} {倍耐力轮胎 兰博基尼Huracan LP580-2前轮原配 true} {夏季冰丝汽车坐垫 1088.00 true} {Tesla_90D(加速时间) 100km/2.9s true} {Avetador(加速时间) 100km/2.8s true} {价格 648.80-801.15万 true} {夏季冰丝汽车坐垫 1088.00 true}]
3.小试牛刀;
还记得我们上次用面向过程编程写的一个学员管理系统吗?其实我们也可以稍微用结构体来装饰一下,实现一下功能。代码如下:
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "fmt" 12 "encoding/json" 13 "bufio" 14 "os" 15 "strings" 16 "strconv" 17 "io/ioutil" 18 ) 19 20 type Student struct { //定义一个名称为“Student”的结构统; 21 ID int //定义学员编号 22 NAME string //定义姓名 23 } 24 25 type ClassRoom struct { //定义一个名为“ClassRoom”的结构体; 26 teacher string //定义一个名称为“teacher”字符串类型的变量, 27 students map[string]*Student //定义一个变量名为“students”其类型为map的变量。 28 } 29 30 var classrooms map[string]*ClassRoom //声明一个名为“classrooms”的变量,指定其类型是map,要注意的是map中的value是指针类型的结构统哟。 31 32 var student_num = make(map[int]Student) //定义存取学生成员的函数,注意这里是要初始化的,字典在使用前必须初始化,不然会报错! 33 34 func (s *ClassRoom)Delete() { 35 36 } 37 38 39 func (c *ClassRoom) Add(args []string) error { //自定义添加学生信息的函数 40 if len(args) != 2 { 41 fmt.Println("您输入的字符串有问题,案例:add 01 bingan") 42 return nil 43 } 44 id := args[0] 45 student_name := args[1] 46 student_id, _ := strconv.Atoi(id) 47 48 for _, s := range student_num { 49 if s.ID == student_id { 50 fmt.Println("您输入的ID已经存在,请重新输入") 51 return nil 52 } 53 } 54 student_num[len(student_num)+1] = Student{ student_id,student_name} 55 fmt.Println("Add successfully!!!") 56 return nil 57 } 58 59 func (c *ClassRoom)Drop(args []string)error { 60 if len(args) != 1 { 61 fmt.Println("你愁啥?改学生ID压根就不存在!") 62 return nil 63 } 64 id := args[0] 65 student_id, _ := strconv.Atoi(id) 66 for i, j := range student_num { 67 if j.ID == student_id { 68 delete(student_num, i) //删除map中该id所对应的key值。但是该功能需要完善! 69 fmt.Println("delete successfully!") 70 return nil 71 } 72 } 73 fmt.Println("你愁啥?学生ID压根就不存在!") 74 return nil 75 } 76 77 78 func (c *ClassRoom)Update(args []string)error { //定义修改的函数 79 if len(args) != 2 { 80 fmt.Println("您输入的字符串有问题,案例:add 01 bingan") 81 return nil 82 } 83 id := args[0] //取出ID 84 student_name := args[1] //取出姓名 85 student_id, _ := strconv.Atoi(id) //将字符串ID变成数字类型。 86 for i, j := range student_num { 87 if j.ID == student_id { 88 student_num[i] = Student{ student_id,student_name} //这其实就是一个赋值的过程。 89 fmt.Println("update successfully!") 90 return nil 91 } 92 } 93 fmt.Println("你愁啥?学生ID压根就不存在!") 94 return nil 95 } 96 97 98 func (c *ClassRoom)List(args []string) error{ //给“ClassRoom”绑定一个“List”方法。 99 if len(student_num) == 0 { 100 fmt.Println("数据库为空,请自行添加相关信息!") 101 return nil 102 } 103 for _,value := range student_num{ 104 fmt.Printf("学员的姓名是:\033[31;1m%s\033[0m,学员编号是:\033[31;1m%d\033[0m\n",value.NAME,value.ID) 105 } 106 return nil 107 } 108 109 func (c *ClassRoom)Save(args []string)error { //定义存取的函数 110 if len(args) == 0 { 111 fmt.Println("请输入您想要保存的文件名,例如:save student.txt") 112 return nil 113 } 114 file_name := args[0] 115 f, err := json.Marshal(student_num) //把变量持久化,也就是将内存的变量存到硬盘的时进行的序列化的过程 116 117 if err != nil { 118 fmt.Println("序列化出错啦!") 119 } 120 ioutil.WriteFile(file_name, f, 0644) //将数据写入硬盘,并制定文件的权限。 121 fmt.Println("写入成功") 122 return nil 123 } 124 125 126 func (c *ClassRoom)Load(args []string) error { //定义加载的函数。 127 if len(args) != 1 { 128 fmt.Println("输入错误,请重新输入.") 129 return nil 130 } 131 file_name := args[0] 132 s, _ := ioutil.ReadFile(file_name) 133 json.Unmarshal(s, &student_num) 134 fmt.Println("读取成功!") 135 return nil 136 } 137 138 139 func (c *ClassRoom)Exit(args []string) error { //定义对出的脚本 140 os.Exit(0) //里面的数字表示用户结束程序的返回值,返回0意味着程序是正常结束的。 141 return nil 142 } 143 144 145 146 func main() { 147 classrooms = make(map[string]*ClassRoom) /*初始化字典,因为上面只定义没有初始化。初始化赋值这里不能加":=", 148 因为作用域不同(会将全局作用域的值给覆盖掉),加了得到的结果返回:"null".*/ 149 fmt.Println("学生管理系统迷你版!") 150 f := bufio.NewReader(os.Stdin) //用它读取用户输入的内容 151 for { 152 fmt.Print("请选择您要去的教室") 153 fmt.Print("请输入:>>>") 154 line, _ := f.ReadString('\n') //将读取的内容按照"\n"换行符来切分,注意里面是单引号哟! 155 line = strings.Trim(line, "\n") //表示只脱去换行符:"\n",你可以自定义脱去字符,等效于line = strings.TrimSpace(line) 156 content := strings.Fields(line) //按照空格将得来的字符串做成一个切片。 157 158 if len(content) == 0 { //脱去空格 159 continue 160 } 161 if len(content) == 1 { 162 fmt.Println("您输入的字符串有问题,案例:select yinzhengjie!") 163 continue 164 } 165 ClassRoom_Chose := content[0] //定义执行命令的参数,如add,upadte,list,delete....等等 166 Classroom_len := content[1:] 167 Classroom := Classroom_len[0] //这个就是讲字符串切片转换成字符串。 168 if len(Classroom_len) == 1 { 169 classrooms[Classroom] = &ClassRoom{ 170 students: make(map[string]*Student), 171 } 172 } else { 173 fmt.Println("您输入的参数有问题!") 174 } 175 if ClassRoom_Chose == "select" || ClassRoom_Chose == "SELECT" { 176 fmt.Printf(" 欢迎来到\033[31;1m%s\033[0m教室\n",Classroom) 177 for { 178 fmt.Print("请输入:>>>") 179 line, _ := f.ReadString('\n') //将读取的内容按照"\n"换行符来切分,注意里面是单引号哟! 180 line = strings.Trim(line, "\n") //表示只脱去换行符:"\n",你可以自定义脱去字符,等效于line = strings.TrimSpace(line) 181 content := strings.Fields(line) //按照空格将得来的字符串做成一个切片。 182 if len(content) == 0 { //脱去空格 183 continue 184 } 185 cmd := content[0] //定义执行命令的参数,如add,upadte,list,delete....等等 186 args := content[1:] //定义要执行的具体内容 187 actiondict := map[string]func([]string) error{ //定义用户的输入内容 188 "add": classrooms[Classroom].Add, //表示用户输入的是字符串"add"时,其要执行的结构体的方法是classrooms[Classroom].Add,也就是“Add”方法,以下定义同理。 189 "list": classrooms[Classroom].List, 190 "update": classrooms[Classroom].Update, 191 "delete": classrooms[Classroom].Drop, 192 "save": classrooms[Classroom].Save, 193 "load": classrooms[Classroom].Load, 194 "exit": classrooms[Classroom].Exit, 195 } 196 action_func := actiondict[cmd] //定义用户执行的函数 197 if action_func == nil { //如果输入有问题,告知用户用法 198 fmt.Println("Usage: {add|list|where|load|upadte|delete|}[int][string]") 199 continue 200 } 201 err := action_func(args) 202 if err != nil { 203 fmt.Println("您输入的字符串有问题,案例:add 01 yinzhengjie") 204 continue 205 } 206 continue 207 } 208 } 209 } 210 }
1 [root@yinzhengjie tmp]# go run student_mange.go 2 学生管理系统迷你版! 3 请选择您要去的教室请输入:>>>list 4 您输入的字符串有问题,案例:select yinzhengjie! 5 请选择您要去的教室请输入:>>>select 中国检科院 6 欢迎来到中国检科院教室 7 请输入:>>>list 8 数据库为空,请自行添加相关信息! 9 请输入:>>>add 1 yinzhengjie 10 Add successfully!!! 11 请输入:>>>add 2 liu 12 Add successfully!!! 13 请输入:>>>add 3 wu 14 Add successfully!!! 15 请输入:>>>add 4 han 16 Add successfully!!! 17 请输入:>>> 18 请输入:>>>list 19 学员的姓名是:wu,学员编号是:3 20 学员的姓名是:han,学员编号是:4 21 学员的姓名是:yinzhengjie,学员编号是:1 22 学员的姓名是:liu,学员编号是:2 23 请输入:>>>update 1 尹正杰 24 update successfully! 25 请输入:>>>list 26 学员的姓名是:liu,学员编号是:2 27 学员的姓名是:wu,学员编号是:3 28 学员的姓名是:han,学员编号是:4 29 学员的姓名是:尹正杰,学员编号是:1 30 请输入:>>>delete 1 31 delete successfully! 32 请输入:>>>list 33 学员的姓名是:liu,学员编号是:2 34 学员的姓名是:wu,学员编号是:3 35 学员的姓名是:han,学员编号是:4 36 请输入:>>>save test 37 写入成功 38 请输入:>>>exit 39 You have new mail in /var/spool/mail/root 40 [root@yinzhengjie tmp]# 41 [root@yinzhengjie tmp]# 42 [root@yinzhengjie tmp]# 43 [root@yinzhengjie tmp]# 44 [root@yinzhengjie tmp]# go run student_mange.go 45 学生管理系统迷你版! 46 请选择您要去的教室请输入:>>>select yinzhengjie 47 欢迎来到yinzhengjie教室 48 请输入:>>>list 49 数据库为空,请自行添加相关信息! 50 请输入:>>>load test 51 读取成功! 52 请输入:>>>list 53 学员的姓名是:liu,学员编号是:2 54 学员的姓名是:wu,学员编号是:3 55 学员的姓名是:han,学员编号是:4 56 请输入:>>>EXIT 57 Usage: {add|list|where|load|upadte|delete|}[int][string] 58 请输入:>>>exit 59 [root@yinzhengjie tmp]#
五.指针接受者;
我们知道如何定义一个对象,以及如何定义对象的方法,那么问题来了,如果我想要修改一个对象的公有属性那该如何处理呢?这个时候我们就得用到指针类型了,学习Golan的时候我们不得不对指针要了解啊,不过Golang语言中的类和其他语言(C++,Java,Python)还真不一样,因为他没有继承的概念,这是很多程序员较好的一点,但是事实证明它的缺点还是不少的,后续会跟大家分享。先看看我们的例子吧。
1.案列一,对象的类型要定义指针类型;
如果类型传的不是指针类型,也就是将Point类型去掉的话,那么其方法是不能修改该类型的源数据的,而是单独开辟了一块内存。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 /* 11 1>.我们希望改变对象的成员; 12 2>.等价于函数的第一个参数是指针; 13 */ 14 15 import ( 16 "fmt" 17 ) 18 19 type Point struct { 20 X,Y float64 21 } 22 23 24 func (p *Point) ScaleBy(factor float64) { //想要修改p的值就得传指针类型"*Point" 25 p.X *= factor // 等价于:X = X * factor,对对象P进行操作,修改其共有属性。 26 p.Y *= factor 27 } 28 29 func main() { 30 //两种调用方式: 31 p := Point{100,200} 32 p.ScaleBy(2) //姿势一:直接调用 33 fmt.Println(p) 34 35 p1 := Point{100,200} //姿势二:声明结构体后再用指针指向 36 p2 :=&p1 //使用结构体调用,再取其内存地址 37 p2.ScaleBy(2) 38 fmt.Println(p2) 39 } 40 41 42 #以上代码执行效果如下: 43 {200 400} 44 &{200 400}
2.原地修改字典的Value;
上面的那个案例就是我们想要修改结构体重的公有属性,需要用到指针,还记得我之前跟大家分享如何修改字典的值吗?http://www.cnblogs.com/yinzhengjie/p/7079626.html(搜索:“结构体的指针”)在这里,我继续给大家一个案例,此时我把value的值修改成一个结构体,方法还是一样的。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 12 type Student struct { 13 ID int 14 NAME string 15 } 16 17 func main() { 18 dict := make(map[int]*Student) 19 dict[1] = &Student{ 20 ID:100, 21 NAME:"yinzhengjie", 22 } 23 24 dict[2] = &Student{ 25 ID:200, 26 NAME:"尹正杰", 27 } 28 29 30 fmt.Println(dict[1]) 31 s := dict[1] 32 s.ID = 100000 //原地修改字典的value. 33 fmt.Println(dict) 34 fmt.Println(dict[1]) 35 } 36 37 38 39 #以上代码执行结果如下: 40 &{100 yinzhengjie} 41 map[1:0xc042044400 2:0xc042044420] 42 &{100000 yinzhengjie}
3.巧说Golang指针类型(*)地址运算符(&)
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 12 type Rec_area struct { 13 height float64 14 width float64 15 } 16 17 func (Area *Rec_area)Size()float64 { 18 return Area.width * Area.height 19 } 20 21 func main() { 22 fmt.Printf("&是取地址符号, 取到Rec_area类型对象的地址为:\033[31;1m%v\033[0m\n ",&Rec_area{100,200}) 23 var s *Rec_area = &Rec_area{10,20} 24 fmt.Printf("*可以表示一个变量是指针类型(*Rec_area是一个指针变量):\033[31;1m%v\033[0m\n",s ) 25 fmt.Printf("*也可以表示指针类型变量所指向的存储单元 ,也就是这个地址所指向的值:\033[31;1m%v\033[0m\n",*s) 26 var id *Rec_area = &Rec_area{10,20} 27 fmt.Printf("查看这个指针变量的地址 , 基本数据类型直接打印地址:\033[31;1m%v\033[0m\n",&id) 28 } 29 30 31 32 #以上代码输出结果如下: 33 &是取地址符号, 取到Rec_area类型对象的地址为:&{100 200} 34 *可以表示一个变量是指针类型(*Rec_area是一个指针变量):&{10 20} 35 *也可以表示指针类型变量所指向的存储单元 ,也就是这个地址所指向的值:{10 20} 36 查看这个指针变量的地址 , 基本数据类型直接打印地址:0xc042056020
六.结构体的公有属性和私有属性;
我们定义了一个结构体其实可以理解是定义了一个类,那么这个结构体本身都有什么属性呢?这个时候我们就得引入两个概念即:结构体的公有属性和私有属性。从名称上来说,公有属性就是共享的,谁都可以调用它的属性,所谓私有属性就是这个类特有的,在外部是不可以被调用的。然后Golang区分公有属性和私有属性的机制就是类的方法是否首字母大写,如果首字母大写的方法就是公有属性,如果首字母小写的话就是私有属性。
还记得我之前给大家演示一个把你本地代码PUSH到GitHub上吗?
(未更新)
七.结构体的绑定方法接口(interface)
刚刚接触到"interface"的小伙伴可能会有些认生,觉得interface并没有说明卵用。赶紧改变你的观念吧!它的功能可强大了,比如我们常用的调试方法(“fmt.Println”)就是用interface实现的。在Golang官网中由很多模块都是用interface实现的。我称之为“大胃王”!因为他可以接受任意的数据类型(当然,接受的这种数据类型得在这个接口中由相应的方法去处理它哟!)。比如指针类型,字符串类型,地址运算符,整型,布尔值啊等等。所以说interface的应用在Golang语言中用法还是很广泛的,其实我们最常用的就是用它来绑定多个方法(method)。好了,可能光这样说你还感受不到接口的方法,接下来,正杰带你一起来体会一下它的喜怒哀乐。
1.为什么需要接口;
比如你要做一个画板程序,其中里面有一个面板类,主要负责绘画功能,然后你就这样定义了这个类,可是在不久将来,你突然发现这个类满足不了你了,然后你又要重新设计这个类,更糟糕是你可能要放弃这个类,那么其他地方可能有引用他,这样修改起来很麻烦,如果你一开始定义一个接口,把绘制功能放在接口里,然后定义类时实现这个接口,然后你只要用这个接口去引用实现它的类就行了,以后要换的话只不过是引用另一个类而已,这样就达到维护方便了。
其实说简单点就是实现归一化处理,实际生产环境中也有不少的案例。比如,不同的机器来自不同的厂商,但是每个厂商都要提供接口和其他实现共享数据。再比如你去电影院买票,想看“摔跤吧,爸爸”这部电影,那么你就得想导购员买票,如果这部电影已经上映了,那么她就会把票卖给你,如果没有上映那就不会卖给你,因为她们的系统里没有影片信息(你可以理解是没有提供相应的接口)。
2.定义一个空接口;
一个函数把interface{}作为参数,那么他可以接受任意类型的值作为参数,如果一个函数返回interface{},那么也就可以返回任意类型的值。是不是很有用啊!
空interface(interface{})不包含任何的method,正因为如此,所有的类型都实现了空interface。空interface对于描述起不到任何的作用(因为它不包含任何的method),但是空interface在我们需要存储任意类型的数值的时候相当有用,因为它可以存储任意类型的数。接下来,我们一起看下例子就知道了。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 12 var null interface{} // 定义"null"为空接口 13 14 func main() { 15 var i int = 30000 16 s := "Yinzhengjie" 17 null = i // "null"可以存储任意类型的数值,可以给他赋值一个int,也可以给他赋值一个string. 18 fmt.Println(null) 19 null = s 20 fmt.Println(null) 21 } 22 23 24 25 #以上代码执行结果如下: 26 30000 27 Yinzhengjie
3.声明接口;
我们上面学习过了结构体,在其他语言中我们称之为“类”。接口就是程序员根据自己的需求,把自己定义好的类按照相应的需求绑定起来的过程。就好比一根柴夫上山去砍柴,它把干的树木困在用绳子捆绑在一起,把心坎的树木又困在一起,然后把他们扛回家,干的树木就用于生火做饭。湿的木头就用来盖一个草屋。我们写代码的程序员,没错,就是你,就好比那个柴夫,而那些干的,湿的树木就好比你自己定义的类,而柴夫用于困树木的绳子就好比你现在正要学习的接口。绳子可以捆绑各种物体,不仅仅限于树木哟,比如说绳子还可以捆人(千万不要污哟,我就是打个比方)等等。
所以说,一个接口有多大能耐不在于它本身,而在于用它的那个人。任何一门语言也是如此,当你说一门语言low的时候,那说明你可能还没有正真的了解他。
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import ( 11 "fmt" 12 ) 13 type Myinstence interface { //里面绑定一个“Instence” 14 Instence() 15 } 16 type Student struct { //定义一个名称为“Student”的结构体,其只存储字符串。 17 Name string 18 } 19 type Techer struct { //定义一个名称为“Techer”的结构体,其只存储数字。 20 ID float64 21 } 22 type STU Student //给“Student”起个别名为“STU”,用户在调用其类型的时候不能调用“Student”,只能调用“STU”, // 您可以理解成这已经是一个二次开发的 23 24 func (s STU) Instence() { //给“STU”类型的结构体定义一个“Instence”方法,注意,我这里故意没有给对象“s”传递的类型 25 // 是“STU”,而非“*STU”类型,所以这个接口对该对象的“Name”属性的修改是不生效的! 26 s.Name += " Golang" 27 } 28 29 30 func (t *Techer) Instence(factor float64) { //发现没有,我这里是故意给对象“t”也绑定一个“Instence”方法(和“s”的方法同名),但是我传的指针类型即“*Techer”,所以这个接口对该对象的“ID”属性的修改是生效的! 31 t.ID *= factor 32 } 33 34 35 func main() { 36 var Instantiation Myinstence //声明一个变量Instantiation,指定其接口为Myinstence,注意:该接口有Instence方法, 37 // 而我们有2个结构体都用到了该方法,那么到底要执行哪个呢?还得看调用者如何去调用接口了,如果调用符合相应的接口就会去执行相应的方法。 38 39 Instantiation = &STU{Name:"yinzhengjie"} 40 Instantiation.Instence() 41 fmt.Println(Instantiation) //修改并未生效,如果给“s”对象传递的数据类型是“*STU”,结果应该是:&{yinzhengjie Golang} 42 43 p := Techer{100} 44 p.Instence(5) 45 fmt.Println(p) //我们会发现之前给“Techer”传入的参数“100”被修改。 46 } 47 48 49 50 #以上代码执行结果如下: 51 &{yinzhengjie} 52 {500}
4.小试牛刀;
A.定义计算周长和面积的接口;
我们可以通过一个接口,来计算各个形状的面积或者是周长,比如下面我就举了一个很简单的例子,你可以通过一个接口,只要生命一次变量。指定这个变量的类型为你自定义的接口,只要你调用的方式符合你的定义的类型就会触发相应的代码,我们可以一起看下这个案例:
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 var PI float64 = 3.1415926 12 type square struct { //定义正方向边长X 13 X float64 14 } 15 type circle struct { //定义圆形的半径 16 r float64 17 } 18 type Area_Perimeter interface { //定义求面积和边长的接口 19 area() float64 20 perimeter() float64 21 } 22 func (s *square) area() float64 { //给正方向绑定求面积方法 23 return s.X*s.X 24 } 25 func (c *circle) area() float64 { //给圆形绑定求面积的方法 26 return PI * c.r* c.r 27 } 28 29 func (s square) perimeter() float64 { 30 square_perimeter := s.X * 4 31 return square_perimeter 32 } 33 func (c circle) perimeter() float64 { 34 circle_perimeter := 2*PI*c.r 35 return circle_perimeter 36 } 37 38 func main() { 39 var s ,c Area_Perimeter 40 s = &square{10} //通过接口给“square”结构体传值。 41 c = &circle{20} 42 fmt.Printf("正方形的面积是:\033[31;1m%v\033[0m,正方形的周长是:\033[31;1m%v\033[0m\n",s.area(),s.perimeter()) 43 fmt.Printf("圆形的面积是:\033[31;1m%v\033[0m,圆形的周长是:\033[31;1m%v\033[0m\n",c.area(),c.perimeter()) 44 } 45 46 47 48 #以上代码执行结果如下: 49 正方形的面积是:100,正方形的周长是:40 50 圆形的面积是:1256.63704,圆形的周长是:125.663704
B.定义统计文件字节大小的接口;
还记得我上次跟大家分享的“io.Copy”吗?没错,就是用来读取的文件的,我们利用它具有读取文件的特性可以获取文件的字节大小。
1 [root@yinzhengjie tmp]# more num.txt 2 1 3 2 4 3 5 4 6 5 7 6 8 尹正杰 9 10 [root@yinzhengjie tmp]# 11 [root@yinzhengjie tmp]# more test.go 12 /* 13 #!/usr/bin/env gorun 14 @author :yinzhengjie 15 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 16 EMAIL:[email protected] 17 */ 18 19 package main 20 21 import ( 22 "io" 23 "os" 24 "fmt" 25 ) 26 27 type Byte_Counter struct { //定义一个统计字节变量的类。 28 Sum int 29 } 30 31 func (b *Byte_Counter) Write(p []byte)(int, error) { //读取文件字节大小。 32 b.Sum += len(p) 33 return len(p),nil 34 } 35 36 func main() { 37 T := new(Byte_Counter) //此时的T其实是指针,要注意其余make的用法区别,new返回指针,make返回初始化后的(非零)值。 38 // make是引用类型初始化的方法。 39 40 io.Copy(T,os.Stdin) //将格式化输入的东西传给指针T 41 42 fmt.Printf("文件的字节大小为:\033[31;1m%v\033[0m\n",T.Sum) 43 } 44 45 [root@yinzhengjie tmp]# 46 [root@yinzhengjie tmp]# 47 [root@yinzhengjie tmp]# go run test.go < num.txt 48 文件的字节大小为:23 49 [root@yinzhengjie tmp]#
当然,上面的那种写法您如果觉得low的话,可以用下面的方法,两种效果是等效的。
1 [root@yinzhengjie tmp]# more num.txt 2 1 3 2 4 3 5 4 6 5 7 6 8 尹正杰 9 10 You have new mail in /var/spool/mail/root 11 [root@yinzhengjie tmp]# more test.go 12 /* 13 #!/usr/bin/env gorun 14 @author :yinzhengjie 15 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 16 EMAIL:[email protected] 17 */ 18 19 package main 20 21 import ( 22 "io" 23 "os" 24 "fmt" 25 ) 26 27 type Byte_Counter int 28 29 func (b *Byte_Counter) Write(p []byte)(int, error) { 30 *b += Byte_Counter(len(p)) 31 return len(p),nil 32 } 33 34 func main() { 35 T := new(Byte_Counter) //此时的b其实是指针 36 io.Copy(T,os.Stdin) 37 fmt.Printf("文件的字节大小为:\033[31;1m%v\033[0m\n",*T) 38 } 39 [root@yinzhengjie tmp]# 40 [root@yinzhengjie tmp]# 41 [root@yinzhengjie tmp]# go run test.go < num.txt 42 文件的字节大小为:23 43 [root@yinzhengjie tmp]#
C.统计文本的行数和大小;
通过上面的案例,其实我们也可以给他加一个功能,即统计行数的功能。
1 [root@yinzhengjie tmp]# more line_byte_cont.go 2 /* 3 #!/usr/bin/env gorun 4 @author :yinzhengjie 5 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 6 EMAIL:[email protected] 7 */ 8 9 package main 10 11 import ( 12 "io" 13 "os" 14 "fmt" 15 ) 16 17 18 type Byte_Counter struct { 19 Sum int 20 } 21 22 func (b *Byte_Counter) Write(p []byte)(int, error) { //读取文件字节大小。 23 b.Sum += len(p) 24 return len(p),nil 25 } 26 27 28 type Line_Counter struct { 29 Sum int 30 } 31 32 func (L *Line_Counter) Write(p []byte)(int, error) { //用于读取文件行号 33 34 for _,j := range p{ //循环p的内容 35 if j == '\n' { //循环读取每一行,遇到换行符就自加“1”。 36 L.Sum += 1 //由于对象“L”传的的是指针类型(*Line_Counter),换句话说,“L”是指针接受者,最终“L 37 ”的参数会变动。 38 } 39 } 40 return len(p),nil 41 } 42 43 func main() { 44 lines := new(Line_Counter) 45 bytes := new(Byte_Counter) 46 w := io.MultiWriter(lines,bytes) //io模块的“MultiWriter”方法可以接受2个指针类型。将两个Writer(lines,bytes 47 ),合并成单个的Writer。类似于管道,但是他们是有区别的。 48 49 io.Copy(w,os.Stdin) //将用户输入的数据传给w,最终交给lines和lines指针去处理。 50 fmt.Printf("该文本的行号是:\033[31;1m%d\033[0m 行\n",lines.Sum) 51 fmt.Printf("该文本的字节大小是:\033[31;1m%d\033[0m 字节\n",bytes.Sum) 52 } 53 54 [root@yinzhengjie tmp]# 55 [root@yinzhengjie tmp]# 56 [root@yinzhengjie tmp]# more a.txt 57 1 58 2 59 3 60 4 61 5 62 6 63 尹正杰 64 7 65 8 66 9 67 10 68 [root@yinzhengjie tmp]# 69 [root@yinzhengjie tmp]# 70 [root@yinzhengjie tmp]# go run line_byte_cont.go < a.txt 71 该文本的行号是:11 行 72 该文本的字节大小是:31 字节 73 [root@yinzhengjie tmp]# 74 [root@yinzhengjie tmp]#
D.扩展,缓存器的应用;
bytes.buffer是一个缓冲byte类型的缓冲器存放着都是byte 。Buffer 是 bytes 包中的一个 type Buffer struct{…}。
1 [root@yinzhengjie tmp]# more buffer.go 2 /* 3 #!/usr/bin/env gorun 4 @author :yinzhengjie 5 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 6 EMAIL:[email protected] 7 */ 8 9 package main 10 11 import ( 12 "io" 13 "fmt" 14 "bytes" 15 ) 16 17 18 type Byte_Counter struct { 19 Sum int 20 } 21 22 func (b *Byte_Counter) Write(p []byte)(int, error) { //读取文件字节大小。 23 b.Sum += len(p) 24 return len(p),nil 25 } 26 27 28 type Line_Center struct { 29 Sum int 30 } 31 32 func (b *Line_Center) Write(p []byte)(int, error) { //读取文件行号. 33 b.Sum = 1 34 for _,j := range p{ 35 if j == '\n' { 36 b.Sum ++ 37 } 38 } 39 return len(p),nil 40 } 41 42 func main() { 43 l := new(Line_Center) 44 b := new(Byte_Counter) 45 buf := new(bytes.Buffer) //bytes.buffer是一个缓冲byte类型的“Buffer”缓冲器。里面存放着都是byte 类型的数据。 46 buf.WriteString(`yinzhengjie`) //往缓冲器中写入字节类型,注意写入是用的符号哟! 47 w := io.MultiWriter(l,b) //可以理解将l和b方法传给w,也就是说w具有这两种方法去处理数据。 48 io.Copy(w,buf) //将缓存的数据传给w,这样w就可以调用它的方法去执行相应的代码啦。 49 fmt.Printf("该文本的行号是:\033[31;1m%d\033[0m行;\n",l.Sum) 50 fmt.Printf("该文本的字节大小是:\033[31;1m%d\033[0m字节.\n",b.Sum) 51 } 52 [root@yinzhengjie tmp]# 53 [root@yinzhengjie tmp]# more a.txt 54 1 55 2 56 3 57 4 58 5 59 6 60 尹正杰 61 7 62 8 63 9 64 10 65 [root@yinzhengjie tmp]# 66 [root@yinzhengjie tmp]# go run buffer.go < a.txt 67 该文本的行号是:1行; 68 该文本的字节大小是:11字节. 69 [root@yinzhengjie tmp]#
八.实现tar包的归档与压缩(面向接口编程)
tar 包实现了文件的打包功能,可以将多个文件或目录存储到单一的 .tar 文件中。下面让我们一起来实现一个解压tar包的功能吧,具体代码是事例如下:
1 [root@yinzhengjie tmp]# tar cf yinzhengjie.tar test/ 2 [root@yinzhengjie tmp]# rm -rf test/ 3 [root@yinzhengjie tmp]# ll 4 total 20 5 -rw-r--r-- 1 root root 1 Jul 13 11:35 tar.go 6 -rw-r--r-- 1 root root 1769 Jul 13 12:12 untar.go 7 -rw-r--r-- 1 root root 10240 Jul 13 12:21 yinzhengjie.tar 8 [root@yinzhengjie tmp]# 9 [root@yinzhengjie tmp]# more untar.go 10 /* 11 #!/usr/bin/env gorun 12 @author :yinzhengjie 13 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 14 EMAIL:[email protected] 15 */ 16 17 package main 18 19 import ( 20 "archive/tar" 21 "os" 22 "io" 23 "fmt" 24 ) 25 26 func main() { 27 tr := tar.NewReader(os.Stdin) /*从 “*.tar”文件中读出数据是通过“tar.Reader”完成的,所以首先要创建“tar.Reader”。也就是说这个文件是可读类型的。"tar.NewReader"只接受一个io.reader类型。 28 , 29 可以通过“tar.NewReader”方法来创建它,该方法要求提供一个“os.Reader”对象,以便从该对象中读出数据。*/ 30 for { 31 hdr,err := tr.Next() //此时,我们就拥有了一个“tar.Reader”对象 tr,可以用“tr.Next()”来遍历包中的文件 32 。 33 if err != nil { 34 return 35 } 36 fmt.Printf("已解压:\033[31;1m%s\033[0m\n",hdr.Name) 37 //io.Copy(ioutil.Discard,tr) //表示将读取到到内容丢弃,"ioutil.Discard"可以看作是Linux中的:/dev/nul 38 l! 39 info := hdr.FileInfo() // 获取文件信息 40 if info.IsDir() { //判断文件是否为目录 41 os.Mkdir(hdr.Name,0755) //创建目录并赋予权限。 42 continue //创建目录后就要跳过当前循环,继续下一次循环了。 43 } 44 f,_ := os.Create(hdr.Name) //如果不是目录就直接创建该文件 45 io.Copy(f,tr) //最终将读到的内容写入已经创建的文件中去。 46 f.Close() /*不建议写成“defer f.Close()”因为“f.Close()”会将缓存中的数据写入到文件中,同时“f.Close()” 47 还会向“*.tar”文件的最后写入结束信息,如果不关闭“f”而直接退出程序,那么将导致“.tar”文件不完整。而 48 “defer f.Close()”是在函数结束后再执行关闭文件,那么在这个过程中,内存始终会被占用着,浪费这不必要的资 49 源。*/ 50 } 51 } 52 53 [root@yinzhengjie tmp]# 54 [root@yinzhengjie tmp]# go run untar.go < yinzhengjie.tar 55 已解压:test/ 56 已解压:test/yinzhengjie/ 57 已解压:test/yinzhengjie/test4.txt 58 已解压:test/yinzhengjie/test1.txt 59 已解压:test/yinzhengjie/test2.txt 60 已解压:test/yinzhengjie/test5.txt 61 已解压:test/yinzhengjie/test3.txt 62 [root@yinzhengjie tmp]# ll 63 total 24 64 -rw-r--r-- 1 root root 1 Jul 13 11:35 tar.go 65 drwxr-xr-x 3 root root 4096 Jul 13 12:21 test 66 -rw-r--r-- 1 root root 1769 Jul 13 12:12 untar.go 67 -rw-r--r-- 1 root root 10240 Jul 13 12:21 yinzhengjie.tar 68 [root@yinzhengjie tmp]# 69 [root@yinzhengjie tmp]# ls -R test/ 70 test/: 71 yinzhengjie 72 73 test/yinzhengjie: 74 test1.txt test2.txt test3.txt test4.txt test5.txt 75 [root@yinzhengjie tmp]#
tar 本身不具有压缩功能,只能打包文件或目录,那么如果你硬是想要你的代码支持压缩功能其实很简单,只需要添加一行代码,就有如此的功效。同理,如果您想要您的代码支持解密的功能,你也可以先对数据进行解密。然后在解压缩,最宠在交给tar去处理解即可。以上代码优化有如下:
1 [root@yinzhengjie tmp]# ll 2 total 12 3 -rw-r--r-- 1 root root 328 Jul 13 17:12 tar.go 4 drwxr-xr-x 3 root root 4096 Jul 13 22:24 test 5 -rw-r--r-- 1 root root 2071 Jul 13 22:15 untar.go 6 [root@yinzhengjie tmp]# tar -zcf yinzhengjie.tar.gz test/ 7 You have new mail in /var/spool/mail/root 8 [root@yinzhengjie tmp]# 9 [root@yinzhengjie tmp]# mv test/ 111 10 [root@yinzhengjie tmp]# ll 11 total 16 12 drwxr-xr-x 3 root root 4096 Jul 13 22:24 111 13 -rw-r--r-- 1 root root 328 Jul 13 17:12 tar.go 14 -rw-r--r-- 1 root root 2071 Jul 13 22:15 untar.go 15 -rw-r--r-- 1 root root 241 Jul 13 22:31 yinzhengjie.tar.gz 16 [root@yinzhengjie tmp]# 17 [root@yinzhengjie tmp]# more untar.go 18 /* 19 #!/usr/bin/env gorun 20 @author :yinzhengjie 21 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 22 EMAIL:[email protected] 23 */ 24 25 package main 26 27 import ( 28 "archive/tar" 29 "os" 30 "io" 31 "fmt" 32 "compress/gzip" 33 "log" 34 ) 35 36 func main() { 37 uncompress,err := gzip.NewReader(os.Stdin) //讲传入的文件解压传给“uncompress” 38 if err != nil { 39 log.Fatal(err) //意思是当程序解压失败时,就立即终止程序,“log.Fatal”一般用于程序初始化。 40 } 41 tr := tar.NewReader(uncompress) /*从 “*.tar”文件中读出数据是通过“tar.Reader”完成的,所以首先要创建“tar.Reader” 42 , 43 可以通过“tar.NewReader”方法来创建它,该方法要求提供一个“os.Reader”对象,以便从该对象中读出数据。*/ 44 for { 45 hdr,err := tr.Next() //此时,我们就拥有了一个“tar.Reader”对象 tr,可以用“tr.Next()”来遍历包中的文件. 46 47 if err != nil { 48 return 49 } 50 fmt.Printf("已解压:\033[31;1m%s\033[0m\n",hdr.Name) 51 //io.Copy(ioutil.Discard,tr) //表示将读取到到内容丢弃,"ioutil.Discard"可以看作是Linux中的:/dev/null. 52 info := hdr.FileInfo() // 获取文件信息 53 if info.IsDir() { //判断文件是否为目录 54 os.Mkdir(hdr.Name,0755) //创建目录并赋予权限。 55 continue //创建目录后就要跳过当前循环,继续下一次循环了。 56 } 57 f,_ := os.Create(hdr.Name) //如果不是目录就直接创建该文件 58 io.Copy(f,tr) //最终将读到的内容写入已经创建的文件中去。 59 f.Close() /*不建议写成“defer f.Close()”因为“f.Close()”会将缓存中的数据写入到文件中,同时“f.Close()” 60 还会向“*.tar”文件的最后写入结束信息,如果不关闭“f”而直接退出程序,那么将导致“.tar”文件不完整。而 61 “defer f.Close()”是在函数结束后再执行关闭文件,那么在这个过程中,内存始终会被占用着,浪费这不必要的资 62 源。*/ 63 } 64 } 65 [root@yinzhengjie tmp]# 66 [root@yinzhengjie tmp]# go run untar.go < yinzhengjie.tar.gz 67 已解压:test/ 68 已解压:test/yinzhengjie/ 69 已解压:test/yinzhengjie/test4.txt 70 已解压:test/yinzhengjie/test1.txt 71 已解压:test/yinzhengjie/test2.txt 72 已解压:test/yinzhengjie/test5.txt 73 已解压:test/yinzhengjie/test3.txt 74 [root@yinzhengjie tmp]# ll 75 total 20 76 drwxr-xr-x 3 root root 4096 Jul 13 22:24 111 77 -rw-r--r-- 1 root root 328 Jul 13 17:12 tar.go 78 drwxr-xr-x 3 root root 4096 Jul 13 22:31 test 79 -rw-r--r-- 1 root root 2071 Jul 13 22:15 untar.go 80 -rw-r--r-- 1 root root 241 Jul 13 22:31 yinzhengjie.tar.gz 81 [root@yinzhengjie tmp]# ll -R test/ 82 test/: 83 total 4 84 drwxr-xr-x 2 root root 4096 Jul 13 22:31 yinzhengjie 85 86 test/yinzhengjie: 87 total 20 88 -rw-r--r-- 1 root root 10 Jul 13 22:31 test1.txt 89 -rw-r--r-- 1 root root 10 Jul 13 22:31 test2.txt 90 -rw-r--r-- 1 root root 10 Jul 13 22:31 test3.txt 91 -rw-r--r-- 1 root root 10 Jul 13 22:31 test4.txt 92 -rw-r--r-- 1 root root 10 Jul 13 22:31 test5.txt 93 [root@yinzhengjie tmp]#
我们既然知道了如果将一个"*.tar"文件解包,那么如果将一个目录制作成一个tar包呢?我试着写了一下,但是又个小bug,希望大神帮忙指正。
1 [root@yinzhengjie tmp]# ll 2 total 32 3 -rw-r--r-- 1 root root 17850 Jul 14 09:21 startup.cfg 4 -rw-r--r-- 1 root root 3179 Jul 14 14:03 tar.go 5 drwxr-xr-x 3 root root 4096 Jul 14 13:59 test 6 -rw-r--r-- 1 root root 2080 Jul 14 09:21 untar.go 7 [root@yinzhengjie tmp]# 8 [root@yinzhengjie tmp]# more tar.go 9 /* 10 #!/usr/bin/env gorun 11 @author :yinzhengjie 12 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 13 EMAIL:[email protected] 14 */ 15 16 package main 17 18 import ( 19 "fmt" 20 "os" 21 "path/filepath" 22 "archive/tar" 23 "io" 24 ) 25 26 var dir_list,file_list []string //创建两个个动态字符串数组,即切片。用来存取文件和目录。 27 28 func walkFunc(path string, info os.FileInfo, err error) error { /*“walkFunc”可以获取3个参数信息,即:文件的绝对路径, 29 通过“os.FileInfo”获取文件信息,用“err”返回错误信息,最后需要返回一个“error”类型的数据。*/ 30 if info.IsDir() { //判断文件类型如果是目录就把他放在目录的动态数组中, 31 dir_list = append(dir_list,path) 32 }else { //如果不是目录那就按照文件处理,将它放在文件的目录中去。 33 file_list = append(file_list,path) 34 } 35 return nil //返回空值。 36 } 37 38 39 func main() { 40 filepath.Walk(os.Args[2], walkFunc) /*将命令行参数的第三个参数传递给“walkFunc”函数。即用“filepath.Walk”遍历“os.Args[2]”目录下的所有的文件名*/ 41 42 f,err := os.Create(os.Args[1]) //创建一个“*.tar”的文件。 43 if err != nil { 44 fmt.Println(err) 45 return 46 } 47 defer f.Close() //有的小伙伴总是忘记关文件,我们可以用defer关键字帮我们忘记关闭文件的坏习惯。 48 49 tw := tar.NewWriter(f) //向“*.tar”文件中写入数据是通过“tar.Writer”完成的,所以首先要创建“tar.Writer”。我们通过“tar.NewWriter”创建他需要提供一个可写的对象,我们上面创建的文件就得到用处。 50 defer tw.Close() 51 52 for _,d_list := range dir_list{ 53 fileinfo,err := os.Stat(d_list) //获取目录的信息 54 if err != nil{ 55 fmt.Println(err) 56 } 57 hdr,err := tar.FileInfoHeader(fileinfo,"")/*“tar.FileInfoHeader”其实是调用“os.FileInfo ”方法获取文件的信息的,你要知道文件有两个属性, 58 一个是文件信息,比如大小啊,编码格式,修改时间等等,还有一个就是文件内容,就是我们所看到的具体内容。 */ 59 if err != nil { 60 fmt.Println(err) 61 } 62 err = tw.WriteHeader(hdr) //由于是目录,里面的内容我们就不用管理,只记录目录的文件信息。 63 if err != nil { 64 fmt.Println(err) 65 } 66 } 67 for _,f_list := range file_list { 68 fileinfo,err := os.Stat(f_list) //同理,我们将文件也做相应的梳理,获取文件的头部信息,将其传给“tar.Writer”处理。 69 if err != nil { 70 fmt.Println(err) 71 } 72 hdr,err := tar.FileInfoHeader(fileinfo,"") 73 if err != nil { 74 fmt.Println(err) 75 } 76 err = tw.WriteHeader(hdr) 77 if err != nil{ 78 fmt.Println(err) 79 } 80 f1,err := os.Open(f_list) //由于是文件,我们就可以看其内容,将头部信息写入后还是不够的,还需要将具体的内容写进去,这样我们得到的才是一个完整的文件。 81 if err != nil{ 82 fmt.Println(err) 83 } 84 io.Copy(tw,f1) //用io.Copy方法将读到的内容传给“tar.Writer”,让其进行写入到他的对象f中去(也就是“tw := tar.NewWriter(f)”中的“f”) 85 } 86 } 87 [root@yinzhengjie tmp]# 88 [root@yinzhengjie tmp]# go run tar.go yinzhengjie.tar test/ 89 [root@yinzhengjie tmp]# ll 90 total 40 91 -rw-r--r-- 1 root root 17850 Jul 14 09:21 startup.cfg 92 -rw-r--r-- 1 root root 3179 Jul 14 14:03 tar.go 93 drwxr-xr-x 3 root root 4096 Jul 14 13:59 test 94 -rw-r--r-- 1 root root 2080 Jul 14 09:21 untar.go 95 -rw-r--r-- 1 root root 7168 Jul 14 14:04 yinzhengjie.tar 96 [root@yinzhengjie tmp]# mkdir test_tar && mv yinzhengjie.tar test_tar && cd test_tar 97 [root@yinzhengjie test_tar]# ll 98 total 8 99 -rw-r--r-- 1 root root 7168 Jul 14 14:04 yinzhengjie.tar 100 [root@yinzhengjie test_tar]# go run /tmp/untar.go < yinzhengjie.tar 101 已解压:test/ 102 已解压:yinzhengjie/ 103 已解压:test1.txt 104 已解压:test2.txt 105 已解压:test3.txt 106 已解压:test4.txt 107 已解压:test5.txt 108 [root@yinzhengjie test_tar]# ll 109 total 36 110 drwxr-xr-x 2 root root 4096 Jul 14 14:06 test 111 -rw-r--r-- 1 root root 10 Jul 14 14:06 test1.txt 112 -rw-r--r-- 1 root root 10 Jul 14 14:06 test2.txt 113 -rw-r--r-- 1 root root 10 Jul 14 14:06 test3.txt 114 -rw-r--r-- 1 root root 10 Jul 14 14:06 test4.txt 115 -rw-r--r-- 1 root root 10 Jul 14 14:06 test5.txt 116 drwxr-xr-x 2 root root 4096 Jul 14 14:06 yinzhengjie 117 -rw-r--r-- 1 root root 7168 Jul 14 14:04 yinzhengjie.tar 118 [root@yinzhengjie test_tar]# cat test1.txt 119 123你好 120 [root@yinzhengjie test_tar]# cat test2.txt 121 111你好 122 [root@yinzhengjie test_tar]#
扩展:
关于Golang结构体的调用姿势还有很多种,这里就给大家举例出来集中调用方式,看你自己习惯用哪一种,总有一种姿势适合你~哈哈~
1 /* 2 #!/usr/bin/env gorun 3 @author :yinzhengjie 4 Blog:http://www.cnblogs.com/yinzhengjie/tag/GO%E8%AF%AD%E8%A8%80%E7%9A%84%E8%BF%9B%E9%98%B6%E4%B9%8B%E8%B7%AF/ 5 EMAIL:[email protected] 6 */ 7 8 package main 9 10 import "fmt" 11 12 type Student struct { 13 Name string 14 Id int 15 } 16 17 func (s *Student) Update(id int) { 18 s.Id = id 19 } 20 21 func main() { 22 var f func(int) 23 s := Student{Name:"yinzhengjie"} 24 f = s.Update 25 f(200) 26 fmt.Println(s) //静态绑定,只能修改s这个学生 27 28 var f1 func(s *Student,id int) 29 f1 = (*Student).Update 30 f1(&s,300) 31 fmt.Println(s) //动态绑定,我们可以修改s这个学生。 32 33 34 s1 := Student{Name:"尹正杰"} 35 f1(&s1,400) //同时也可以修改 s1这个学生。 36 fmt.Println(s1) //动态绑定,我们可以说是延迟绑定。 37 38 } 39 40 41 42 #以上代码直接结果如下: 43 {yinzhengjie 200} 44 {yinzhengjie 300} 45 {尹正杰 400}