技术报告:Go 中的并发停车模拟器的开发
简介
该项目由使用 Go 开发的并发停车模拟器组成,使用 Fyne 图形库作为用户界面。其目标是实时模拟停车场的行为,同时管理车辆的进出,并直观地显示停车位的更新状态。
该项目结合了并发概念、观察者设计模式和图形界面中的动态渲染。本报告详细介绍了这些工具的使用、遇到的挑战(特别是观察者模式和 Fyne 模式)以及解决方法,旨在为其他开发者提供技术参考。
1.法恩初始化
Fyne 是一个使用 Go 开发图形界面的现代库。基本初始化遵循以下步骤:
- 使用app.New()创建一个新应用程序。
- 使用app.NewWindow()配置主窗口。
- 使用 Fyne 容器和小部件设计内容。
- 调用 ShowAndRun() 来运行应用程序。
在模拟器中,创建了一个主窗口,集成了停车场视图并连接到并发逻辑模型:
func main() { myApp := app.New() mainWindow := myApp.NewWindow("Simulador de Parking") estacionamiento := models.NewEstacionamiento(20) parkingView := views.NewParkingView() mainScene := scenes.NewMainScene(estacionamiento, parkingView) mainWindow.SetContent(parkingView.Container) mainWindow.ShowAndRun() }
这个基本流程有利于业务逻辑和图形界面之间的分离。
2.使用观察者模式
为什么使用观察者模式
观察者模式用于保持模型层和视图层同步。当车辆进入或离开停车场时,模型会通知视图,视图会更新相应的图形元素。此模式非常适合多个组件必须对同一事件做出反应的系统。
在 Go 中使用观察者模式遇到的问题
在 Go 中实现观察者模式可能具有挑战性,特别是对于那些习惯于在 Java 或 C# 等面向对象语言中实现的人来说。在 Go 中使用此模式的一个常见问题是通知观察者时处理并发和死锁。
最初,迭代模型(停车)中注册的观察者来报告事件会导致竞争条件和崩溃。发生这种情况是因为注册新观察者的方法没有得到适当的保护,导致同时访问观察者列表。
怎么解决的
为了解决这个问题,使用了互斥锁(sync.Mutex)来保护对观察者列表的并发访问。此外,还实现了注册观察者和报告事件的安全方法:
func main() { myApp := app.New() mainWindow := myApp.NewWindow("Simulador de Parking") estacionamiento := models.NewEstacionamiento(20) parkingView := views.NewParkingView() mainScene := scenes.NewMainScene(estacionamiento, parkingView) mainWindow.SetContent(parkingView.Container) mainWindow.ShowAndRun() }
项目中完成实施
停车场模型充当可观察主体,而 MainScene 和其他组件(例如图形视图)是观察者:
1.观察者接口定义:
func (e *Estacionamiento) RegistrarObservador(o Observer) { e.mu.Lock() defer e.mu.Unlock() e.observadores = append(e.observadores, o) } func (e *Estacionamiento) NotificarVehiculoEntra(id, cajon, espaciosDisponibles, capacidad int) { e.mu.Lock() defer e.mu.Unlock() for _, o := range e.observadores { o.OnVehiculoEntra(id, cajon, espaciosDisponibles, capacidad) } } func (e *Estacionamiento) NotificarVehiculoSale(id, cajon, espaciosDisponibles, capacidad int) { e.mu.Lock() defer e.mu.Unlock() for _, o := range e.observadores { o.OnVehiculoSale(id, cajon, espaciosDisponibles, capacidad) } }
- 来自模型的事件通知:
package models type Observer interface { OnVehiculoEntra(id, cajon, espaciosDisponibles, capacidad int) OnVehiculoSale(id, cajon, espaciosDisponibles, capacidad int) }
- 观察者回应:
func (e *Estacionamiento) VehiculoEntra(id int) { // Lógica para manejar la entrada del vehículo espaciosDisponibles := e.capacidad - e.ocupados e.NotificarVehiculoEntra(id, cajon, espaciosDisponibles, e.capacidad) } func (e *Estacionamiento) VehiculoSale(id int) { // Lógica para manejar la salida del vehículo espaciosDisponibles := e.capacidad - e.ocupados e.NotificarVehiculoSale(id, cajon, espaciosDisponibles, e.capacidad) }
此解决方案可确保更新一致,并且竞争条件不会影响系统性能。
3.技术问题:渲染与位置计算
上下文
主要的技术挑战是计算图形界面中抽屉的位置并实时更新其颜色。抽屉应该:
- 排成两行,间距均匀。
- 动态改变颜色(红色表示忙碌,黑色表示可用)。
发现的问题
- 动态位置计算:停车位必须排成两排,间距均匀。然而,计算和更新这些位置很复杂,因为它们依赖于无布局容器 (container.NewWithoutLayout()) 内的精确坐标。
- 视觉同步:处理多个并发线程时,尝试实时更新抽屉颜色时会出现视觉不一致。有时更改未反映或导致图形错误。
位置计算
使用绝对坐标来定义初始位置和间距:
func (s *MainScene) OnVehiculoEntra(id, cajon, espaciosDisponibles, capacidad int) { s.View.UpdateState(espaciosDisponibles, capacidad, id, cajon, "entra") } func (s *MainScene) OnVehiculoSale(id, cajon, espaciosDisponibles, capacidad int) { s.View.UpdateState(espaciosDisponibles, capacidad, id, cajon, "sale") }
动态渲染
实现了根据抽屉的状态绘制抽屉的功能:
xStart, yTop, yBottom := float32(185), float32(120), float32(200) spotSpacing := float32(55) // Fila superior for i := 0; i < 10; i++ { parkingSpots = append(parkingSpots, fyne.Position{X: xStart + float32(i)*spotSpacing, Y: yTop}) } // Fila inferior for i := 0; i < 10; i++ { parkingSpots = append(parkingSpots, fyne.Position{X: xStart + float32(i)*spotSpacing, Y: yBottom}) }
视觉同步
为了确保视觉变化与系统状态一致,主标签文本和抽屉状态在中央功能内更新:
func main() { myApp := app.New() mainWindow := myApp.NewWindow("Simulador de Parking") estacionamiento := models.NewEstacionamiento(20) parkingView := views.NewParkingView() mainScene := scenes.NewMainScene(estacionamiento, parkingView) mainWindow.SetContent(parkingView.Container) mainWindow.ShowAndRun() }
这可确保始终提供准确且最新的图形表示。
结论
这个项目不仅实现了模拟并发停车的目标,还面临着实际的开发问题,例如使用观察者模式和使用 Fyne 创建图形界面。遇到的问题和实施的解决方案旨在为其他开始使用 Go 或面临类似挑战的开发人员提供指导。
尤其是 Go 中观察者模式的实现,演示了如何安全高效地处理并发。本报告通过记录这些问题和解决方案,旨在为有兴趣学习和应用这些工具的程序员社区做出贡献,促进他们的学习和开发过程。
如果您对此的实现和解决方案有任何疑问,可以查阅我的 github 存储库:simulador-parking.git
以上是技术报告:Go 中的并发停车模拟器的开发的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

Golang在性能和可扩展性方面优于Python。1)Golang的编译型特性和高效并发模型使其在高并发场景下表现出色。2)Python作为解释型语言,执行速度较慢,但通过工具如Cython可优化性能。

Golang在并发性上优于C ,而C 在原始速度上优于Golang。1)Golang通过goroutine和channel实现高效并发,适合处理大量并发任务。2)C 通过编译器优化和标准库,提供接近硬件的高性能,适合需要极致优化的应用。

goisidealforbeginnersandsubableforforcloudnetworkservicesduetoitssimplicity,效率和concurrencyFeatures.1)installgromtheofficialwebsitealwebsiteandverifywith'.2)

Golang适合快速开发和并发场景,C 适用于需要极致性能和低级控制的场景。1)Golang通过垃圾回收和并发机制提升性能,适合高并发Web服务开发。2)C 通过手动内存管理和编译器优化达到极致性能,适用于嵌入式系统开发。

GoimpactsdevelopmentPositationalityThroughSpeed,效率和模拟性。1)速度:gocompilesquicklyandrunseff,ifealforlargeprojects.2)效率:效率:ITScomprehenSevestAndArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdEcceSteral Depentencies,增强开发的简单性:3)SimpleflovelmentIcties:3)简单性。

C 更适合需要直接控制硬件资源和高性能优化的场景,而Golang更适合需要快速开发和高并发处理的场景。1.C 的优势在于其接近硬件的特性和高度的优化能力,适合游戏开发等高性能需求。2.Golang的优势在于其简洁的语法和天然的并发支持,适合高并发服务开发。

Golang和Python各有优势:Golang适合高性能和并发编程,Python适用于数据科学和Web开发。 Golang以其并发模型和高效性能着称,Python则以简洁语法和丰富库生态系统着称。

Golang和C 在性能上的差异主要体现在内存管理、编译优化和运行时效率等方面。1)Golang的垃圾回收机制方便但可能影响性能,2)C 的手动内存管理和编译器优化在递归计算中表现更为高效。
