需要更快的DataGridView批量编辑方法

时间:2009-12-02 16:21:58

标签: c# winforms datatable

这是我目前用于在绑定的DataGridView上实现“全部检查”功能的内容:

int max = ((DataTable)dataGridRes.DataSource).Rows.Count;
for (int i = 0; i < max; i++)
{
    if(((DataTable)dataGridRes.DataSource).Rows[i].Field<long>(0) == 0)
        ((DataTable)dataGridRes.DataSource).Rows[i].SetField(0, 1);
}

然而,这段代码无可救药地缓慢。在625行DataTable上,我的计算机需要大约5秒才能完成。非常不可接受。

我做错了什么?我可以用什么更好的方法在DataGridView上进行批量编辑?

4 个答案:

答案 0 :(得分:1)

此博客包含您所需的内容:DataGridView CheckBox Select All

为后代发布的代码:

public delegate void CheckBoxClickedHandler(bool state);
public class DataGridViewCheckBoxHeaderCellEventArgs : EventArgs
{
    bool _bChecked;
    public DataGridViewCheckBoxHeaderCellEventArgs(bool bChecked)
    {
        _bChecked = bChecked;
    }
    public bool Checked
    {
        get { return _bChecked; }
    }
}

class DatagridViewCheckBoxHeaderCell : DataGridViewColumnHeaderCell
{
    Point checkBoxLocation;
    Size checkBoxSize;
    bool _checked = false;
    Point _cellLocation = new Point();
    System.Windows.Forms.VisualStyles.CheckBoxState _cbState =
        System.Windows.Forms.VisualStyles.CheckBoxState.UncheckedNormal;

    public event CheckBoxClickedHandler OnCheckBoxClicked;

    public DatagridViewCheckBoxHeaderCell()
    {
    }

    protected override void Paint(System.Drawing.Graphics graphics,
        System.Drawing.Rectangle clipBounds,
        System.Drawing.Rectangle cellBounds,
        int rowIndex,
        DataGridViewElementStates dataGridViewElementState,
        object value,
        object formattedValue,
        string errorText,
        DataGridViewCellStyle cellStyle,
        DataGridViewAdvancedBorderStyle advancedBorderStyle,
        DataGridViewPaintParts paintParts)
    {
        base.Paint(graphics, clipBounds, cellBounds, rowIndex,
            dataGridViewElementState, value,
            formattedValue, errorText, cellStyle,
            advancedBorderStyle, paintParts);

        Point p = new Point();

        Size s = CheckBoxRenderer.GetGlyphSize(graphics,
            System.Windows.Forms.VisualStyles.CheckBoxState.UncheckedNormal);

        p.X = cellBounds.Location.X +
            (cellBounds.Width / 2) - (s.Width / 2);
        p.Y = cellBounds.Location.Y +
            (cellBounds.Height / 2) - (s.Height / 2);

            _cellLocation = cellBounds.Location;
            checkBoxLocation = p;
            checkBoxSize = s;
        if (_checked)
            _cbState = System.Windows.Forms.VisualStyles.
            CheckBoxState.CheckedNormal;
        else
            _cbState = System.Windows.Forms.VisualStyles.
            CheckBoxState.UncheckedNormal;
            CheckBoxRenderer.DrawCheckBox
            (graphics, checkBoxLocation, _cbState);
    }

    protected override void OnMouseClick(DataGridViewCellMouseEventArgs e)
    {
        Point p = new Point(e.X + _cellLocation.X, e.Y + _cellLocation.Y);
        if (p.X >= checkBoxLocation.X && p.X <=
            checkBoxLocation.X + checkBoxSize.Width
            && p.Y >= checkBoxLocation.Y && p.Y <=
            checkBoxLocation.Y + checkBoxSize.Height)
        {
        _checked = !_checked;
        if (OnCheckBoxClicked != null)
        {
            OnCheckBoxClicked(_checked);
            this.DataGridView.InvalidateCell(this);
        }
    }
    base.OnMouseClick(e);
  }
}

下面给出了为标题添加复选框的代码和相应的事件代码。

private void FormatGrid()
{
    DataView dv = new DataView();
    dv.Table = _loginDs.Tables[0];

    DataGridViewCheckBoxColumn chkbox = new DataGridViewCheckBoxColumn();
    DatagridViewCheckBoxHeaderCell chkHeader = new DatagridViewCheckBoxHeaderCell();
    chkbox.HeaderCell = chkHeader;
    chkHeader.OnCheckBoxClicked += new CheckBoxClickedHandler(chkHeader_OnCheckBoxClicked);
    _chkBoxGrid.Columns.Add(chkbox);

    DataGridViewTextBoxColumn uname = new DataGridViewTextBoxColumn();
    uname.HeaderText = "user";
    uname.Name = "username";
    uname.DataPropertyName = "username";
    _chkBoxGrid.Columns.Add(uname);

    _chkBoxGrid.DataSource = dv;
}

void chkHeader_OnCheckBoxClicked(bool state)
{
    foreach (DataGridViewRow row in _chkBoxGrid.Rows)
        row.Cells[0].Value = state;

}

答案 1 :(得分:1)

我找到了一种方法来加速到可接受的性能。基本上取消绑定控件,对DataTable进行更新并重新绑定。

DataTable dt = (DataTable)dataGridRes.DataSource;
dataGridRes.DataSource = null;

for (int i = 0; i < dt.Rows.Count; i++)
    dt.Rows[i].SetField(0, 1);

dataGridRes.DataSource = dt;

答案 2 :(得分:0)

第二次尝试。

  1. 你投的次数太多了
  2. 您正在执行无关紧要的if声明
  3. 尝试类似:

    foreach(DataRow r in ((DataTable)dataGridRes.DataSource).Rows)
    {
        r.SetField(0, 1);
    }
    

答案 3 :(得分:0)

在DataTable中编辑多个行或列时,请先调用BeginLoadData 1st,然后在完成后调用EndLoadData