2

我试图从DataTable AllItems行中删除行DataTables Items;这样做的目的是获取DataTable AllItems不在里面的物品DataTable Items

所有这些行都来自同一个 Excel 文件,该文件包含几列并且是相等的。

我尝试过使用 foreach 循环:

foreach(DataRow dr in AllItems.Rows)
{
if (Items.Contains(dr))
{
AllItems.Rows.Remove(dr);
}

但我收到以下错误:表没有主键。

有谁知道我如何删除这些行?

4

2 回答 2

3

您在这里有几个选择:

1.添加主键

您可以在创建数据表时为其添加主键。

假设您有一个名为“Id”的列,那么您将这样做:

AllItems.PrimaryKey = new DataColumn[] { workTable.Columns["Id"] };} 

或者,对于您的主键是复合键(多列)的情况:

AllItems.PrimaryKey = new DataColumn[] { 
                             workTable.Columns["Id"], 
                             workTable.Columns["Name"] };} 

这将允许 Contains 正常工作。

2.使用数据视图

您可以使用 DataView 过滤掉不同的行;

DataView view = new DataView(AllItems);
DataTable distinctValues = view.ToTable(true, "Column1", "Column2" , ..., "ColumnN");

3. 使用 Select 查找匹配行

或者,您可以依靠SelectItems方法根据类似于 SQL WHEREclause 的语句来测试 DataTable 中是否存在相应的行:

List<DataRow> rowsToRemove = new List<DataRow>();

foreach(DataRow allItemRow in AllItems.Rows)
{
    if(Items.Select(String.Format("Id = {0}"), 
          allItemRow.Field<Int32>("Id")).Length == 0)
    {
        rowsToRemove.Add(allItemRow);
    }
}

rowsToRemove.ForEach(x => x.Delete());

AllItems.AcceptChanges();

请注意,在迭代 AllItems 中的 Rows 集合时,不要删除行,这一点很重要——而是收集这些行,然后将它们删除。

4. 中途过滤

另请注意,我还没有尝试过,但是,根据您从 Excel 中选择行的方式,您可以使用 SQL DISTINCT 子句;如果您使用ODBC 从 Excel 加载数据,那么您可以尝试从源头过滤。

于 2013-05-15T07:19:51.950 回答
0

你可以试试这个:

var exceptItems = AllItems.Rows.Cast<DataRow>()
    .Except(Items.Rows.Cast<DataRow>(), DataRowComparer.Default)
    .ToList();

allItems作为替代方案,如果您想在从中删除行后继续使用数据表items,您可以尝试这样做(假设您Id在两个数据表中都有该列,它唯一地标识每个数据表的一行):

var exceptItems = AllItems.Rows.Cast<DataRow>()
    .Select((i, index) => new { id = i["Id"], index })
    .Intersect(Items.Rows.Cast<DataRow>()
        .Select((i, index) => new { id = i["Id"], index }))
    .ToList();

for (int i = exceptItems.Count()-1; i >= 0; i--)
{
    AllItems.Rows.RemoveAt(exceptItems[i].index);
}

这是上面最后一个示例的更好安排:

AllItems.Rows.Cast<DataRow>()
    .Select((i, index) => new { id = i["Id"], index })
    .Intersect(Items.Rows.Cast<DataRow>()
        .Select((i, index) => new { id = i["Id"], index }))
    .OrderByDescending(i => i.index)
    .ToList()
    .ForEach(i => AllItems.Rows.RemoveAt(i.index));
于 2013-05-15T07:26:18.323 回答