代码之家  ›  专栏  ›  技术社区  ›  XXXXX

如何在DataGridView中以编程方式设置单元格值?

  •  36
  • XXXXX  · 技术社区  · 15 年前

    我有一个数据报视图。一些单元从串行端口接收数据:我想将数据推送到单元中,并让它更新底层绑定对象。

    我正在尝试这样的方法:

    SetValueFromSerial (decimal newValue)
    {
        dataGridView.CurrentCell.Value = newValue;
    }
    

    使用字符串没有帮助:

        dataGridView.CurrentCell.Value = newValue.ToString ();
    

    在这两种情况下,我在网格中看不到任何内容,并且基础值不变。

    我在这里搜索过,但什么也没找到。(我可能错过了什么,也许是明显的,但我不是 完全地 懒惰的)

    14 回复  |  直到 8 年前
        1
  •  33
  •   Thomas Levesque    15 年前

    如果 DataGridView 是数据绑定的,不应直接修改单元格的内容。相反,您应该修改数据绑定对象。您可以通过 DataBoundItem DataGridViewRow :

    MyObject obj = (MyObject)dataGridView.CurrentRow.DataBoundItem;
    obj.MyProperty = newValue;
    

    注意,绑定对象应该实现 INotifyPropertyChanged 这样变化就反映在 数据表格控件

        2
  •  28
  •   Chuck Savage    12 年前
    dataGridView1[1,1].Value="tes";
    
        3
  •  5
  •   BornToCode    12 年前

    如果出于某种原因不想修改数据绑定对象(例如,希望在网格中显示某个视图,但不希望它作为数据源对象的一部分),则可能需要执行以下操作:

    1.手动添加列:

        DataGridViewColumn c = new DataGridViewColumn();
        DataGridViewCell cell = new DataGridViewTextBoxCell();
    
        c.CellTemplate = cell;
        c.HeaderText = "added";
        c.Name = "added";
        c.Visible = true;
    dgv.Columns.Insert(0, c); 
    

    2.在数据绑定完成事件中,执行以下操作:

    foreach (DataGridViewRow row in dgv.Rows)
    {if (row.Cells[7].Value.ToString()=="1")
    row.Cells[0].Value = "number one"; }
    

    (只是一个愚蠢的例子)

    但请记住,它必须在databindingComplete中,否则值将保持空白。

        4
  •  4
  •   Graviton    15 年前

    你记得吗 refresh 数据报视图?

    datagridview.refresh();
    
        5
  •  4
  •   T30    11 年前

    我也有同样的问题 使用SQL数据适配器更新数据等

    以下内容对我很有用

    mydatgridview.Rows[x].Cells[x].Value="test"
    mydatagridview.enabled = false 
    mydatagridview.enabled = true 
    
        6
  •  4
  •   Razib Ali    10 年前

    我搜索了如何插入新行以及如何像Excel一样设置其中单元格的单个值的解决方案。我用以下代码解决了:

    dataGridView1.ReadOnly = false; //Before modifying, it is required.
    dataGridView1.Rows.Add(); //Inserting first row if yet there is no row, first row number is '0'
    dataGridView1.Rows[0].Cells[0].Value = "Razib, this is 0,0!"; //Setting the leftmost and topmost cell's value (Not the column header row!)
    dataGridView1[1, 0].Value = "This is 0,1!"; //Setting the Second cell of the first row!
    

    注:

    1. 以前我在设计模式下设计了这些列。
    2. 我已将DataGridView属性中的行标题可见性设置为false。
    3. 最后一行必须理解: 当您直接给出DataGridView的索引时,第一个数字是单元格编号,第二个数字是行号!记住它!

    希望这对你有帮助。

        7
  •  3
  •   Mayank    9 年前

    尝试这种方式:

    dataGridView.CurrentCell.Value = newValue;
    
    dataGridView.EndEdit();
    
    dataGridView.CurrentCell.Value = newValue;
    
    dataGridView.EndEdit();
    

    需要写两次…

        8
  •  1
  •   Jesse    11 年前

    以下工作。我可能搞错了,但是加了一个 String 值似乎与DataGridView单元格不兼容(但我没有尝试过或尝试过任何黑客攻击)。

    DataGridViewName.Rows[0].Cells[0].Value = 1;
    
        9
  •  1
  •   noelicus    10 年前

    如果 DataGridView 已由填充 DataSource = x (即数据绑定)然后需要更改 绑定数据 ,而不是DataGridView单元格本身。

    从已知的行或列获取数据的一种方法是:

    (YourRow.DataBoundItem as DataRowView).Row['YourColumn'] = NewValue;
    
        10
  •  1
  •   Matthis Kohli    8 年前

    我遇到了同样的问题,并解决了它,如下为vb.net。它是.NET框架,因此您应该能够适应。想比较一下我的解决方案,现在我发现似乎没有人能用我的方式来解决。

    进行字段声明。

    Private _currentDataView as DataView

    因此,循环遍历所有行,并搜索一个包含我知道的值的单元格,该值位于我要更改的单元格旁边。

    Public Sub SetCellValue(ByVal value As String)
        Dim dataView As DataView = _currentDataView
    
        For i As Integer = 0 To dataView.Count - 1
            If dataView(i).Row.Item("projID").ToString.Equals("139") Then
                dataView(i).Row.Item("Comment") = value
                Exit For ' Exit early to save performance
            End If
        Next
    End Sub
    

    这样你才能更好地理解它。 我知道专栏名“Projid”是139。我循环直到找到它,然后我可以在我的例子“comment”中更改“columnnameofcell”的值。我将此用于在运行时添加的注释。

    dataview

        11
  •  1
  •   Eric Draven    8 年前

    在VB中,你可以使用这个

    Dim selectedRow As DataRowView
    selectedRow = dg.Rows(dg.CurrentCell.RowIndex).DataBoundItem
    selectedRow("MyProp") = "myValue"
    dg.NotifyCurrentCellDirty(True)
    

    感谢赛义德·塞尔普善最后一排

        12
  •  0
  •   FrenkyB    11 年前

    正如@thomas所说,要更改的元素必须实现inotifyPropertiesChanged。但是,数据源也很重要。它必须是bindingList,您可以从列表轻松创建。

    下面是我的示例-数据源首先是数据表,我将其传输到列表,然后创建绑定列表。然后我创建bindingsource并使用bindinglist作为bindingsource的数据源。最后,DataGridView中的数据源使用此bindingSource。

     sp_Select_PersonTableAdapter adapter = new sp_Select_PersonTableAdapter();
    
     DataTable tbl = new DataTable();
     tbl.Merge(adapter.GetData());
    
     List<Person> list = tbl.AsEnumerable().Select(x => new Person
     {
         Id = (Int32) (x["Id"]),
         Ime = (string) (x["Name"] ?? ""),
         Priimek = (string) (x["LastName"] ?? "")
     }).ToList();
    
     BindingList<Person> bindingList = new BindingList<Person>(list);
    
     BindingSource bindingSource = new BindingSource();
     bindingSource.DataSource = bindingList;
     dgvPerson.DataSource = bindingSource;
    

    同样重要的是:每个类的成员setter必须调用OnPropertyChanged()。没有这个,它就不起作用。所以,我的课看起来是这样的:

    public class Person : INotifyPropertyChanged
        {
            private int _id;
            private string _name;
            private string _lastName;
    
            public int Id
            {
                get { return _id; }
                set
                {
                    if (value != _id)
                    {
                        _id = value;
                        OnPropertyChanged();
                    }
                }
            }
            public string Name
            {
                get { return _name; }
                set
                {
                    if (value != _name)
                    {
                        _name = value;
                        OnPropertyChanged();
                    }
                }
            }
            public string LastName
            {
                get { return _lastName; }
                set
                {
                    if (value != _lastName)
                    {
                        _lastName= value;
                        OnPropertyChanged();
                    }
                }
            }
            public event PropertyChangedEventHandler PropertyChanged;
    
            [NotifyPropertyChangedInvocator]
            protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
            {
                PropertyChangedEventHandler handler = PropertyChanged;
                if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
            }
    
        }
    

    有关此主题的详细信息: http://msdn.microsoft.com/en-us/library/system.componentmodel.inotifypropertychanged(v=vs.110).aspx

        13
  •  0
  •   Asif Ali    11 年前
    private void btn_Addtoreciept_Click(object sender, EventArgs e)
    {            
        serial_number++;
        dataGridView_inventory.Rows[serial_number - 1].Cells[0].Value = serial_number;
        dataGridView_inventory.Rows[serial_number - 1].Cells[1].Value =comboBox_Reciept_name.Text;
        dataGridView_inventory.Rows[serial_number - 1].Cells[2].Value = numericUpDown_recieptprice.Value;
        dataGridView_inventory.Rows[serial_number - 1].Cells[3].Value = numericUpDown_Recieptpieces.Value;
        dataGridView_inventory.Rows[serial_number - 1].Cells[4].Value = numericUpDown_recieptprice.Value * numericUpDown_Recieptpieces.Value;
        numericUpDown_RecieptTotal.Value = serial_number;
    }
    

    第一次做得很好,但第二次按会出错 “索引超出范围。必须为非负且小于集合的大小。 参数名称:index“,但是当我单击单元格时,会出现另一行,然后它将在下一行工作,并继续执行…

        14
  •  0
  •   gamendola    8 年前

    我尝试了很多方法,唯一有效的方法是updateCellValue:

    dataGridView.Rows[rowIndex].Cells[columnIndex].Value = "New Value";
    dataGridView.UpdateCellValue(columnIndex, rowIndex);
    

    我希望能帮上忙。=)