- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
背景:
我有一个包含多个操作的类,需要几秒钟以上才能完成。与此同时,我想更新用户界面。所以通常 BackgroundWorker 是要走的路。但出于某种原因,BackGroundWorker 并不总是按照我想要的方式工作(例如:当我尝试将 WebBrowser 与事件一起使用并调用 ReportProgress 事件时,BackgroundWorker 似乎崩溃了)。
所以我通过将 Ui 与主线程分开来避免所有这些。
这个伪代码更好地解释了它:
public Ui ui;
main
{
Thread threadUi = new Thread(initiateUi);
//Initiate and start Thread
//Everything I will do from here on will not have any consequences
//on my ui.
//
//Object Ui can still be publicly accessed, making it possible to
//update the user interface.
}
现在,当我有一个 Bar 类的实例时,我会像这样让 UI 可以访问它:
public Bar bar1;
public Bar bar2;
main
{
//
//other stuff here
//
Thread threadBar1 = //New Thread where I call the Bar initializer function
//and pass bar1 as parameter.
Thread threadBar2 = //idem dito, except with bar2 as parameter
//
//other stuff here
//
}
通过这种设计,我可以使用以下函数从我的用户界面调用 bar1 和 bar2:
Program.bar1.someFunction();
问题:
现在假设我有一个名为 FooHandler 的类。这个类有一个函数在某个 FooDepository 和其他函数中搜索 Foo 的所有实例来操作 Foo 对象。这是一个静态类,因为在我的例子中,它不需要有多个实例。
但是如果我要从 FooHandler 调用一个函数,该函数将在我的 UI 线程中运行,因为那是调用线程(我不太确定,但我找不到关于这个主题的任何文档)。所以我很有可能要面对我开始时遇到的问题。
问题:
是否可以在不使用调用线程的处理能力的情况下访问静态类的函数?
最佳答案
首先:方法范围(定义它的地方)与程序流程无关。定义方法的位置(FooHandler、BarProvider 或 ThreadX)不会影响它的调用位置。实际上方法总是在调用者的线程中被调用。
因为您没有提到任何模型、 View 或 View 模型,而且标题中写着“c#”,所以我假设您在谈论 WinForms。
在 WinForms 中,UI 控件需要从用于创建它们的线程(通常是主线程)调用(更新)。所有 UI 控件都实现了 ISynchronizeInvoke 接口(interface),这就是为了做到这一点。所以,而不是常规的:
progress.Position = 7;
你需要调用Invoke
:
progress.Invoke(new Action(() => progress.Position = 7), null)
因为有很多样板代码,我自己写了一些扩展函数:
public static class ControlExtensions
{
public static void Synchronize(this Control control, Action action)
{
if (control == null || !control.InvokeRequired)
{
action();
}
else
{
control.Invoke(action, null);
}
}
}
现在你可以:
progress.Synchronize(() => progress.Position = 7);
(打字少一点,更容易阅读)
从技术上讲,调用 ISynchronizeTarget 并不真正调用给定的操作。它只是将消息(好的旧 WM_xxxx)放入消息队列(但在调用者的线程中执行),并以委托(delegate)作为参数。然后,如果目标(控件的)线程正在处理消息(在它自己的线程中),它会收到此 WM_xxxx 消息,调用委托(delegate)(在调用者线程中 - 但这次是 UI 线程)并返回。
如果您需要新线程来调用 FooHandler,并且您不想等待,请使用任务(这可能是最简单的方法):
Task.Factory.StartNew(() => FooHandler.SearchOrWhatever(...));
它不会等待(不会阻塞 UI 线程)。
尽管说了这么多,但不要以为它已经完成了。多线程很难。所有那些支持省去打字的构造,但困难的部分仍然存在:死锁、竞争条件、饥饿等。
关于C# 从调用线程外部的静态类访问函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17358979/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!