- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
首先,道歉:我是第一次在这个网站上发帖,所以对于格式或信息错误我深表歉意。我已经看到很多关于从表单上的串行端口获取数据并使用它来填充文本的答案主窗体上的框、图形等,使用“调用”,因为串行端口在不同的线程中运行。
我正在尝试将我们一直使用的一些通信内容“概括”到一个类中(是的,老 VB6 程序员正在努力成长 :-) 但我遇到了问题。如果我在主 program.cs 中强制使用表单名称并为该类使用相同的命名空间,我可以做一些事情,但这有点违背了目的。我还尝试在类中的串行端口的“已接收”上添加一个事件,以在主窗体上引发一个事件。该事件试图引发,但发生跨线程异常。
此时的代码相当大,所以我将尝试“概述”它。在简单的形式中,假设我有一个名为“Form1”的 for,其中包含一个名为 textbox1 的文本框和一个名为“SerialThing”的类:
SerialThing mySerialThing ;
mySerialThing = new SerialThing();
Textbox1.Text = "You Got Data!";
Static SerialPort myDevice;
myDevice = new SerialPort;
myDevice.DataReceived += new SerialDataReceivedEventHandler(devicePort_DataReceived);
this.Invoke(new EventHandler(DisplayData));
如果串口放在主窗体上,上面的方法将起作用,但如果在类内部创建则不起作用。
再次抱歉,如果过于复杂或过于简单。我正在寻找一种“简单”的方法来执行此操作,但要保持类“通用”(理想情况下不必使工作区名称匹配等)。
-文
最佳答案
有很多很多方法可以做到这一点。我将介绍使用自定义事件、委托(delegate)和 Invoke() 的经典方法,因为我认为理解该过程很重要。一旦你了解了这一点,你就可以跳到一些较新的方法。
首先,在您的 SerialThing() 类中,您声明一个自定义事件以在收到数据时传递数据:
class SerialThing
{
public delegate void DataReceivedDelegate(string data);
public event DataReceivedDelegate DataReceived;
static SerialPort myDevice;
public SerialThing()
{
myDevice = new SerialPort();
myDevice.DataReceived += new SerialDataReceivedEventHandler(myDevice_DataReceived);
}
void myDevice_DataReceived(object sender, SerialDataReceivedEventArgs e)
{
// ... grab the data and place into a string called "data" ...
string data = "";
// raise our custom event:
if (DataReceived != null)
{
DataReceived(data);
}
}
}
现在,在 Form1 中,您在创建 SerialThing 实例时订阅了该自定义事件。此外,当收到该事件时,您可以使用 InvokeRequired、Invoke 和委托(delegate)将调用从辅助线程编码到主线程:
public partial class Form1 : Form
{
SerialThing mySerialThing;
public Form1()
{
InitializeComponent();
}
private void Form1_Load(object sender, EventArgs e)
{
mySerialThing = new SerialThing();
mySerialThing.DataReceived += new SerialThing.DataReceivedDelegate(mySerialThing_DataReceived);
}
private delegate void DataReceivedDelegate(string data);
void mySerialThing_DataReceived(string data)
{
if (this.InvokeRequired)
{
this.Invoke(new DataReceivedDelegate(mySerialThing_DataReceived), new Object[] { data });
}
else
{
textBox1.Text = data;
}
}
}
编辑:回应您在下面的评论...
将委托(delegate)简单地视为“指向方法的指针”。当您执行委托(delegate)时,关联的方法就会运行。
InvokeRequired() 部分确定代码是否在与创建控件的线程不同的线程中运行。在这种情况下,控件是表单本身 (this
)。如果返回 true,则事件是在不同的线程中接收到的。然后我们继续 If
block 的真实部分内的 this.Invoke()
行。 this
再次引用表单。因此,表单请求在创建它的线程(主 UI 线程)上调用(“运行”)传递的委托(delegate)。我们创建了一个委托(delegate)实例,它实际上指向我们已经在导致递归调用的相同方法。第二个参数只是一个对象数组,用于将参数与委托(delegate)一起传递。
当运行 Invoke() 时,由于递归调用,我们最终会重新进入该方法。然而此时,InvokeRequired() 检查将返回 false,因为我们现在正在主 UI 线程中运行。因此,我们进入更新 TextBox 的 If
语句的 false 部分。在此模式中,在 If
语句的 else
block 中更新 GUI 控件是安全的。
请注意,这里不需要递归调用。这只是一种风格选择。我们本可以改用委托(delegate)指向的第二个“帮助程序”函数,然后调用它。递归方法减少了所需方法的数量。
这可能是解决此类问题的最冗长的方法。不过,我喜欢它,因为它显示了事件和数据的流动以及线程之间的移动。
我们可以使用匿名委托(delegate)将所有表单代码缩短为:
private void Form1_Load(object sender, EventArgs e)
{
mySerialThing = new SerialThing();
mySerialThing.DataReceived += delegate (string data)
{
this.Invoke((MethodInvoker)(delegate() { textBox1.Text = data; }));
};
}
我不了解你,但作为一名前 VB6 程序员,当你第一次看到这种类型的东西时,这看起来很奇怪。
I've also used components that I know have things running in different threads, yet the "form code" has never had to use the delegate stuff, so maybe there's something that can be buried into the class?
是的,可以在一个类中加入一些“魔法”,这样它就可以在主 UI 线程上引发事件,因此不需要任何 Invoke() 调用。一种方法是使用 SynchronizationContext .
解决此类问题的另一种可能性是使用 BackgroundWorker() 控件,它具有诸如 ProgressChanged() 和 RunWorkerCompleted() 之类的事件,这些事件在主 UI 线程中为您引发(它们执行必要的调用类型的东西为您揭秘)。
关于c# - 从类中的串行端口访问 WinForms 控件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19918341/
我想实现自定义搜索,但遇到了一个麻烦。我需要将 UIButton、SearchBar 组合在一个控件中,以便我可以通过指针引用它。然后我将向该组合控件动态添加更多 UIbutton。最重要的是,我想将
它没有在我的方法中执行 if block 中的语句 母版页:- 页面加载事件:- Control c = new Control(); DoSomething(c); 我的方法:- protecte
ComboBox 控件有一个 setConverter 方法,请参阅 JavaFX ComboBox - Display text but return ID on selection举个例子。我正在
我没有找到任何包含用于标记化(标记)文本输入的控件的 wpf 库。也许我找不到库,因为我错误地调用了这个组件。怎么叫或者哪里有这样的库? 最佳答案 DevExpress WPF 库包含多个数据编辑控件
是否有 Silverlight 控件允许您输入文本并将其突出显示为代码? 例如: foreach (client in Clients){ client.Save();} would become
我有以下用户控件 a) Panel.ZIndex="99999999" 是否是将此控件设置为该控件中 TopMost 的正
是否可以在 Windows 窗体中使用 C# 在窗体加载时隐藏所有特定控件,例如标签或按钮,然后选择显示我不想显示的那些? 我有一个包含很多按钮和标签的程序,但我只想在加载时显示一两个,我觉得对每个标
这个问题已经有答案了: 已关闭11 年前。 Possible Duplicate: Duplicating components at Run-Time 我有一个TMyControl ( Contro
我正在尝试在 Delphi 中编写一个 dll 库,其中包含一个创建 TFrame 后代实例并返回它的函数。但是当我在应用程序中导入这个函数时,每次调用它时,我都会得到一个异常,例如“'xxx'控件没
是否有 Win32 API 调用来确定哪些窗口和/或控件在特定坐标和/或鼠标下可见? 最佳答案 您可以使用GetWindowFromPoint 。它将返回窗口句柄,以便您可以使用 GetClassNa
我需要在编辑控件中输入以下公式: 公式是在 MS Word 中制作的。尝试将其复制/粘贴到编辑控件(单行或多行)后,我得到 M 0.33 Q10T9.1-9.7。 当我输入这个时,我正在研究 Rich
我只是想成功地将它添加到我的窗口中,但这却出奇地困难。 我已经尝试过 #include "windef.h" #include "winbase.h" #include "initguid.h" #i
我希望能够使用 google maps api v3 拥有自己的“街景”按钮。单击按钮时,我希望它根据我的标记经纬度加载街景。然后我希望按钮更改为“返回 map ”,然后再次加载默认 map View
我目前正在用 PHP 开发(另一个)开源 CMS,我想使用 javascript 控件,尤其是管理面板。问题是,是否有任何具有 PHP 接口(interface)的开源、可自由分发的控件(用于创建 j
我为其编写软件的产品之一是会计类应用程序。它用 C++ 编写,使用 C++ Builder 和 VCL 控件,连接到运行在 Linux 上的 PostgreSQL 数据库。 PostgreSQL 数据
我使用 Key Listener 来读取用户的输入,但我遇到了问题。首先,我读到 JTextField“请输入您的姓名”。如果用户输入一个名字,例如 John,它将更改为 John。但是,如果用户输入
我正在尝试对齐数据表列中的复选框(h=center,v=middle) ... 但复选框仍然显示在错误的位置(见附图)
我有一个包含统计信息的 JSON 数据树: { prefix: "a", count: 20, children: [ { prefix: "a:b", c
我在 Photoshop 中设计了一个模型,我打算将它应用到我的产品目录的 ListView 控件中,但它似乎没有正确显示(未对齐?),我希望这里的人可以像我一样指出我的错误试图弄清楚无济于事。 预期
您是使用 ASP.NET 控件还是仅使用带有 CSS 的 HTML? 我在 TextBox 和 DropDownList 的宽度方面遇到了一些问题。在不同的浏览器中,宽度会有所不同,控件的大小也不会相
我是一名优秀的程序员,十分优秀!