Golang面向对象编程(二)

文章目录

  • 封装
    • 基本介绍
    • 封装的实现
    • 工厂函数
  • 继承
    • 基本介绍
    • 继承的实现
    • 字段和方法访问细节
    • 多继承

封装

基本介绍

基本介绍

  • 封装(Encapsulation)是面向对象编程(OOP)中的一种重要概念,封装通过将数据和相关的方法组合在一起,形成一个称为类的抽象数据类型,只暴露必要的接口供外部使用。
  • 封装可以隐藏数据的实际实现细节,外部只能通过公共(public)接口来访问和修改数据,使得代码更加模块化和结构化,同时可以防止不恰当的访问和操作,提高数据的安全性。
  • 封装将相关的数据和方法组织在一起,形成了一个独立的单元,外部使用者只需关心公共接口,无需了解内部实现细节,简化了使用方式,提高了代码的可读性和可维护性。
  • 封装使得内部实现可以独立于外部接口进行修改,如果内部实现发生了变化,只需要确保公共接口的兼容性,而不会影响使用该类的其他代码,提供了更好的灵活性和可扩展性。

封装的实现

封装的实现

  • Go中的封装是通过命名约定和访问控制来实现的,而不像一些其他面向对象语言那样使用访问修饰符(如public、private、protected),因此开发者需要自觉遵守约定来保持封装的效果。
  • Go中通过结构体将相关的字段和方法组合在一起,并通过首字母大小写来控制其可访问性。结构体中的字段和方法使用大写字母开头表示公共的(可导出的),可以被其他包访问,而使用小写字母开头表示私有的(不可导出的),只能在当前包内使用。
  • Go中的封装更加宽泛,其封装的基本单元不是结构体而是包(package),包内的所有标识符(变量、函数、结构体、方法等)都通过首字母大小写来控制其可访问性

封装案例如下:

package model

import "fmt"

type Student struct {
	name   string
	age    int
	gender string
}

// 访问name字段
func (stu Student) GetName() string {
	return stu.name
}
func (stu *Student) SetName(name string) {
	stu.name = name
}

// 访问age字段
func (stu Student) GetAge() int {
	return stu.age
}
func (stu *Student) SetAge(age int) {
	stu.age = age
}

// 访问gender字段
func (stu Student) GetGender() string {
	return stu.gender
}
func (stu *Student) SetGender(gender string) {
	stu.gender = gender
}

// Student的其他方法
func (stu Student) Print() {
	fmt.Printf("student info: <name: %s, age: %d, gender: %s>\n",
		stu.name, stu.age, stu.gender)
}

使用上述包内结构体的案例如下:

package main

import (
	"go_code/OOP2/Encapsulate/model"
)

func main() {
	// var stu = model.Student{"Alice", 12, "女"} // 隐式赋值
	var stu model.Student
	stu.SetName("Alice")
	stu.SetAge(12)
	stu.SetGender("女")
	stu.Print() // student info: <name: Alice, age: 12, gender: 女>
}

注意: Go中无法对结构体中不可导出的字段进行隐式赋值,可以通过给结构体绑定对应的setter和getter方法对不可导出的字段进行访问和赋值。

工厂函数

工厂函数

如果结构体类型的首字母小写(不可导出),那么其他包将无法直接使用该结构体类型来创建实例,这时可以在包内提供对应可导出的工厂函数来创建实例并返回。如下:

package model

import "fmt"

type student struct {
	name   string
	age    int
	gender string
}

// 工厂函数
func NewStudent(name string, age int, gender string) *student {
	return &student{
		name:   name,
		age:    age,
		gender: gender,
	}
}

func (stu student) Print() {
	fmt.Printf("student info: <name: %s, age: %d, gender: %s>\n",
		stu.name, stu.age, stu.gender)
}

其他包通过调用包中可导出的工厂函数,即可创建对应不可导出的结构体实例。如下:

package main

import (
	"go_code/OOP2/Encapsulate/model"
)

func main() {
	var stu = model.NewStudent("Alice", 12, "女")
	stu.Print() // student info: <name: Alice, age: 12, gender: 女>
}

继承

基本介绍

基本介绍

  • 继承是面向对象编程中的一个重要概念,其允许一个类(子类/派生类)继承另一个类(父类/基类)的属性和方法,子类继承父类后可以直接访问和使用父类中字段和方法,同时可以添加自己的字段和方法。
  • 继承的主要优势在于代码复用,继承可以在不重复编写相似代码的情况下扩展现有的类,使代码更具可维护性和可扩展性。

继承示意图如下:

在这里插入图片描述

继承的实现

继承的实现

  • Go中的继承是通过嵌套匿名结构体的方式来实现的,如果一个结构体中嵌套了另一个匿名结构体,那么这个结构体可以直接访问匿名结构体中的字段和方法,从而实现了继承的效果。

继承案例如下:

package main

import (
	"fmt"
)

// 基类
type Person struct {
	Name string
	Age  int
}
func (per Person) PrintInfo() {
	fmt.Printf("name = %s, age = %d\n", per.Name, per.Age)
}

// 派生类
type Student struct {
	Person    // 嵌套匿名结构体
	StudentId int
}
func (stu Student) Study() {
	fmt.Printf("student %d is studying...\n", stu.StudentId)
}

// 派生类
type Teacher struct {
	*Person   // 嵌套匿名结构体指针
	TeacherId int
}
func (tch Teacher) Teach() {
	fmt.Printf("teacher %d is teaching...\n", tch.TeacherId)
}

func main() {
	var stu = Student{Person{"Alice", 12}, 100}
	stu.PrintInfo() // name = Alice, age = 12
	stu.Study()     // student 100 is studying...

	var tch = Teacher{&Person{"Bob", 22}, 200}
	tch.PrintInfo() // name = Bob, age = 22
	tch.Teach()     // teacher 200 is teaching...
}

说明一下:

  • 在嵌套匿名结构体时,可以通过Type的方式嵌套匿名结构体,也可以通过*Type的方式嵌套匿名结构体指针。
  • 在创建结构体变量时,如果要通过字段名的方式初始化结构体字段,那么匿名结构体的字段名由匿名结构体的类型名充当。
  • 在结构体中嵌套匿名结构体后,可以通过结构体实例访问匿名结构体的字段和方法,但在访问时仍然遵循Go的命名约定和访问控制。如果被嵌套的匿名结构体的定义在其他包,那么通过结构体实例只能访问匿名结构体可导出的字段和方法。
  • 结构体中嵌入的匿名字段也可以是基本数据类型,在访问结构体中的匿名基本数据类型字段时,以对应基本数据类型的类型名作为其字段名。比如结构体中嵌入了一个匿名int字段,则通过结构体变量名.int的方式对其进行访问。

组合

在结构体中嵌套有名结构体属于组合关系,在访问组合的结构体字段和方法时,必须带上结构体的字段名。如下:

package main

import (
	"fmt"
)

// 车轮
type Wheel struct {
	Color string
	price int
}

// 自行车
type Bicycle struct {
	FrontWheel Wheel // 组合
	RearWheel  Wheel // 组合
	// ...
}
func (bc Bicycle) Print() {
	fmt.Printf("前轮<color:%s, price:%d元> 后轮<color:%s, price:%d元>\n",
		bc.FrontWheel.Color, bc.FrontWheel.price, bc.RearWheel.Color, bc.RearWheel.price)
}

func main() {
	var bc = Bicycle{
		FrontWheel: Wheel{"black", 100},
		RearWheel:  Wheel{"blue", 200},
	}
	bc.Print() // 前轮<color:black, price:100元> 后轮<color:blue, price:200元>
}

字段和方法访问细节

字段和方法访问细节

结构体字段和方法的访问流程:

  1. 先查看当前结构体类型中是否有对应的字段或方法,如果有则访问。
  2. 再查看结构体中嵌入的匿名结构体中是否有对应的字段或方法,如果有则访问。
  3. 继续查找更深层次嵌入的匿名结构体中是否有对应的字段或方法,如果有则访问,否则产生报错。

案例如下:

package main

import (
	"fmt"
)

// 基类
type Person struct {
	Name string
	Age  int
}
func (per Person) PrintInfo() {
	fmt.Printf("name = %s, age = %d\n", per.Name, per.Age)
}

// 派生类
type Student struct {
	Person
	StudentId int
}

func (stu Student) Study() {
	fmt.Printf("student %d is studying...\n", stu.StudentId)
}
func (stu Student) PrintInfo() {
	fmt.Printf("name = %s, age = %d, id = %d\n", stu.Name, stu.Age, stu.StudentId)
}

func main() {
	var stu = Student{Person{"Alice", 12}, 100}
	fmt.Printf("name = %s\n", stu.Name) // name = Alice
	stu.PrintInfo()                     // name = Alice, age = 12, id = 100
}

代码说明:

  • 在访问字段时,由于Student结构体中没有Name字段,因此Student变量.Name访问的是嵌套的匿名结构体Person中的Name字段。
  • 在访问方法时,由于Student结构体有PrintInfo方法,因此Student变量.PrintInfo()访问的是Student结构体的PrintInfo方法,而不是Person结构体的PrintInfo方法。

多继承

多继承

多继承指的是一个结构体中嵌套了多个匿名结构体,如果嵌套的多个匿名结构体中有相同的字段名或方法名,那么在访问时需要通过匿名结构体的类型名进行指明访问。如下:

package main

import (
	"fmt"
)

type Cat struct {
	Name string
	Age  int
}

type Dog struct {
	Name string
	Age  int
}

// 多继承
type Pet struct {
	Cat
	Dog
}

func main() {
	var pet = Pet{
		Cat: Cat{
			Name: "小猫",
			Age:  2,
		},
		Dog: Dog{
			Name: "小狗",
			Age:  3,
		},
	}
	fmt.Printf("cat name = %s\n", pet.Cat.Name) // cat name = 小猫
	fmt.Printf("dog name = %s\n", pet.Dog.Name) // dog name = 小狗
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/621536.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

RobbitMQ基本消息队列的消息接收

1.先给工程引入依赖 父工程有了子工程就不用导了 <!--AMQP依赖&#xff0c;包含RabbitMQ--> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId> </dependency> 2.配置yml…

基于大数据+Hadoop的豆瓣电子图书推荐系统设计和实现

博主介绍&#xff1a;✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行交流合作✌ 主要内容&#xff1a;SpringBoot、Vue、SSM、HLM…

linux学习:多媒体开发库SDL+视频、音频、事件子系统+处理yuv视频源

目录 编译和移植 视频子系统 视频子系统产生图像的步骤 api 初始化 SDL 的相关子系统 使用指定的宽、高和色深来创建一个视窗 surface 使用 fmt 指定的格式创建一个像素点​编辑 将 dst 上的矩形 dstrect 填充为单色 color​编辑 将 src 快速叠加到 dst 上​编辑 更新…

sqli-labs 第十七关

目录 找注入点&#xff1a; 源码分析&#xff1a; 测试&#xff1a; 奇怪现象&#xff1a; &#xff08;1&#xff09;&#xff1a;当我们输入的密码为字符进行注入时。 &#xff08;2&#xff09;&#xff1a;当我们输入的密码为整数时。 产生原因&#xff1a; 解决方法…

Docker:docker在项目中常用的一些命令

简介   Docker 是一个开源的容器化平台&#xff0c;它允许开发者将应用程序及其依赖项打包到一个可移植的容器中&#xff0c;并发布到任何安装了 Docker 引擎的机器上。这些容器是轻量级的&#xff0c;包含了应用程序运行所需的所有东西&#xff0c;如代码、系统库、系统工具…

SpringBoot集成Redis环境搭建及配置详解

前言 Redis作为当前最火的NoSQL数据库&#xff0c;支持很多语言客户端操作Redis。 而SpringBoot作为java当前最火的开发框架&#xff0c;提供了Spring-data-redis框架实现对Redis的各种操作。 在springboot1.5.x版本的默认的Redis客户端都是Jedis实现的&#xff0c;springboot…

大模型时代下两种few shot高效文本分类方法

介绍近年(2022、2024)大语言模型盛行下的两篇文本分类相关的论文&#xff0c;适用场景为few shot。两种方法分别是setfit和fastfit&#xff0c;都提供了python的包使用方便。 论文1&#xff1a;Efficient Few-Shot Learning Without Prompts 题目&#xff1a;无需提示的高效少…

浪潮信息企业级存储逆势增长 市场份额位列中国前二

2023年&#xff0c;中国企业级存储市场竞争激烈&#xff0c;在挑战重重之下&#xff0c;浪潮信息仍然实现逆势增长&#xff0c;销售额增幅达4.7%&#xff0c;市场份额相比2022年扩大0.6%&#xff0c;位列中国前二。另外&#xff0c;在高端和全闪存阵列细分市场&#xff0c;浪潮…

Vue3实战Easy云盘(三):文件删除+文件移动+目录导航+上传优化/文件过滤/搜索

一、文件删除 &#xff08;1&#xff09;选中了之后才可以删除&#xff0c;没有选中时就显示暗调删除按钮 &#xff08;2&#xff09;实现选中高亮功能 &#xff08;3&#xff09;单个删除 &#xff08;4&#xff09;批量删除 Main.vue中 <!-- 按钮3 --><!-- 如果sel…

鸿蒙内核源码分析(用户态锁篇) | 如何使用快锁Futex(上)

快锁上下篇 鸿蒙内核实现了Futex&#xff0c;系列篇将用两篇来介绍快锁&#xff0c;主要两个原因: 网上介绍Futex的文章很少&#xff0c;全面深入内核介绍的就更少&#xff0c;所以来一次详细整理和挖透。涉及用户态和内核态打配合&#xff0c;共同作用&#xff0c;既要说用户…

【Linux】文件描述符和重定向

目录 一、回顾C文件 二、系统文件I/O 2.1 系统调用 open 2.2 标志位传参 2.3 系统调用 write 2.4 文件描述符fd 2.5 struct file 2.6 fd的分配规则 2.7 重定向 2.7.1 基本原理&#xff1a; 2.7.2 系统调用 dup2 2.8 标准错误 一、回顾C文件 文件 内容 属性 对…

3分钟,学会一个 Lambda 小知识之【流API】

之前给大家介绍的 Lambda 小知识还记得吗&#xff1f;今天再来给大家介绍&#xff0c; 流API 的相关知识要点。 流API Stream是Java8中处理集合的关键抽象概念&#xff0c;它可以指定你对集合的&#xff0c;可以执行查找、过滤和映射等数据操作。 Stream 使用一种类似用 SQ…

资料如何打印更省钱

在日常工作和学习中&#xff0c;我们经常需要打印各种资料。然而&#xff0c;随着打印成本的不断提高&#xff0c;如何更省钱地打印资料成为了大家关注的焦点。今天&#xff0c;就为大家分享一些资料打印的省钱技巧&#xff0c;并推荐一个省钱又省心的打印平台。 首先&#xff…

冥想的时候怎么专注自己

冥想的时候怎么专注自己&#xff1f;我国传统的打坐养生功法&#xff0c;实际最早可追溯到五千年前的黄帝时代。   每天投资两个半小时的打坐&#xff0c;有上千年之久的功效。因为当你们打坐进入永恒时&#xff0c;时间停止了。这不只是两个半小时&#xff0c;而是百千万亿年…

深入探讨黑盒测试:等价类划分与边界值分析

文章目录 概要黑盒测试等价类划分边界值分析 设计测试用例小结 概要 在软件开发领域&#xff0c;测试是确保产品质量的关键步骤之一。而黑盒测试方法作为其中的一种&#xff0c;通过关注输入与输出之间的关系&#xff0c;而不考虑内部实现的细节&#xff0c;被广泛应用于各种软…

使用git系统来更新FreeBSD ports源码

FreeBSD跟其它系统相比一大特色就是ports系统。 The Ports Collection is a set of Makefiles, patches, and description files. Each set of these files is used to compile and install an individual application on FreeBSD, and is called a port. By default, the Po…

5 款免费好用的精品软件推荐!

AI视频生成&#xff1a;小说文案智能分镜智能识别角色和场景批量Ai绘图自动配音添加音乐一键合成视频https://aitools.jurilu.com/ 1.系统优化软件 - Wise Care 365 Wise Care 365 -全球最快的系统优化软件&#xff01;精简系统、管理启动项、清理和优化注册表、清理个人隐私…

基于51单片机的冰箱控制系统设计( proteus仿真+程序+设计报告+原理图+讲解视频)

基于51单片机冰箱控制系统设计( proteus仿真程序设计报告原理图讲解视频&#xff09; 基于51单片机冰箱控制系统设计 1. 主要功能&#xff1a;2. 讲解视频&#xff1a;3. 仿真4. 程序代码5. 设计报告6. 原理图7. 设计资料内容清单&&下载链接资料下载链接&#xff1a; …

Debian mariadb 10.11 XXXX message from server: “Too many connections“

问题表现 报错信息&#xff1a;Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Data source rejected establishment of connection, message from server: "Too many connections" 处理步骤 1、尝试能不能通过终端登录&…

从头理解transformer,注意力机制(下)

交叉注意力 交叉注意力里面q和KV生成的数据不一样 自注意力机制就是闷头自学 解码器里面的每一层都会拿着编码器结果进行参考&#xff0c;然后比较相互之间的差异。每做一次注意力计算都需要校准一次 编码器和解码器是可以并行进行训练的 训练过程 好久不见输入到编码器&…