GO 学习笔记 (三) : method 和 interface

为了面向对象(首先你得有个对象)

method

定义一个method

type Person struct {
    name string
    age int
}

func (p *Person) growup() {  //想想去掉 * 会怎样?
    p.age += 1
}

func (p Person) getName() string {
    return p.name
}

func (p *Person) setName(name string) {
    p.name = name
}

func main(){
    p := Person{"wq",10}
    println(p, p.name, p.age)
    p.growup()
    println(p.age)
    p.setName("gudqs")
    println(p.name)
}

使用 func ( 方法属于者 方法属于者类型) 方法名 (方法参数列表) (返回参数列表) {..}定义一个方法
传递 指针类型 使结构体值改变
所有自定义类型, 和一些内置类型均可作为方法属于者,即可拥有方法

方法继承

type Person struct {
    name string
    age int
}

type Student struct {
    Person
    no int
    phone string
}

func (p Person) sayhi() {
    println(p.name,p.age)
}

func main(){
    s := Student{Person{"wq",20},1,"110"}
    s.sayhi()
}

利用匿名字段可继承改字段类型的方法 同样的可以直接调用

方法重写

在之前代码 后添加一个 Student 的sayhi方法 , 实现重写

func (s Student) sayhi() {
    println(s.name,s.no,s.phone,s.age)
}

查看执行结果, 如果学过java ,你一定已经露出了纯洁的笑容     : )

TIP

在方法定义时 , 方法属于者类型 为一个指针时, 方法种操作 指针不需要 加 *也可, go会自动加
不能为 int , []int 等类型 添加方法, 但通过自定义类型 ,如 type Int int , 则又可以添加方法 ,因此自定义类型具有高扩展性啊

interface (接口)

定义接口

type UserDao interface {
    add()
    update()
    remove()
}

使用接口

package main

import f "fmt"

type UserDao interface {
    add(u User)
    update(u User)
    remove(u User)
    findAll() []User
}

type User struct {
    name string
    id int
}   

type UserDaoMysqlImpl struct {
    
}

type UserDaoOracleImpl struct {

}

func (dao UserDaoMysqlImpl) add(u User) {
    f.Println("add user[mysql] :",u)
}

func (dao UserDaoMysqlImpl) update(u User) {
    f.Println("update user[mysql] :",u)
}

func (dao UserDaoMysqlImpl) remove(u User) {
    f.Println("remove user[mysql] :",u.id)
}

func (dao UserDaoOracleImpl) add(u User) {
    f.Println("add user[oralce] :",u)
}

func (dao UserDaoOracleImpl) update(u User) {
    f.Println("update user[oracle] :",u)
}

func (dao UserDaoOracleImpl) remove(u User) {
    f.Println("remove user[oracle] :",u.id)
}

func (dao UserDaoMysqlImpl) findAll() []User {
    users := make([]User,3)
    users[0], users[1], users[2] = User{"wq",1}, User{"aa",2}, User{"gudqs",3}
    return users;
}

func (dao UserDaoOracleImpl) findAll() []User {
    users := make([]User,3)
    users[0] = User{"gg",007}
    return users
}

func main(){
    var dao UserDao
    u := User{"qq",88}
    dao =UserDaoOracleImpl{}
    dao.add(u)
    dao.update(u)
    dao.remove(u)
    for i,u := range dao.findAll() {
        f.Println(i,u)
    }
    dao =UserDaoMysqlImpl{}
    dao.add(u)
    dao.update(u)
    dao.remove(u)
    for j,u2 := range dao.findAll() {
        f.Println("mysql:",j,u2)
    }
}

一个东东 具有 某个接口的所有方法, 那么这个东东 可以赋值给这个接口类型的变量
这个特性说白了就是类似多态

接口类型 作为函数(方法) 参数

在上面代码基础上添加 如下代码:

type UserServiceImpl struct {}

func (service UserServiceImpl) add(u User, dao UserDao) {
    print("service do :")
    dao.add(u)
}

func main(){
    service := UserServiceImpl{}
    u := User{"wq",20}
    dao := UserDaoMysqlImpl{}
    service.add(u,dao)
    dao =UserDaoOracleImpl{}
    service.add(u,dao)
}

添加类型UserServiceImpl和对应的add方法, 使用UserDao作为参数, 然后修改main()

方法参数为接口类型, 好处是 可以接受更多的类型 , 只要那种类型有 这个接口的所有方法

空interface

type a interface {}

func main(){
    var i int = 99
    var str string = "wq"
    a = i
    println(a)
    a = str
    println(str)
}

这样 a类型 就可以接受任何类型了, 就像 java 的Object类型一样 !

嵌套 interface

type a interface {
    swap()
    len()
}

type b interface {
    a
    value()
}

然后b就有了 3个方法 , 对的, 想匿名字段继承一样

Comma-ok断言

type a interface {}

type b struct {
    name string
}

func main(){
    list := make([]a,3)
    a[0] = 1
    a[1] = "wq"
    a[2] = b{"wq"}
    for _,element := range list {
        ok,val := element.(b)
        if ok {
            println(" b is a a")
        }
        ok,val = element.(int)
        if ok {
            println("int is a a")
        }
    }
    //solution 2
    for _,ele := range list {
        switch value :=ele.(type) {
            case int :
                println("ele is an int",)
            case b :
                println("ele is a b")
            default :
                println("ele is default type")
        }
    }
}

通过 变量.(类型) 获得返回值 ok 判断 该变量存的是否是 该类型
变量.(type) 仅在switch中 可以使用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,767评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,969评论 19 139
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,211评论 30 472
  • 一直都佩服那些能够坚持的人,不管是坚持早起,还是坚持运动,或者说是坚持任何一件小事,都让我觉得了不起,他们有这样的...
    马达18阅读 213评论 0 0
  • 时间过的好快、还有一周课程就要结束了、看到好多小伙伴都画的好美、再看看自己画的、此处省略一万字。但是我还是很高兴的...
    biubiu_嘿宝阅读 717评论 1 0