1

我使用 PostgreSQL 视图在以下位置显示数据DataGridView

dataSource = new BindingSource();
dataSource.DataSource = Program.DB.GetView(dbView);  // returns a DataTable
dgData.DataSource = dataSource;

现在,在我使用 PostgreSQL 函数添加记录后,我刷新了网格中的数据(我不Rows.Add()调用DataGridView

protected void RefreshData() {
    dataSource.DataSource = Program.DB.GetView(dbView);
}

PostgreSQL 的插入函数返回插入行的 ID,所以我知道 ID(即主键)并想Selectedtrue. DataGridView该行可以在集合中的任何位置,因为视图是按名称排序的,而不是 ID。我认为我可以通过循环遍历所有行并设置Selected找到它的时间来做到这一点,但这在大型数据集上可能会变得很慢。

有没有办法以某种方式将数据源的行绑定到数据网格?

4

1 回答 1

1

向事件添加事件处理程序ListChanged

dataSource.ListChanged += dataSource_ListChanged;

这是事件处理程序定义:

void dataSource_ListChanged(object sender, ListChangedEventArgs e)
{
    if (dgData.Rows.Count > 0)
        dgData.CurrentCell = dgData.Rows[e.NewIndex].Cells[0];
}

[更新]

正如我在评论中所建议的那样,也许您不应该在每次插入(或更新)时重新填充数据源。为了证明我的观点,我将发布一个使用 a DataGridView、 2TextBoxes和 2 Buttons(用于插入和更新)和 a的代码示例SqlDataAdapter。SQL 表有 2 列 (idvalue)。

这是代码(我没有涉及删除):

public partial class Form1 : Form
{
    static BindingSource dataSource;
    static string dbView = "default";

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        dgData.MultiSelect = false;
        dataSource = new BindingSource();
        dataSource.ListChanged += dataSource_ListChanged;
        RefreshData();
        dgData.DataSource = dataSource;
        dgData.Sort(dgData.Columns[1], ListSortDirection.Ascending);
    }

    void dataSource_ListChanged(object sender, ListChangedEventArgs e)
    {
        if (dgData.Rows.Count > 0)
            dgData.CurrentCell = dgData.Rows[e.NewIndex].Cells[0];
    }

    protected void RefreshData()
    {
        dataSource.DataSource = DB.GetView(dbView);
    }

    private void insert_Click(object sender, EventArgs e)
    {
        DB.Insert(textBox1.Text);
        RefreshData();
    }

    private void update_Click(object sender, EventArgs e)
    {
        DB.UpdateRandomRow(textBox2.Text);
        RefreshData();
    }
}

class DB
{
    static DataTable dt;
    static string conStr = "yourConnectionString";
    static SqlDataAdapter _adapter;
    static Random r = new Random(10);

    public static SqlDataAdapter CreateSqlDataAdapter(SqlConnection connection)
    {
        _adapter = new SqlDataAdapter();
        _adapter.MissingSchemaAction = MissingSchemaAction.AddWithKey;

        _adapter.SelectCommand = new SqlCommand(
            "SELECT * FROM test", connection);
        _adapter.InsertCommand = new SqlCommand(
            "INSERT INTO test (value) " +
            "VALUES (@value)", connection);
        _adapter.UpdateCommand = new SqlCommand(
            "UPDATE test SET [value] = @value " +
            "WHERE id = @id", connection);
        _adapter.InsertCommand.Parameters.Add("@value",
            SqlDbType.NVarChar, 50, "value");
        _adapter.UpdateCommand.Parameters.Add("@id",
            SqlDbType.Int, 4, "id").SourceVersion = DataRowVersion.Current;
        _adapter.UpdateCommand.Parameters.Add("@value",
            SqlDbType.NVarChar, 50, "value").SourceVersion = DataRowVersion.Current;

        return _adapter;
    }

    // random update, to demonstrate dynamic
    // repositioning
    public static DataTable UpdateRandomRow(string value)
    {
        var currentRandom = r.Next(dt.Rows.Count);
        dt.Rows[currentRandom].SetField<string>(1, value);
        using (var con = new SqlConnection(conStr))
        {
            con.Open();
            _adapter = CreateSqlDataAdapter(con);
            _adapter.Update(dt);
        }
        return dt;
    }

    internal static DataTable GetView(string dbView)
    {
        if (dt == null)
        {
            dt = new DataTable();
            using (var con = new SqlConnection(conStr))
            {
                con.Open();
                _adapter = CreateSqlDataAdapter(con);
                _adapter.Fill(dt);
            }
        }
        return dt;
    }

    internal static void Insert(string value)
    {
        if (dt == null)
            GetView("");
        var dr = dt.NewRow();
        dr[1] = value;
        dt.Rows.Add(dr);
        using (var con = new SqlConnection(conStr))
        {
            con.Open();
            _adapter = CreateSqlDataAdapter(con);
            _adapter.Update(dt);
        }
    }
}

如果您对其进行测试,您将看到您的问题的请求对于插入和更新都得到了满足。此外,请注意性能改进,因为不会在每次操作时重新创建数据表。

于 2013-03-01T09:26:36.030 回答