Начать тему с двумя параметрами - PullRequest
3 голосов
/ 13 апреля 2010

У меня есть метод, который вызывается для события, который представляет мне две переменные varA, varB (обе строки).Этот метод вызывается с новой информацией довольно часто, поэтому я создал отдельный метод, который принимает два параметра.Я хочу запустить этот метод в потоке, однако столкнулся с проблемой, что Thread.Start не будет принимать параметры.

Я пробовал несколько предполагаемых методов, но до сих пор не получилось .. Я думаю,Лучше всего создать отдельный класс и обработать его там. Однако у меня есть список, в который я вставляю данные, и я зашел в тупик, когда отдельный класс попытался получить доступ к этому списку, поскольку он был в другом классе.

Может кто-нибудь помочь мне здесь, пожалуйста?

Ответы [ 5 ]

4 голосов
/ 13 апреля 2010

Простое решение ...
Это выведет varA, varB на консоль.

new RunTask<string, string>("varA", "varB").StartThread();

public class RunTask<TA, TB>
{
    public TA VarA { get; private set; }
    public TB VarB { get; private set; }

    public RunTask(TA varA, TB varB)
    {
        VarA = varA;
        VarB = varB;
    }
    public void StartThread()
    {
        ThreadPool.QueueUserWorkItem(Worker, this);
    }
    public void Worker(object obj)
    {
        var state = obj as RunTask<TA,TB>;
        Console.WriteLine(state.VarA + ", " + state.VarB);
    }
}

Edit:
Если вам нужно решение, в котором необходимо обновить пользовательский интерфейс, возможно, это лучшее решение.
И это, вероятно, лучшее решение, чтобы поместить аргументы в отдельный класс, как этот.
Вставить код в список обновлений после комментария // Обновить список

public class ListForm : Form
{
    private static readonly object _listResultLock = new object();
    private readonly Action<TaskResult> _listResultHandler;

    public ListForm()
    {
        Load += ListForm_Load;
        _listResultHandler = TaskResultHandler;
    }

    private void ListForm_Load(object sender, EventArgs e)
    {
        new RunTask(new Task("varA", "varB", TaskResultHandler)).StartThread();
    }
    public void TaskResultHandler(TaskResult result)
    {
        if (InvokeRequired)
        {
            Invoke(_listResultHandler, result);
            return;
        }
        lock (_listResultLock)
        {
            // Update List
        }
    }
}

public class Task
{
    public Action<TaskResult> Changed { get; private set; }
    public string VarA { get; private set; }
    public string VarB { get; private set; }

    public Task(string varA, string varB, Action<TaskResult> changed)
    {
        VarA = varA;
        VarB = varB;
        Changed = changed;
    }
}
public class TaskResult
{
    public string VarA { get; private set; }
    public string VarB { get; private set; }

    public TaskResult(string varA, string varB)
    {
        VarA = varA;
        VarB = varB;
    }
}
public class RunTask
{
    private readonly Task _task;

    public RunTask(Task task)
    {
        _task = task;
    }
    public void StartThread()
    {
        ThreadPool.QueueUserWorkItem(Worker, _task);
    }
    public void Worker(object obj)
    {
        var state = obj as Task;
        if (state == null) return;
        if (state.Changed == null) return;
        state.Changed(new TaskResult("this is " + state.VarA, "this is " + state.VarA));
    }
}
3 голосов
/ 13 апреля 2010

Метод запуска потока принимает параметр объекта.

Если ваш метод принимает несколько параметров, вы можете очень хорошо передать объект вашего класса, содержащий параметры в него. Затем вы можете распаковать его в вашем методе.

Thread.start (объект)

http://msdn.microsoft.com/en-us/library/system.threading.thread.start.aspx

Обновление

В вашем случае попробуйте это,

string varC = varA + "," + varB;
Thread.Start(varC);

и в вашем методе

string args[] = ((string)par).Split(',');
1 голос
/ 13 апреля 2010

Если вы знаете, что будет предоставлено потоку, то, возможно, вы захотите передать ему объекты.

Подробнее здесь -> http://msdn.microsoft.com/en-us/library/6x4c42hc.aspx

Пример:

void someFunction()
{
    Thread t = new Thread(doWork);

Тогда

    t.Start(new int[] { 1, 2 }); //two values passed to the thread.

Или,

    t.Start(1); //one value passed to the thread.
}



Теперь метод -

void doWork(object data)
{
    System.Collections.IList list = data as System.Collections.IList;
    if (list != null)
    {
        object[] _objArr = data as object[];
        foreach (object io in list)
        {
            System.Diagnostics.Trace.WriteLine(io);
        }
    }
    else
    {
        System.Diagnostics.Trace.WriteLine(data);
    }
}
0 голосов
/ 13 апреля 2010

Вместо использования System.Thread вы можете использовать делегата.

например:.

public delegate void RunMyMethod(String a, String b);

RunMyMethod myDelegate = new RunMyMethod(MyMethod);
myDelegate.BeginInvoke(someAFromSomeWhere, someBFromSomeWhere);

Где MyMethod - метод, который вы хотите запустить, а параметры извлекаются из любого места.

0 голосов
/ 13 апреля 2010

Вы можете выполнить метод с параметрами в другом потоке различными способами: через класс Thread, через System.ComponentModel.BackgroundWorker, через ThreadPool. Если метод вызывается часто, я думаю, что ThreadPool подойдет:

        ThreadPool.QueueUserWorkItem(
        (args)=>
            {
                try
                {
                //args - object, you can put in it string[] or
                // whatever you want

                // do work
                }
                catch(Exception e)
                {
                    // dont throw exceptions in this thread
                    // or application will crashed
                }
            }
        );

Вы можете прочитать об этих классах на MSDN:

http://msdn.microsoft.com/en-us/library/system.threading.threadpool.aspx

http://msdn.microsoft.com/en-us/library/system.threading.parameterizedthreadstart.aspx

http://msdn.microsoft.com/en-us/library/system.componentmodel.backgroundworker.aspx

...