protected void Image1_ServerClick(object sender, ImageClickEventArgs e)
{
imgLoad.Visible = true;
imgLoad.Style.Add("background-color", "#999");
System.Timers.Timer time = new System.Timers.Timer(5000);
time.AutoReset = false;
time.Elapsed += new ElapsedEventHandler(time_Elapsed);
time.Enabled = true;
time.Start();
} void time_Elapsed(object sender, ElapsedEventArgs e)
{
this.imgLoad.Visible = false;
this.imgLoad.Visible = false;
this.text1.Value = "1111111111111";
}在我的程序里有一个按钮,点击以后显示一张图片,之后停留五秒钟。再让这张图片隐藏掉,在textbox1里写一串文字。
当我调用timer方法的时候,方法执行了,委托也执行了、委托里面的内容也执行了。
但页面就是不送显,
页面上的图片既不消失,也没有文字写入。
如果把这段代码放到
Image1_ServerClick(object sender, ImageClickEventArgs e)
{}
方法里就ok!!
特别奇怪 不知道为什么 !!!
请教高手。。
{
imgLoad.Visible = true;
imgLoad.Style.Add("background-color", "#999");
System.Timers.Timer time = new System.Timers.Timer(5000);
time.AutoReset = false;
time.Elapsed += new ElapsedEventHandler(time_Elapsed);
time.Enabled = true;
time.Start();
} void time_Elapsed(object sender, ElapsedEventArgs e)
{
this.imgLoad.Visible = false;
this.imgLoad.Visible = false;
this.text1.Value = "1111111111111";
}在我的程序里有一个按钮,点击以后显示一张图片,之后停留五秒钟。再让这张图片隐藏掉,在textbox1里写一串文字。
当我调用timer方法的时候,方法执行了,委托也执行了、委托里面的内容也执行了。
但页面就是不送显,
页面上的图片既不消失,也没有文字写入。
如果把这段代码放到
Image1_ServerClick(object sender, ImageClickEventArgs e)
{}
方法里就ok!!
特别奇怪 不知道为什么 !!!
请教高手。。
访问 Windows 窗体控件本质上不是线程安全的。如果有两个或多个线程操作某一控件的状态,则可能会迫使该控件进入一种不一致的状态。还可能出现其他与线程相关的 bug,包括争用情况和死锁。确保以线程安全方式访问控件非常重要。.NET Framework 有助于在以非线程安全方式访问控件时检测到这一问题。在调试器中运行应用程序时,如果创建某控件的线程之外的其他线程试图调用该控件,则调试器会引发一个 InvalidOperationException,并提示消息:“从不是创建控件 control name 的线程访问它。”有三种方法可以从线程访问Win窗体的控件:非线程安全方式;线程安全调用;使用BackgroundWorker进行的线程安全调用。其中,只有线程的安全调用可以宏观并行处理。(另外两种方式都是在线程运行时接受命令,但在线程执行完以后才执行)。一:对Windows窗体控件的非线程安全调用该方式是从辅助线程直接调用。调用应用程序时,调试器会引发一个InvalidOperationException,警告对控件的调用不是线程安全的。可以通过将 CheckForIllegalCrossThreadCalls 属性的值设置为 false 来禁用此异常。这会使控件以与在 Visual Studio 2003 下相同的方式运行。具体做法如下: public Form1()
...{
InitializeComponent();
Control.CheckForIllegalCrossThreadCalls = false;//这一行是关键 用于对线程的不安全调用
}
二:对Windows窗体控件的线程安全调用对窗体控件的线程安全调用需要用委托的方式。主要思路:1、查询控件的 InvokeRequired 属性。2、如果 InvokeRequired 返回 true,则使用实际调用控件的委托来调用 Invoke。3、如果 InvokeRequired 返回 false,则直接调用控件。例子:在TextBox控件中输出相应的信息,SetText为textbox的内容设置方法,SetTextDelegate的委托类型封装 SetText方法。TextBox控件的InvokeRequired返回true是,SetText方法创建SetTextDelegate的一个实例,并调用窗体的Invoke方法。是的SetText方法被创建TextBox控件的线程调用。 // 该事件句柄创建一个对窗体控件线程安全调用的线程private void setTextSafeBtn_Click( object sender, EventArgs e)
...{
this.demoThread =
new Thread(new ThreadStart(this.ThreadProcSafe)); this.demoThread.Start();
}// 该方法在Worker线程中执行并且发出一个线程安全的调用private void ThreadProcSafe()
...{
this.SetText("This text was set safely.");
}//如果被调用的线程和创建的TextBox控件不同,该方法就创建一个SetTextCallback,// 并且用Invoke方法异步调用自己。
// 如果相同,则直接调用方法设置Text的属性。private void SetText(string text)
...{
// 获取的InvokeRequired将调用的线程ID和创建的线程ID向比较。
//如果两个线程ID不同,则返回true if (this.textBox1.InvokeRequired)
...{
SetTextCallback d = new SetTextCallback(SetText);
this.Invoke(d, new object[] ...{ text });
}
else
...{
this.textBox1.Text = text;
}
}
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;namespace sss
{
public partial class timmer : Form
{
public delegate void SetTextCallback(string text);
public timmer()
{
InitializeComponent();
} private void button1_Click(object sender, EventArgs e)
{
imgLoad.Visible = true;
//imgLoad.Style.Add("background-color", "#999");
System.Timers.Timer time = new System.Timers.Timer(5000);
time.AutoReset = false;
time.Elapsed += new System.Timers.ElapsedEventHandler(time_Elapsed);
time.Enabled = true;
time.Start();
}
private void time_Elapsed(object sender, EventArgs e)
{
SetText("11111111111111111111"); }
private void SetText(string text)
{
// 获取的InvokeRequired将调用的线程ID和创建的线程ID向比较。
//如果两个线程ID不同,则返回true if (this.textBox1.InvokeRequired)
{
SetTextCallback d = new SetTextCallback(SetText);
this.Invoke(d, new object[] { text });
}
else
{
this.textBox1.Text = text;
}
} private void timmer_Load(object sender, EventArgs e)
{
//DelegateProgress muy = new DelegateProgress(SetText);
}
}
}
如果不会 网上好多教程 很简单
而且 用处很大