>  기사  >  백엔드 개발  >  C# 다중 스레드 매개변수 전달

C# 다중 스레드 매개변수 전달

黄舟
黄舟원래의
2017-02-17 10:51:301720검색

1. 엔터티 클래스를 통과합니다(여러 매개변수를 전달하고 반환 값을 얻을 수 있음). 데모는 다음과 같습니다.

호출해야 하는 함수 스레드에서:

namespace ThreadParameterDemo
{
    public class FunctionClass
    {
        public static string TestFunction(string name, int age)
        {
            //内部处理省略
            return name + " 的年龄是:" + age;
        }
    }
}

엔티티를 통해 캡슐화됨:

namespace ThreadParameterDemo
{
    /// <summary>
    /// 过渡类
    /// </summary>
    public class TransitionalClass
    {
        private string name = string.Empty;
        private int age;
        public string acceptResults = string.Empty;
        public TransitionalClass(string name, int age)
        {
            this.name = name;
            this.age = age;
        }

        public void TestFunction()
        {
            acceptResults = FunctionClass.TestFunction(this.name, this.age);
        }
    }
}

전화:

아아앙


작은 메모:

IsBackground 문제에 주의해야 합니다. IsBackground가 false인 경우 Windows 프로그램은 종료 시 스레드를 자동으로 종료하지 않습니다. 즉, 귀하의 신청서가 실제로 종료되지 않았습니다.

MSDN 권장 사항: 멀티 스레드 메서드 호출에 매개 변수를 제공하는 가장 좋은 방법은 대상 메서드를 클래스로 래핑하고 제공하는 것입니다. 클래스는 새 스레드의 매개변수로 사용될 필드를 정의합니다.

이 접근 방식의 장점은 새 스레드를 시작하려고 할 때마다 자체 매개변수를 사용하여 클래스의 새 인스턴스를 만들 수 있다는 것입니다.

BackgroundWorker 클래스

ThreadStart의 함수에는 반환 값과 매개 변수가 없습니다

2. 비동기 호출의 매개변수 및 반환 값
매개변수 및 반환 값에 대한 완벽한 솔루션은 비동기 호출을 사용하는 것입니다. Thread와 비교할 때 비동기 호출의 가장 큰 단점 중 하나는 우선 순위를 제어할 수 없다는 것입니다.

구체 코드는 다음과 같습니다.

  private void Form1_Load(object sender, EventArgs e)
        {
            //实例化ThreadWithState类,为线程提供参数  
            TransitionalClass tc = new TransitionalClass(" Jack", 42);
            // 创建执行任务的线程,并执行  
            Thread t = new Thread(new ThreadStart(tc.TestFunction));
            t.Start();
            //获取返回值,通过 tc.acceptResults;  
        }

간체:

        public delegate string delegateFunction(string name,int age);//委托
        delegateFunction df;
        private void Form1_Load(object sender, EventArgs e)
        {
            //指向需要调用的方法
            df = new delegateFunction(FunctionClass.TestFunction);
            string name = "my name";//输入参数 
            int age = 19;
            IAsyncResult result = df.BeginInvoke(name,age, null, null);
            string myResult = df.EndInvoke(result);//用于接收返回值 
            MessageBox.Show(myResult);
        }

작은 메모:

이 방법으로 새 ​​스레드 생성 은 백그라운드(백그라운드)에서 실행 중이고 우선순위는 보통


3. BackgroundWorker를 사용하세요

여러 스레드에서 값을 반환하는 가장 간단한 방법은 BackgroundWorker 구성 요소를 사용하여 스레드를 관리하고 작업이 완료되면 이벤트를 발생시킨 다음 이벤트 핸들러를 사용하여 결과를 처리하는 것입니다.

참고:
BackgroundWorker 구성 요소는 데이터베이스 트랜잭션 및 파일 다운로드와 같이 시간이 많이 걸리는 비동기 작업을 수행하는 데 사용됩니다.
VS를 사용하는 경우 애플리케이션에 BackgroundWorker 인스턴스를 추가하면 도구에서 애플리케이션으로 직접 끌 수 있습니다.

 public Func<string, int, string>  df;//委托
        private void Form1_Load(object sender, EventArgs e)
        {
            //指向需要调用的方法
            df += FunctionClass.TestFunction;
            string name = "my name";//输入参数 
            int age = 19;
            IAsyncResult result = df.BeginInvoke(name, age, null, null);
            string myResult = df.EndInvoke(result);//用于接收返回值 
            MessageBox.Show(myResult);
        }

백그라운드에서 작업을 시작하려면 BackgroundWorker의 RunWorkerAsync() 메서드를 호출해야 합니다. 이 메서드가 호출되면 BackgroundWorker는 DoWork 이벤트의 코드가 실행되어 백그라운드 작업을 시작합니다. 또 다른 스레드.
백그라운드 작업이 완료되면 완료 또는 취소 여부에 관계없이 RunWorkerCompleted 이벤트가 트리거되며 이 방법을 통해 백그라운드 작업의 완료 결과를 사용자에게 피드백할 수 있습니다.
또한 RunWorkerCompletedEventArgs 인스턴스의 Canceled 속성을 사용하여 취소 작업으로 인해 백그라운드 작업이 종료되었는지 확인합니다.


구체적인 데모는 다음과 같습니다.


BackgroundWorker backgroundWorker1 = new BackgroundWorker();


데모 코드는 MSDN에서 제공됩니다. 링크를 열려면 클릭하세요


참조 기사: 링크를 열려면 클릭하세요

4. 값을 반환하는 대신 어떻게 우아하게 작성해야 할까요? 익명 함수 C# 다중 스레드 매개변수 전달

FunctionClass 클래스가 추가되었으며 테스트 함수는 다음과 같습니다.

using System;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    public partial class Form2 : Form
    {
        public Form2()
        {
            InitializeComponent();
        }

        private void Form2_Load(object sender, EventArgs e)
        {
            //TransitionalClass tc = new TransitionalClass("xiaoming", 10);
            //ThreadPool.QueueUserWorkItem(new WaitCallback(TransitionalClass.TestFunction), tc);
        }

        private void button1_Click(object sender, EventArgs e)
        {
            this.TestArea2();
        }

        private System.ComponentModel.BackgroundWorker BackgroundWorker1
    = new System.ComponentModel.BackgroundWorker();

        private void TestArea2()
        {
            InitializeBackgroundWorker();

            AreaClass2 AreaObject2 = new AreaClass2();
            AreaObject2.Base = 30;
            AreaObject2.Height = 40;

            // Start the asynchronous operation.
            BackgroundWorker1.RunWorkerAsync(AreaObject2);
        }

        private void InitializeBackgroundWorker()
        {
            // Attach event handlers to the BackgroundWorker object.
            BackgroundWorker1.DoWork +=
                new System.ComponentModel.DoWorkEventHandler(BackgroundWorker1_DoWork);
            BackgroundWorker1.RunWorkerCompleted +=
                new System.ComponentModel.RunWorkerCompletedEventHandler(BackgroundWorker1_RunWorkerCompleted);
        }

        private void BackgroundWorker1_DoWork(
            object sender,
            System.ComponentModel.DoWorkEventArgs e)
        {
          //在执行DoWork 事件时,DoWorkEventArgs 实例的Result 属性,返回值到用户;在RunWorkerCompleted 事件里,RunWorkerCompletedEventArgs 实例的Result 属性接收值;
            AreaClass2 AreaObject2 = (AreaClass2)e.Argument;
            // Return the value through the Result property.
            e.Result = AreaObject2.CalcArea();
        }

        private void BackgroundWorker1_RunWorkerCompleted(
            object sender,
            System.ComponentModel.RunWorkerCompletedEventArgs e)
        {
            // Access the result through the Result property. 
            double Area = (double)e.Result;
            MessageBox.Show("The area is: " + Area.ToString());
        }
    }
}

전화 내용은 다음과 같습니다.

   public static void TestFunction2(string name, int age)
        {
            //内部处理省略            
        }

작은 메모:

WCF를 통해 호출하는 경우 스레드 시작 기능을 서버에 배치해야 합니다. 클라이언트에 배치하면 WCF 클라이언트의 시간 제한으로 인해 메인 프로그램이 설명할 수 없는 충돌을 일으키기 쉽습니다.

크래시가 발생하는 주된 이유는 클라이언트 wcf 응답 시간이 제한되어 있기 때문입니다.

위 내용은 C# 멀티스레드 매개변수 전달 내용입니다. 더 많은 관련 내용은 PHP 중국어 홈페이지(www.php.cn)를 참고해주세요!


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.