Поток DataRow безопасен? Как обновить один dataow в datatable, используя несколько потоков? - .net 2.0 - PullRequest
1 голос
/ 13 апреля 2010

Я хочу обновить один dataow в datatable, используя несколько потоков. Это действительно возможно?

Я написал следующий код, реализующий простую многопоточность для обновления одного datarow. Я получаю разные результаты каждый раз. Почему это так?

public partial class Form1 : Form
{
    private static DataTable dtMain;
    private static string threadMsg = string.Empty;
    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        Thread[] thArr = new Thread[5];
        dtMain = new DataTable();
        dtMain.Columns.Add("SNo");
        DataRow dRow;
        dRow = dtMain.NewRow();
        dRow["SNo"] = 5;
        dtMain.Rows.Add(dRow);
        dtMain.AcceptChanges();
        ThreadStart ts = new ThreadStart(delegate { dtUpdate(); });
        thArr[0] = new Thread(ts);
        thArr[1] = new Thread(ts);
        thArr[2] = new Thread(ts);
        thArr[3] = new Thread(ts);
        thArr[4] = new Thread(ts);

        thArr[0].Start();
        thArr[1].Start();
        thArr[2].Start();
        thArr[3].Start();
        thArr[4].Start();

        while (!WaitTillAllThreadsStopped(thArr))
        {
            Thread.Sleep(500);
        }

        foreach (Thread thread in thArr)
        {
            if (thread != null && thread.IsAlive)
            {
                thread.Abort();
            }
        }
        dgvMain.DataSource = dtMain;

    }

    private void dtUpdate()
    {
        for (int i = 0; i < 1000; i++)
        {
            try
            {
                dtMain.Rows[0][0] = Convert.ToInt32(dtMain.Rows[0][0]) + 1;
                dtMain.AcceptChanges();
            }
            catch
            {
                continue;
            }
        }
    }

    private bool WaitTillAllThreadsStopped(Thread[] threads)
    {
        foreach (Thread thread in threads)
        {
            if (thread != null && thread.ThreadState == ThreadState.Running)
            {
                return false;
            }
        }
        return true;
    }


}

Есть мысли по этому поводу?

Спасибо

NLV

Ответы [ 2 ]

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

Согласно MSDN :

Этот тип безопасен для многопоточных операций чтения. Вы должны синхронизировать любые операции записи.

Таким образом, поскольку вы обновляете объекты DataRow и DataTable, вам потребуется использовать некоторую форму синхронизации, чтобы гарантировать, что ваш код является поточно-ориентированным.

Кстати, вы также никогда не должны вызывать Thread.Sleep или делать занятый цикл в вашем потоке пользовательского интерфейса. В противном случае пользовательский интерфейс будет полностью не отвечать, пока все потоки не завершат выполнение. Вместо этого у вас должен быть какой-то индикатор выполнения (или просто счетчик), отображаемый в вашем пользовательском интерфейсе, который может обновляться событиями из ваших потоков. Класс BackgroundWorker специально разработан для облегчения этой работы, поэтому вы можете рассмотреть возможность его использования.

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

Если в документации к типу или элементу не указано иное:

  • Статические члены являются потокобезопасными.
  • Члены экземпляра не потокобезопасны.
...