使用wxpython开发跨平台桌面应用,对wxpython控件实现类似C#扩展函数处理的探究
本人之前对C#开发非常喜欢,也从事开发C#开发桌面开发、Web后端、Vue前端应用开发多年,最近一直在研究使用Python,希望能够把C#的一些好的设计模式、开发便利经验引入到Python开发中,很多时候类似的开发方式,可以极大提高我们开发的效率,本篇随笔对wxpython控件实现类似C#扩展函数处理的探究总结。
1、C#扩展函数特点及便利性回顾
C# 的扩展方法具有以下几个特点和便利性:
语法简洁:扩展方法允许在不修改原始类型的情况下,向现有类型添加新功能。调用时看起来像是实例方法。
易于使用:可以在调用扩展方法时使用点语法,这使得代码更易读且更自然。例如,对于字符串类型,可以直接调用扩展方法而不是传递实例。
静态类和静态方法:扩展方法必须定义在静态类中,并且本身也是静态方法。第一个参数指定了要扩展的类型,并使用
this
关键字修饰。提升代码组织性:将相关功能组织到扩展方法中,可以减少主类中的代码量,提高可维护性。
与 LINQ 的结合:扩展方法在 LINQ 中的应用非常广泛,使得集合操作更加直观和简洁。
支持多种数据类型:可以为基本类型、集合类型甚至自定义类型添加扩展方法,从而提供更广泛的功能。
总的来说,扩展方法在提高代码可读性和可维护性方面具有明显的优势,是C#语言设计中的一项重要特性。
我在开发C#Winform应用前端的时候,在自己的公用类库上实现了很多扩展方法,特别是对于一些控件,增加了很多如绑定数据列表、绑定字典大类名称后直接加载数据列表等,以及一些对数据类型的通用处理,如字符串的格式判断或者裁剪等等。
如在随笔《使用扩展函数方式,在Winform界面中快捷的绑定树形列表TreeList控件和TreeListLookUpEdit控件》中介绍过,对于一些常规控件的数据绑定处理。
对于常规的列表绑定,我们可以用简单的一个扩展函数实现,如下所示。
//常规类别绑定
this.txtProjectList4.BindDictItems(list, "Text", "Value", true, columns.ToArray());
定义了扩展方法,就很容易实现数据的绑定,减少涉及控件处理的细节。
那么对于Python如何使用类似的方式实现呢,我们需要对Python 的语言特性进行了解和实际测试下。
2、基于Python实现数据的快速绑定
我们让ChatGPT来进行解答,它给出的代码答案是如下所示。
1)属性绑定方式
import wx # 定义扩展方法
def BindDictItems(self, items_dict):
self.Clear() # 清空现有项
for key, value in items_dict.items():
self.Append(value, key) # 添加项,key 为用户数据,value 为显示内容 # 将扩展方法绑定到 wx.ComboBox 类
setattr(wx.ComboBox, "BindDictItems", BindDictItems) # 测试应用
class MyApp(wx.App):
def OnInit(self):
frame = wx.Frame(None, title="ComboBox BindDictItems Example", size=(300, 200))
panel = wx.Panel(frame) combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10)) # 使用扩展方法绑定字典项
items = {
"item1": "Item One",
"item2": "Item Two",
"item3": "Item Three"
}
combo_box.BindDictItems(items) frame.Show()
return True if __name__ == "__main__":
app = MyApp()
app.MainLoop()
解释
- 定义
BindDictItems
方法:这个方法接受一个字典,将字典中的项绑定到ComboBox
中。- 使用
self.Clear()
清空现有项。 - 遍历字典,并使用
Append
方法将项添加到ComboBox
。
- 使用
- 使用
setattr
绑定方法:通过setattr
将定义的方法添加到wx.ComboBox
类中。 - 测试应用:在
MyApp
中创建一个wx.ComboBox
并调用BindDictItems
方法。
这样,你就可以为 wx.ComboBox
提供一个 BindDictItems
方法。
测试发现,对于功能实现没有问题,但是对于开发的时候,combo_box 没有函数提示,也就是没有强类型的函数提示,这样只是进行了动态绑定,无法在开发的时候,获得IDE的智能提示,很容易写错。
另外也可以使用下面代码实现类似setattr的处理,相当于直接附加方法,同样在使用的时候也是没有函数的智能提示,但是能够正常工作,俗称 猴子补丁 。
# 使用猴子补丁将方法添加到 wx.ComboBox
wx.ComboBox.BindDictItems = bind_dict_items
2)自定义子类方式
对该结果不满意,继续追问实现该功能的其他途径,给出的另一个解决方案是继承父类,在子类实现更多的自定义函数。
为了实现更好的智能提示和类型检查,可以通过使用类型注解和创建一个自定义子类来扩展 wx.ComboBox
。这样可以确保 IDE 提供正确的代码补全和提示。
给出的代码如下所示。
import wx
from typing import Dict class MyComboBox(wx.ComboBox):
def BindDictItems(self, items_dict: Dict[str, str]) -> None:
"""Bind a dictionary of items to the ComboBox."""
self.Clear() # 清空现有项
for key, value in items_dict.items():
self.Append(value, key) # 添加项,key 为用户数据,value 为显示内容 class MyApp(wx.App):
def OnInit(self):
frame = wx.Frame(None, title="ComboBox BindDictItems Example", size=(300, 200))
panel = wx.Panel(frame) combo_box = MyComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10)) # 使用扩展方法绑定字典项
items = {
"item1": "Item One",
"item2": "Item Two",
"item3": "Item Three"
}
combo_box.BindDictItems(items) frame.Show()
return True if __name__ == "__main__":
app = MyApp()
app.MainLoop()
通过这种方式,可以确保在使用 BindDictItems
时能够获得智能提示,从而减少错误。
这样的方式,确实有强类型的函数提示了。虽然可以获得智能提示,但是也就是破坏了界面类的代码,也就是需要修改相关的使用代码,而非C#扩展方法那样,隐式的扩展了。
3)通过中介辅助类的方式实现
有些情况下,不适合继承关系,不可能为每个类都提供一个子类来封装,有时候提供一些辅助类可能更具有性价比。
要在不继承父类的情况下实现类似 C# 的扩展方法,并确保获得智能提示,推荐使用类型注解和一个中介类来包装扩展方法。通过这种方式,IDE 可以识别这些扩展方法并提供智能提示。
创建一个名为 ComboBoxExtensions.py
的文件,其中定义扩展方法。
# combo_box_extensions.py
from typing import Dict
import wx class ComboBoxExtensions:
@staticmethod
def bind_dict_items(combo_box: wx.ComboBox, items_dict: Dict[str, str]) -> None:
"""Bind a dictionary of items to the ComboBox."""
combo_box.Clear() # 清空现有项
for key, value in items_dict.items():
combo_box.Append(value, key) # 添加项,key 为用户数据,value 为显示内容
在主应用程序中,导入扩展类并使用其方法。
import wx
from combo_box_extensions import ComboBoxExtensions # 导入扩展类 class MyApp(wx.App):
def OnInit(self):
frame = wx.Frame(None, title="ComboBox Extensions Example", size=(300, 200))
panel = wx.Panel(frame) combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10)) # 使用扩展方法绑定字典项
items = {
"item1": "Item One",
"item2": "Item Two",
"item3": "Item Three"
}
ComboBoxExtensions.bind_dict_items(combo_box, items) # 这里应该有智能提示 frame.Show()
return True if __name__ == "__main__":
app = MyApp()
app.MainLoop()
ComboBoxExtensions
类包含一个静态方法 bind_dict_items
,该方法接受 wx.ComboBox
实例和字典作为参数。
在主应用程序中,调用 ComboBoxExtensions.bind_dict_items(combo_box, items)
,这将获得智能提示。
4)使用协议类型的方式处理,并在使用的时候转换为协议类
为了确保在不继承的情况下实现扩展方法并获得智能提示,最佳方案是结合类型注解和一个特定的函数注册过程。以下是一个经过验证的方式,确保能够在实例上调用扩展方法,同时获得 IDE 的智能提示。
在 combo_box_extensions.py
中定义扩展函数,并使用 cast
来确保类型正确。
# combo_box_extensions.py
from typing import Dict, Protocol
import wx
from typing import cast class ComboBoxWithBindDictItems(Protocol):
def BindDictItems(self, items_dict: Dict[str, str]) -> None:
... def bind_dict_items(self: wx.ComboBox, items_dict: Dict[str, str]) -> None:
"""Bind a dictionary of items to the ComboBox."""
self.Clear() # 清空现有项
for key, value in items_dict.items():
self.Append(value, key) # 添加项,key 为用户数据,value 为显示内容 # 将扩展方法绑定到 wx.ComboBox
wx.ComboBox.BindDictItems = bind_dict_items
在主应用程序中调用扩展方法,并确保正确使用类型注解。
import wx
from combo_box_extensions import ComboBoxWithBindDictItems # 导入协议类型 class MyApp(wx.App):
def OnInit(self):
frame = wx.Frame(None, title="ComboBox Extensions Example", size=(300, 200))
panel = wx.Panel(frame) combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10)) # 确保类型为 ComboBoxWithBindDictItems,以获得智能提示
cast(ComboBoxWithBindDictItems, combo_box).BindDictItems({
"item1": "Item One",
"item2": "Item Two",
"item3": "Item Three"
}) # 这里应该有智能提示 frame.Show()
return True if __name__ == "__main__":
app = MyApp()
app.MainLoop()
可以看到,通过cast的方式转换后,具有函数代码的智能提示了。
协议类型:定义 ComboBoxWithBindDictItems
,它确保 BindDictItems
方法存在。
使用 cast
:在调用 BindDictItems
方法时,使用 cast
来明确指定 combo_box
的类型为 ComboBoxWithBindDictItems
,这样 IDE 能够识别并提供智能提示。
智能提示:通过类型注解和 cast
,IDE 能够识别扩展方法并提供智能提示。
无继承:避免了复杂的继承结构,同时实现了功能扩展。
以上几种在Python开发中,对于实现C#扩展函数方法的实现,不过总体来说,虽然能够实现类似的方式,却没有C#那种简洁明了,不知道以后Python发展后解决或者是我没有研究透彻的原因,很多时候如果要实现自定义函数的处理方式,估计我只能结合子类继承和辅助类的方式一起解决这个问题。
使用wxpython开发跨平台桌面应用,对wxpython控件实现类似C#扩展函数处理的探究的更多相关文章
- Electron+Vue开发跨平台桌面应用
Electron+Vue开发跨平台桌面应用 xiangzhihong发布于 2019-12-23 虽然B/S是目前开发的主流,但是C/S仍然有很大的市场需求.受限于浏览器的沙盒限制,网页应用无法满足某 ...
- Windows Phone开发(10):常用控件(上)
原文:Windows Phone开发(10):常用控件(上) Windows Phone的控件有几个来源,和传统的桌面应用程序开发或Web开发一样,有默认提供的控件和第三方开者发布的控件.一般而言,如 ...
- winform快速开发平台 -> 快速绑定ComboBox数据控件
通常我们在处理编辑窗体时.往往会遇到数据绑定.例如combobox控件绑定数据字典可能是我们经常用到的.然而在我的winform快速开发平台中我是如何处理这个频繁的操作呢? 首先,我们要绑定combo ...
- winform快速开发平台 -> 基础组件之分页控件
一个项目控件主要由及部分的常用组件,当然本次介绍的是通用分页控件. 处理思想:我们在处理分页过程中主要是针对数据库操作. 一般情况主要是传递一些开始位置,当前页数,和数据总页数以及相关关联的业务逻辑. ...
- iOS开发UI篇—使用picker View控件完成一个简单的选餐应用
iOS开发UI篇—使用picker View控件完成一个简单的选餐应用 一.实现效果 说明:点击随机按钮,能够自动选取,下方数据自动刷新. 二.实现思路 1.picker view的有默认高度为162 ...
- iOS开发UI篇—Quartz2D(自定义UIImageView控件)
iOS开发UI篇—Quartz2D(自定义UIImageView控件) 一.实现思路 Quartz2D最大的用途在于自定义View(自定义UI控件),当系统的View不能满足我们使用需求的时候,自定义 ...
- Windows Phone开发(11):常用控件(下)
原文:Windows Phone开发(11):常用控件(下) WP控件大部分都可以从Silverlight中继承过来,这里我也只能拿一部分作演示,对于其它控件如何使用,可以参考SDK相关说明以及Sil ...
- ASP.NET2.0自定义控件组件开发 第六章 深入讲解控件的属性
原文:ASP.NET2.0自定义控件组件开发 第六章 深入讲解控件的属性 深入讲解控件的属性持久化(一) 系列文章链接: ASP.NET自定义控件组件开发 第一章 待续 ASP.NET自定义控件组件开 ...
- iOS项目开发实战——学会使用TableView列表控件(四)plist读取与Section显示
文本将会实现把数据存储到plist文件里.然后在程序中进行读取.在TableView控件中依据不同的类别显示Section. 有关TableView 的其它实现,请參考<iOS项目开发实战--学 ...
- iOS 9应用开发教程之使用开关滑块控件以及滚动部署视图
iOS 9应用开发教程之使用开关滑块控件以及滚动部署视图 使用ios9中的开关.滑块控件 开关和滑块也是用于和用户进行交互的控件.本节将主要讲解这两种控件. ios9开关 开关控件常用来控制某个功能的 ...
随机推荐
- Java栈溢出|内存泄漏|内存溢出
Java虚拟机栈是线程私有的,它的生命周期和线程同步 一个线程每执行到一个方法,JVM就会创建一个栈帧(用于存储基本数据类型.对象指针和返回值等),并将栈帧压入栈中. 代码示例: public cla ...
- __set_BASEPRI(); 使用不正常
虽然 BASEPRI 是8位寄存器,但是STM32的CortexM3&M4只用了高4位,低四位是没有用到的. __set_BASEPRI(0x0f); // 无效,小于0x0f的值无效 __s ...
- Prometheus部署以及问题解决
Prometheus作用: Prometheus监控(Prometheus Monitoring)是一种开源的系统监控和警报工具.它最初由SoundCloud开发并于2012年发布,并在2016年加入 ...
- Win32 插入符光标跟随的打字小程序
1.先创建插入符光标 在WM_CREATE消息中 LRESULT OnCreate(HWND hWnd, WPARAM wParam, LPARAM lParam) { HDC hdc = GetDC ...
- React挂载dom无效的问题
话不多说,先上代码. 根据我的猜测,ReactDOM.render()这个函数,也就是挂载的意思是将内容进行替换,所以我的vdom1在调试的时候没有展示出来. 然后我创建了两个div块,分别挂载vdo ...
- Angular 18+ 高级教程 – Coding Style Guide 编码风格
前言 Angular 从 v14 开始大改特改,改最多的就是编码风格. 以前是 class first,Decorator first,mutable first. 现在变成了 function fi ...
- Spring —— bean配置
基础配置 别名配置 作用范围配置 适合交给容器进行管理的bean (复用性的对象,无需重复创建的对象) 表现层对象 业务层对象 数据层对象 工具对象 不适合交给容器管 ...
- QT数据可视化框架编程实战之三维曲面图QML组件 使用高度图生成三维曲面图 补天云QT技术培训专家
QT数据可视化框架编程实战之三维曲面图 使用高度图生成三维曲面图 补天云QT技术培训专家 简介 本文将介绍如何使用QT数据可视化框架中的QT三维曲面图QML组件,通过一幅高度图的图片来生成三维曲面图. ...
- 系统编程-操作系统概论PART2
<1> 操作系统提供的服务 <2> 现代操作系统的特征 2.1 微内核(英文中常译作µ-kernel或者micro kernel) 这是一种能够提供必要服务的操作系统内核: ...
- [OI] 二项式期望 DP
OSU OSU yet Another OSU yet yet Another OSU OSU 的题目是这样的:有一些相邻的块,给定每一个块的联通概率,每个连通块对答案有 \(size^{3}\) 的 ...