几个进程在大多数情况下要包含很多的子线程,那么他们之间免不了的要互相传递很多的参数,那么参数怎么传递的呢?


主线程向子线程传递参数的方法


第一种方法:Thraed类有一个带参数的委托类型的重载形式,这个委托的定义如下:

public delegate void ParameterizedThreadStart(Object obj)

这个Thread类的构造方法的定义如下:

public Thread(ParameterizedThreadStart start);

下面的代码使用了这个带参数的委托向线程传递一个字符串参数:

public static void myStaticParamThreadMethod(Object obj)

{

    Console.WriteLine(obj);

}

 

static void Main(string[] args)

{

    Thread thread = new Thread(myStaticParamThreadMethod);

    thread.Start("通过委托的参数传值");

}

注意这种形式,委托就是Thread要执行的方法,这个委托有一个类的实例对象作为参数。然后在Thread的Start()方法中把这个对象传进去。

如果使用了不带参数的委托,当然也能很正常的启动线程,别学傻了。

第二种方法:定义一个类来传递参数

class Program
{
static void Main(string[] args)
{
MyData myData = new MyData("abcd", );
Thread thread = new Thread(myData.ThreadMethod);
thread.Start(); Console.ReadKey();
}
}//class //定义一个类传递参数
public class MyData
{
private string d1;
private int d2; public MyData(string d1, int d2)
{
this.d1 = d1;
this.d2 = d2;
} public void ThreadMethod()
{
Console.WriteLine(d1);
Console.WriteLine(d2);
}
}//class

这种方法的特点是:子线程的执行入口是在另一个类中,这样正好可以借助这个类的成员函数,给子线程传参。

第三种方法:定义一个新的线程类,让所有的子线程类都继承自这个类

abstract class MyThread

{

    Thread thread = null;

 

    abstract public void run();

 

    public void Start()

    {

        if (thread == null)

        {

            thread = new Thread(run);

            thread.Start();

        }

    }

}

class Utility : MyThread

{

    private string d1;

    private int d2;

 

    public override void run()

    {

        Console.WriteLine(d1);

        Console.WriteLine(d2);

    }

 

}//class

其实上面的两种方法的原理是一样的,这是一个面向数据,一个面向线程。


子线程向主线程传递参数


这里看到是传递参数,也就是说子线程要调用主线程的一个方法,然后把参数传递给主线程的那个方法。说一下方法是前面已经讲过的Invoke和BeginInvoke,但是那时调用主线程的方法并没有传递参数,今天看一下带参数的调用主线程的指定方法。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Threading; namespace WindowsFormsApplication3
{
public partial class Form1 : Form
{
private int count = ; private delegate void DoWorkUIThreadDelegate();
public Form1()
{
InitializeComponent();
} private void btnStart_Click(object sender, EventArgs e)
{
Thread thread = new Thread(ThreadMethod);
thread.IsBackground = true;
thread.Start();
} private void ThreadMethod()
{
while (true)
{
//lblResult.Text = DateTime.Now.ToString();
//这句话不能直接调用,因为子线程不能直接调用UI线程中的控件
if (this.InvokeRequired)
{
this.BeginInvoke(new DoWorkUIThreadDelegate(DoWorkUIThread), null);
}
else
{
DoWorkUIThread();
} //子线程还是可以访问UI线程的普通变量的,只是不能访问控件
//因为普通变量是属于整个类的,属于整个进程的,各个线程时共享的
//对访问共享的数据,加一个lock的锁更加的好
count++;
Thread.Sleep();
} } private void DoWorkUIThread()
{
txtTime.Text = DateTime.Now.ToString() + " " + count;
}
}
}

上面的这种情况似乎用不着子线程给UI线程返回数据,反正子线程可以访问主线程的成员变量。但是另一种情况来了,当子线程不再UI线程所在的类的时候,也就是说子线程在一个工具类中,UI类new出来一个工具类完成一定的工作,UI类可以初始化工具类,但是工具类完成了一定任务后怎么通知UI类呢?现在有这种假设:

UI类要Socket连接网络,现在有一个SocketUtil工具可以完成这项任务,所以UI类就New出来一个SocketUtil,然后调用指定的函数,把IP和Port传递进去。在完成了一些任务以后,SocketUtil要反馈一些信息给UI类。让UI类显示反馈的信息。现在面临两个问题:

1. UI类怎么知道SocketUtil完成了这项任务,然后取显示数据呢?

2. UI类即使知道了什么时候显示信息,那么要显示的内容,UI类怎么知道是什么呢?

第一个问题的解决方法就是事件,利用事件就能在SocketUtil完成一些任务之后,通知UI类接下来怎么做。

第二个问题就是利用Invoke让子线程给主线程调用主线程的函数完成任务的时候,给一些参数。

这里就直接在同一个类中展示一下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Threading; namespace WindowsFormsApplication3
{
public partial class Form1 : Form
{
private int count = ; private delegate void DoWorkUIThreadDelegate(string name, int id);
public Form1()
{
InitializeComponent();
} private void btnStart_Click(object sender, EventArgs e)
{
Thread thread = new Thread(ThreadMethod);
thread.IsBackground = true;
thread.Start();
} private void ThreadMethod()
{
string strName = "stemon";
int ID = ; while (true)
{
//lblResult.Text = DateTime.Now.ToString();
//这句话不能直接调用,因为子线程不能直接调用UI线程中的控件
if (this.InvokeRequired)
{
object[] myArray = new object[];
//类型的装箱是自动的
//类型的拆箱要强制转换
myArray[] = strName;
myArray[] = ID; this.BeginInvoke(new DoWorkUIThreadDelegate(DoWorkUIThread), myArray);
}
else
{
//DoWorkUIThread(strName, ID);
} //子线程还是可以访问UI线程的普通变量的,只是不能访问控件
//因为普通变量是属于整个类的,属于整个进程的,各个线程时共享的
//对访问共享的数据,加一个lock的锁更加的好
count++;
Thread.Sleep();
} } private void DoWorkUIThread(string name, int id)
{
txtTime.Text = name + " " + id + " " + DateTime.Now.ToString() + " " + count;
}
}
}

这个方法是这样操作的,BeginInvoke可以传递一个数组,这个数组是boject类型的,这样就可以把所有的参数都装箱成object类型的,弄成一个object类型的数组,然后到调用方在拆封就可以了。

[C#参考]主线程和子线程之间的参数传递的更多相关文章

  1. (转)C#/.NET主线程与子线程之间的关系

    一般 一个应用程序就对应一个进程,一个进程可有一个或多个线程,而一般有一个主线程.       有的博客上说“至少一个主线程”,这一说法持有怀疑         主线程与子线程之间的关系        ...

  2. C#/.NET主线程与子线程之间的关系

    以前一直没有在程序中写过总结,再翻开程序时却不知所云,所以我决定写总结        一般 一个应用程序就对应一个进程,一个进程可有一个或多个线程,而一般有一个主线程.       有的博客上说“至少 ...

  3. Android 使用handler实现线程间发送消息 (主线程 与 子线程之间)、(子线程 与 子线程之间)

    keyword:Android 使用handler实现线程间发送消息 (主线程 与 子线程之间).(子线程 与 子线程之间) 相信大家平时都有使用到异步线程往主线程(UI线程)发送消息的情况. 本文主 ...

  4. Handler详解系列(四)——利用Handler在主线程与子线程之间互发消息,handler详解

    MainActivity如下: package cc.c; import android.app.Activity; import android.os.Bundle; import android. ...

  5. Handler具体解释系列(四)——利用Handler在主线程与子线程之间互发消息

    MainActivity例如以下: package cc.c; import android.app.Activity; import android.os.Bundle; import androi ...

  6. android 主线程和子线程之间的消息传递

    从主线程发送消息到子线程(准确地说应该是非UI线程)  package com.zhuozhuo; import android.app.Activity; import android.os.Bun ...

  7. 在C#主线程和子线程将数据传递给对方如何实现

    在C#中主线程和子线程怎样实现互相传递数据 老帅 在C#中创建线程Thread时,能够有多种方法,而主线程和子线程之间又怎样实现互相传递数据,每种创建方法传递參数的效果是不同的,逐一看一下:  一.不 ...

  8. Android笔记(三十二) Android中线程之间的通信(四)主线程给子线程发送消息

    之前的例子都是我们在子线程(WorkerThread)当中处理并发送消息,然后在主线程(UI线程)中获取消息并修改UI,那么可以不可以在由主线程发送消息,子线程接收呢?我们按照之前的思路写一下代码: ...

  9. Java并发编程原理与实战六:主线程等待子线程解决方案

    本文将研究的是主线程等待所有子线程执行完成之后再继续往下执行的解决方案 public class TestThread extends Thread { public void run() { Sys ...

随机推荐

  1. c++中,保证头文件只被编译一次,避免多重包含的方法

    保证头文件只被编译一次 #pragma once这是一个比较常用的C/C++杂注,只要在头文件的最开始加入这条杂注,就能够保证头文件只被编译一次. #pragma once是编译器相关的,有的编译器支 ...

  2. google base 之MessagePumpForUI

    base库中比较有意思就是这个类了,如同很多界面库一样,创建了一个隐藏窗口来处理需要在界面线程处理的消息,大体原理也就是需要执行task的时候发送一个自定义的消息,当窗口接收到task的时候调用保存起 ...

  3. C++ Primer 读书笔记: 第8章 标准IO库

    第8章 标准IO库 8.1 面向对象的标准库 1. IO类型在三个独立的头文件中定义:iostream定义读写控制窗口的类型,fstream定义读写已命名文件的类型,而sstream所定义的类型则用于 ...

  4. 加密传输SSL协议6_验证公钥

    如上图所示,我怎么能确定我手里的公钥就是我心中的接收方的公钥呢?怎么防止被钓鱼呢? 解决的办法就是引入一个第三方,一个权威机构,一个我们都相信的机构. 验证公钥,Digital Certificate ...

  5. 创建、更新、删除文档。 --- Mongodb权威指南阅读。

    插入文档: db.foo.insert({ "key" : "value"}); 使用insert插入一个数据,文档中如果没有_id 会自动给文档增加_id. ...

  6. 使用ajax异步提交表单

    虽然这篇文章的标题是提交表单,但是主要的难点在于使用ajax提交文本域的内容, 在工作中的经常会需要ajax跨域的问题,通常的需求使用jsonp就可以得到解决,但是当前项目中有一个图片服务器,客户端需 ...

  7. node anyproxy ssi简易支持

    在项目中,ssi include是一个比较常用的功能,这样我们就可以通过web服务器的支持,将公用的html提取出来,改一个文件就会修改全部内容 但是这也带来了问题,在开发的时候没办法的刷新查看,需要 ...

  8. python退格、方向键无法正常使用解决方法

    CentOS 6.5 自带的Python 2.6.6 箭头以及退格键(Backspace)可正常使用: 自定义所安装的Python 2.7.6却发现箭头以及退格键(Backspace)在使用的时候出现 ...

  9. Linux SSH 互信

    第一步: 创建用于身份认证的两个密钥文件 ssh-keygen #注明.想省事的话打完这个命令后一直回车就行了. 第二步: 把公钥上传到目标主机上去 ssh-copy-id -i id_rsa.pub ...

  10. Java面试题之四

     十六.运行时异常与一般异常有何异同 Java提供了两类主要的异常:runtime exception和checked exception. 1.checked exception:这种异常也就是我们 ...