Go 语言接口及使用接口实现链表插入(下)

简介: 4. 接口嵌套5. 类型断言5.1 断言判断5.2 多类型判断6. 使用接口实现链表插入

4. 接口嵌套


  • 示例:

文件读写测试


package main
import "fmt"
//读取的接口
type Reader interface {
  Read()
}
//写入的接口
type Writer interface {
  Writer()
}
//接口的嵌套
type ReadWriter interface {
  Reader
  Writer
}
//文件结构体
type File struct{}
//实现Reader接口
func (f *File) Read() {
  fmt.Println("文件读取")
}
//实现Writer接口
func (f *File) Writer() {
  fmt.Println("文件写入")
}
//定义读写操作函数
func Test(rw ReadWriter) {  //rw为接口变量
  rw.Read()       //使用读写的方法
  rw.Writer()
}
func main() {
  var f File        //定义结构体,初始化文件
  Test(&f)
}
//输出结果如下
文件读取
文件写入


5. 类型断言


  • 作用:因为接口是一般类型,需要明确具体类型的时候就需要使用类型断言

示例


package main
import "fmt"
func main() {
  //定义空接口
  var a interface{}
  var b int
  a = b //a为int类型
  //断言赋值
  fmt.Printf("a= %v, 类型: %T\n", a, a)
  c := a.(int)
  fmt.Printf("c= %v, 类型: %T\n", c, c)
}
//输出结果如下
a= 0, 类型: int
c= 0, 类型: int


5.1 断言判断


package main
import "fmt"
func main() {
  //定义空接口
  var a interface{}
  var b string
  a = b //a为int类型
  //断言赋值
  fmt.Printf("a= %v, 类型: %T\n", a, a)
  c, err := a.(int)
  if err {
    fmt.Printf("c= %v, 类型: %T\n", c, c)
  } else {
    fmt.Println("不是int类型")
  }
}
//输出结果如下
a= , 类型: string
不是int类型
package main
import "fmt"
func Test(t interface{}) {
  //转换类型判断
  v, err := t.(int)
  if !err {
    fmt.Println("type is not int")
    return
  }
  v++
  fmt.Println(v)
}
func main() {
  a := "张三"
  Test(a)
}
//输出结果如下
type is not int


5.2 多类型判断


package main
import "fmt"
func classifier(items ...interface{}) {
  //遍历复杂集合
  for i, v := range items {
    //变量.(type)职能作用在switch语句中,专门用于判断类型
    switch v.(type) {
    case bool:
      fmt.Printf("第 %d 个数据类型是 bool\n", i)
    case int, int32, int64:
      fmt.Printf("第 %d 个数据类型是 int\n", i)
    case float32, float64:
      fmt.Printf("第 %d 个数据类型是 float\n", i)
    case string:
      fmt.Printf("第 %d 个数据类型是 string\n", i)
    default:
      fmt.Printf("第 %d 个数据类型是其他类型\n", i)
    }
  }
}
func main() {
  //传入多种类型参数
  classifier("张三", 3.14, true, 80, nil)
}
//输出结果如下
第 0 个数据类型是 string
第 1 个数据类型是 float
第 2 个数据类型是 bool
第 3 个数据类型是 int
第 4 个数据类型是其他类型


6. 使用接口实现链表插入


package main
import "fmt"
//节点结构体
type LinkNode struct {
    data interface{}
    next *LinkNode
}
//链表结构体
type Link struct {
    head *LinkNode
    tail *LinkNode
}
//从头部插入
func (p *Link) InsertHead(data interface{}) {
    node := &LinkNode{
        data: data,
        next: nil,
    }
    //判断是否为空链表
    if p.head == nil && p.tail == nil {
        p.head = node
        p.tail = node
        return
    }
    //当前节点的next是原头部节点
    node.next = p.head
    //更新头部
    p.head = node
}
//从尾部插入
func (p *Link) InsertTail(data interface{}) {
    node := &LinkNode{
        data: data,
        next: nil,
    }
    //判断是否为空链表
    if p.head == nil && p.tail == nil {
        p.head = node
        p.tail = node
        return
    }
    //原尾部节点的next是当前节点
    p.tail.next = node
    //更新尾部
    p.tail = node
}
//遍历方法
func (p *Link) Req() {
    lp := p.head
    for lp != nil {
        fmt.Println(lp)
        lp = lp.next
    }
}
func main() {
    //定义链表
    var intLink Link
    for i := 0; i < 10; i++ {
        //intLink.InsertHead(i)
        intLink.InsertTail(i)
    }
    intLink.Req()
}
//输出结果如下
&{0 0xc000096078}
&{1 0xc000096090}
&{2 0xc0000960a8}
&{3 0xc0000960c0}
&{4 0xc0000960d8}
&{5 0xc0000960f0}
&{6 0xc000096108}
&{7 0xc000096120}
&{8 0xc000096138}
&{9 <nil>}
相关文章
|
5天前
|
JSON 测试技术 Go
零值在go语言和初始化数据
【7月更文挑战第10天】本文介绍在Go语言中如何初始化数据,未初始化的变量会有对应的零值:bool为`false`,int为`0`,byte和string为空,pointer、function、interface及channel为`nil`,slice和map也为`nil`。。本文档作为指南,帮助理解Go的数据结构和正确使用它们。
53 22
零值在go语言和初始化数据
|
5天前
|
JSON Java Go
Go 语言性能优化技巧
在Go语言中优化性能涉及数字字符串转换(如用`strconv.Itoa()`代替`fmt.Sprintf()`)、避免不必要的字符串到字节切片转换、预分配切片容量、使用`strings.Builder`拼接、有效利用并发(`goroutine`和`sync.WaitGroup`)、减少内存分配、对象重用(`sync.Pool`)、无锁编程、I/O缓冲、正则预编译和选择高效的序列化方法。这些策略能显著提升代码执行效率和系统资源利用率。
42 13
|
1天前
|
Cloud Native Java Go
为什么要学习Go语言?
GO logo的核心理念,即简单胜于复杂。使用现代斜体无衬线字体与三条简单的运动线相结合,形成一个类似于快速运动的两个轮子的标记,传达速度和效率。字母的圆形暗示了GO地鼠的眼睛,创造了一个熟悉的形状,让标记和吉祥物很好地搭配在一起。
12 4
|
5天前
|
设计模式 Go
Go语言设计模式:使用Option模式简化类的初始化
在Go语言中,面对构造函数参数过多导致的复杂性问题,可以采用Option模式。Option模式通过函数选项提供灵活的配置,增强了构造函数的可读性和可扩展性。以`Foo`为例,通过定义如`WithName`、`WithAge`、`WithDB`等设置器函数,调用者可以选择性地传递所需参数,避免了记忆参数顺序和类型。这种模式提升了代码的维护性和灵活性,特别是在处理多配置场景时。
41 8
|
5天前
|
存储 Go
go语言中fmt格式化包和内置函数汇总
【7月更文挑战第10天】本文介绍fmt包和`Errorf`用于创建格式化的错误消息。`fmt`包还涉及一些接口,如`Formatter`、`GoStringer`、`ScanState`、`Scanner`和`Stringer`,支持自定义格式化和输入/输出处理。
17 1
|
5天前
|
Go
go语言中格式化输出的占位符
【7月更文挑战第10天】`fmt` 包在 Go 语言中用于格式化输出,包括不同类型的占位符:%v(默认格式)、%+v(带字段名的结构体)、%#v(Go语法表示)、%T(类型表示)、%%(百分号)。布尔值用%t,整数有%b、%c、%d、%o、%q、%x、%X和%U。浮点数和复数用%b、%e、%E、%f、%g、%G。字符串和字节切片用%s、%q、%x、%X。指针用%p。占位符可配合+、-、#、空格和0进行调整。宽度和精度控制输出格式,例如 %.4g 控制小数精度。Go 没有 `%u`,但无符号整数默认打印为正数。运算符包括逻辑、比较、加减、乘除、移位、按位和按位异或等。
17 1
|
7天前
|
存储 Go 索引
在go语言中自定义泛型的变长参数
【7月更文挑战第8天】在Go语言中,由于官方1.18以前的版本不支持泛型,可以通过空接口和反射模拟泛型。泛型适用于通用数据结构和函数,虽牺牲了一些性能,但提高了代码复用和类型安全性。
41 1
|
3天前
|
安全 Go
Go语言map并发安全,互斥锁和读写锁谁更优?
Go并发编程中,`sync.Mutex`提供独占访问,适合读写操作均衡或写操作频繁的场景;`sync.RWMutex`允许多个读取者并行,适用于读多写少的情况。明智选择锁可提升程序性能和稳定性。示例展示了如何在操作map时使用这两种锁。
6 0
|
3天前
|
安全 Go 开发者
Go语言map并发安全使用的正确姿势
在Go并发编程中,由于普通map不是线程安全的,多goroutine访问可能导致数据竞态。为保证安全,可使用`sync.Mutex`封装map或使用从Go 1.9开始提供的`sync.Map`。前者通过加锁手动同步,后者内置并发控制,适用于多goroutine共享。选择哪种取决于具体场景和性能需求。
6 0
|
3天前
|
存储 安全 Java
Go语言中的map为什么默认不是并发安全的?
Go语言的map默认不保证并发安全,以优化性能和简洁性。官方建议在需要时使用`sync.Mutex`保证安全。从Go 1.6起,并发读写map会导致程序崩溃,鼓励开发者显式处理并发问题。这样做的哲学是让代码更清晰,并避免不必要的性能开销。
4 0