- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
本人之前对C#开发非常喜欢,也从事开发C#开发桌面开发、Web后端、Vue前端应用开发多年,最近一直在研究使用Python,希望能够把C#的一些好的设计模式、开发便利经验引入到Python开发中,很多时候类似的开发方式,可以极大提高我们开发的效率,本篇随笔对wxpython控件实现类似C#扩展函数处理的探究总结.
C# 的扩展方法具有以下几个特点和便利性:
语法简洁:扩展方法允许在不修改原始类型的情况下,向现有类型添加新功能。调用时看起来像是实例方法.
易于使用:可以在调用扩展方法时使用点语法,这使得代码更易读且更自然。例如,对于字符串类型,可以直接调用扩展方法而不是传递实例.
静态类和静态方法:扩展方法必须定义在静态类中,并且本身也是静态方法。第一个参数指定了要扩展的类型,并使用 this 关键字修饰.
提升代码组织性:将相关功能组织到扩展方法中,可以减少主类中的代码量,提高可维护性.
与 LINQ 的结合:扩展方法在 LINQ 中的应用非常广泛,使得集合操作更加直观和简洁.
支持多种数据类型:可以为基本类型、集合类型甚至自定义类型添加扩展方法,从而提供更广泛的功能.
总的来说,扩展方法在提高代码可读性和可维护性方面具有明显的优势,是C#语言设计中的一项重要特性.
我在开发C#Winform应用前端的时候,在自己的公用类库上实现了很多扩展方法,特别是对于一些控件,增加了很多如绑定数据列表、绑定字典大类名称后直接加载数据列表等,以及一些对数据类型的通用处理,如字符串的格式判断或者裁剪等等.
如在随笔《使用扩展函数方式,在Winform界面中快捷的绑定树形列表TreeList控件和TreeListLookUpEdit控件》中介绍过,对于一些常规控件的数据绑定处理.
对于常规的列表绑定,我们可以用简单的一个扩展函数实现,如下所示.
//常规类别绑定 this.txtProjectList4.BindDictItems(list, "Text", "Value", true, columns.ToArray());
定义了扩展方法,就很容易实现数据的绑定,减少涉及控件处理的细节.
那么对于Python如何使用类似的方式实现呢,我们需要对Python 的语言特性进行了解和实际测试下.
。
我们让ChatGPT来进行解答,它给出的代码答案是如下所示.
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
。
对该结果不满意,继续追问实现该功能的其他途径,给出的另一个解决方案是继承父类,在子类实现更多的自定义函数.
为了实现更好的智能提示和类型检查,可以通过使用类型注解和创建一个自定义子类来扩展 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#扩展方法那样,隐式的扩展了.
有些情况下,不适合继承关系,不可能为每个类都提供一个子类来封装,有时候提供一些辅助类可能更具有性价比.
要在不继承父类的情况下实现类似 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),这将获得智能提示.
为了确保在不继承的情况下实现扩展方法并获得智能提示,最佳方案是结合类型注解和一个特定的函数注册过程。以下是一个经过验证的方式,确保能够在实例上调用扩展方法,同时获得 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#扩展函数处理的探究的文章就讲到这里了,如果你想了解更多关于使用wxpython开发跨平台桌面应用,对wxpython控件实现类似C#扩展函数处理的探究的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
您好,我想知道是否有办法检查是否按住了某个键。 这是一个情况示例 self.button2.Bind(wx.EVT_LEFT_DOWN, self.clickedbutton) def clicked
我正在使用 wxPython 设计一个表单。请注意图片中的搜索面板:“搜索依据:”标签、嵌套在它上面的组合框和搜索控件——它们的文本都与顶部对齐。如何将它们全部对齐(垂直)在中间? 这是搜索面板代码:
我对以下代码有一个我不明白的问题。 为什么是panel1不扩大? 谢谢。 import wx class MyFrame(wx.Frame): def __init__(self, paren
我刚刚开始使用 wxPython。我有以下代码: import wx class SASFrame(wx.Frame): def __init__(self,parent,id,title):
是否可以以列表的形式获取wx.ListCtrl列中的所有值(项目)? 我在文档中看到您可以获取指定的项目,但不能获取整个列,GetValue() 也不起作用,有什么帮助吗? 最佳答案 这是一种方法:
我需要通过单击按钮在 wxpython 中创建一个额外的新窗口(与主窗口在物理上是分开的)。我想在不关闭最后一个窗口的情况下执行此操作。 这是我到目前为止所拥有的: class Prototype(w
我正在创建一个文件对话框,允许用户在我的应用程序中编辑文件后保存文件。我想在对话框中添加一个复选框,以便用户可以选择保存文件的格式。我想我需要创建一些继承自 FileDialog 的新类,并将一个复选
我正在创建一个文件对话框,允许用户在我的应用程序中编辑文件后保存文件。我想在对话框中添加一个复选框,以便用户可以选择保存文件的格式。我想我需要创建一些继承自 FileDialog 的新类,并将一个复选
我已经编写了我的第一个 wxpython 应用程序并且它运行良好,但是我在正确调整主窗口的大小方面遇到了困难。该面板包含一个 wx.BoxSizer,后者又包含一个 wx.FlexGridSizer,
在 wxpython 中使用 SetTransparent(val) 可以使整个框架透明。但是我可以让其中的单个面板透明吗? 我试过使用 panelobj.SetTransparent(val) 但没
我读了这个Style Guide for wxPython code其中说: 3. Don't use IDs. There is very rarely a good reason to use t
我的顶级窗口需要知道自定义控件的内部状态何时发生变化,以便它可以更新 GUI 的各个其他部分。如何在控件中生成自定义事件,以便它可以传播并由顶级窗口处理? 最佳答案 我知道这是一个老问题,但是在 wx
如果这是正确的词,我正在寻找一种方法来获取小部件,例如面板设备上下文。我想获取面板的位图及其所有子小部件,以便我可以在油漆 DC 中操纵它以淡入淡出面板的动画。这是可能的,如果是这样,代码是什么?我希
我在我的 RHEL6 系统上安装了 Anaconda Python。为了运行 gnuradio,我需要 wxPython。出于这个原因,我做到了 conda install wxpython 问题是当
我想要一个只接受数字的文本控件。 (只是整数值,例如 45 或 366) 最好的方法是什么? 最佳答案 我必须做类似的事情,检查字母数字代码。 EVT_CHAR 上的提示是正确的: class Tes
用于 wxPython 的最佳拖放式 GUI 生成器?我试过 wxGlade 但它不是拖放,发现它很难理解,你必须使用 sizer。我更喜欢像 VisualTinker 那样完全拖放的应用程序,但对于
问题:我在 Ubuntu 12.04 上安装 wxPython 时遇到问题。我想找到一种简单直接的方法来做到这一点。 到目前为止我做了什么:到目前为止我能找到的最相关的说明可以在这里找到:http:/
我正在尝试在 wxPython 2.8(使用 Python 2.7)中编写一个 GUI,其中涉及在 sizer 中布置几个 wx.StaticText。我想要的是一个 sizer,如果它检测到当前行的
我有一个只有一个 child 的 wx.Frame。在设置 child 的 wx.Sizer 时,我可以使用 wx.SHAPED 标志,它保持 child 的宽高比锁定。但是,Frame 在纵横比方面
我已经尝试了很多次来找到这个路径但无济于事(链接:How do I import modules in pycharm?)我认为这是因为我使用的是较新版本的 PyCharm,但这是我的项目解释器: 我
我是一名优秀的程序员,十分优秀!