Select DISTINCT on DataTable
From:http://weblogs.asp.net/eporter/default.aspx
 using System;
using System;
 using System.Data;
using System.Data;
 using System.Configuration;
using System.Configuration;
 using System.Collections;
using System.Collections;
 using System.Web;
using System.Web;
 using System.Web.Security;
using System.Web.Security;
 using System.Web.UI;
using System.Web.UI;
 using System.Web.UI.WebControls;
using System.Web.UI.WebControls;
 using System.Web.UI.WebControls.WebParts;
using System.Web.UI.WebControls.WebParts;
 using System.Web.UI.HtmlControls;
using System.Web.UI.HtmlControls;

 public partial class SelectDistinctDataTable : System.Web.UI.Page
public partial class SelectDistinctDataTable : System.Web.UI.Page
 {
{
 protected void Page_Load(object sender, EventArgs e)
    protected void Page_Load(object sender, EventArgs e)
 {
    {
 //Create table
        //Create table
 DataTable dt = new DataTable("items");
        DataTable dt = new DataTable("items");
 //add two columns to datatable
        //add two columns to datatable
 DataColumn col;
        DataColumn col;
 col = new DataColumn("id", typeof(System.Int32));
        col = new DataColumn("id", typeof(System.Int32));
 col.AutoIncrement = true;
        col.AutoIncrement = true;
 dt.Columns.Add(col);
        dt.Columns.Add(col);
 col = new DataColumn("item", typeof(System.String));
        col = new DataColumn("item", typeof(System.String));
 dt.Columns.Add(col);
        dt.Columns.Add(col);
 //config AcceptChange
        //config AcceptChange
 dt.AcceptChanges();
        dt.AcceptChanges();
 //set primary key
        //set primary key
 dt.PrimaryKey = new DataColumn[] { dt.Columns["id"] };
        dt.PrimaryKey = new DataColumn[] { dt.Columns["id"] };

 DataTable DesTbl = dt.Clone();
        DataTable DesTbl = dt.Clone();


 DataRow dr;
        DataRow dr;
 for (int i = 0; i < 10; i++)
        for (int i = 0; i < 10; i++)
 {
        {
 dr = dt.NewRow();
            dr = dt.NewRow();
 dr[1] = "Roboth" + i;
            dr[1] = "Roboth" + i;
 dt.Rows.Add(dr);
            dt.Rows.Add(dr);
 }
        }
 for (int i = 0; i < 10; i++)
        for (int i = 0; i < 10; i++)
 {
        {
 dr = dt.NewRow();
            dr = dt.NewRow();
 dr[1] = "Roboth" + i;
            dr[1] = "Roboth" + i;
 dt.Rows.Add(dr);
            dt.Rows.Add(dr);
 }
        }
 PrintValues(dt, "before select distinct");
        PrintValues(dt, "before select distinct");
 DataTable distTbl = SelectDistinct(dt, new string[] {"item" });
        DataTable distTbl = SelectDistinct(dt, new string[] {"item" });
 PrintValues(distTbl, "the result for select distinct datatable");
        PrintValues(distTbl, "the result for select distinct datatable");

 }
    }
 private static void PrintValues(DataTable table, string lbl)
    private static void PrintValues(DataTable table, string lbl)
 {
    {
 // Display the values in the supplied DataTable:
        // Display the values in the supplied DataTable:

 System.Web.HttpContext.Current.Response.Write(lbl + "<br />");
        System.Web.HttpContext.Current.Response.Write(lbl + "<br />");
 foreach (DataRow row in table.Rows)
        foreach (DataRow row in table.Rows)
 {
        {
 foreach (DataColumn column in table.Columns)
            foreach (DataColumn column in table.Columns)
 {
            {
 System.Web.HttpContext.Current.Response.Write(row[column, DataRowVersion.Current]);
                System.Web.HttpContext.Current.Response.Write(row[column, DataRowVersion.Current]);
 }
            }
 System.Web.HttpContext.Current.Response.Write("<br />");
            System.Web.HttpContext.Current.Response.Write("<br />");
 }
        }
 }
    }
 private static DataTable SelectDistinct(DataTable SourceTable, params string[] FieldNames)
    private static DataTable SelectDistinct(DataTable SourceTable, params string[] FieldNames)
 {
    {
 object[] lastValues;
        object[] lastValues;
 DataTable newTable;
        DataTable newTable;
 DataRow[] orderedRows;
        DataRow[] orderedRows;

 if (FieldNames == null || FieldNames.Length == 0)
        if (FieldNames == null || FieldNames.Length == 0)
 throw new ArgumentNullException("FieldNames");
            throw new ArgumentNullException("FieldNames");

 lastValues = new object[FieldNames.Length];
        lastValues = new object[FieldNames.Length];
 newTable = new DataTable();
        newTable = new DataTable();

 foreach (string fieldName in FieldNames)
        foreach (string fieldName in FieldNames)
 newTable.Columns.Add(fieldName, SourceTable.Columns[fieldName].DataType);
            newTable.Columns.Add(fieldName, SourceTable.Columns[fieldName].DataType);

 orderedRows = SourceTable.Select("", string.Join(", ", FieldNames));
        orderedRows = SourceTable.Select("", string.Join(", ", FieldNames));

 foreach (DataRow row in orderedRows)
        foreach (DataRow row in orderedRows)
 {
        {
 if (!fieldValuesAreEqual(lastValues, row, FieldNames))
            if (!fieldValuesAreEqual(lastValues, row, FieldNames))
 {
            {
 newTable.Rows.Add(createRowClone(row, newTable.NewRow(), FieldNames));
                newTable.Rows.Add(createRowClone(row, newTable.NewRow(), FieldNames));

 setLastValues(lastValues, row, FieldNames);
                setLastValues(lastValues, row, FieldNames);
 }
            }
 }
        }

 return newTable;
        return newTable;
 }
    }

 private static bool fieldValuesAreEqual(object[] lastValues, DataRow currentRow, string[] fieldNames)
    private static bool fieldValuesAreEqual(object[] lastValues, DataRow currentRow, string[] fieldNames)
 {
    {
 bool areEqual = true;
        bool areEqual = true;
 /*
        /*
 id username sex
         id username sex
 0   1        2
         0   1        2
 */
         */
 for (int i = 0; i < fieldNames.Length; i++)
        for (int i = 0; i < fieldNames.Length; i++)
 {
        {
 if (lastValues[i] == null || !lastValues[i].Equals(currentRow[fieldNames[i]]))
            if (lastValues[i] == null || !lastValues[i].Equals(currentRow[fieldNames[i]]))
 {
            {
 areEqual = false;
                areEqual = false;
 break;
                break;
 }
            }
 }
        }

 return areEqual;
        return areEqual;
 }
    }

 private static DataRow createRowClone(DataRow sourceRow, DataRow newRow, string[] fieldNames)
    private static DataRow createRowClone(DataRow sourceRow, DataRow newRow, string[] fieldNames)
 {
    {
 //id username sex
        //id username sex
 // 0   1       2
        // 0   1       2
 foreach (string field in fieldNames)
        foreach (string field in fieldNames)
 newRow[field] = sourceRow[field];
            newRow[field] = sourceRow[field];

 return newRow;
        return newRow;
 }
    }

 private static void setLastValues(object[] lastValues, DataRow sourceRow, string[] fieldNames)
    private static void setLastValues(object[] lastValues, DataRow sourceRow, string[] fieldNames)
 {
    {
 //id username,sex
        //id username,sex
 // 0   1        2
        // 0   1        2
 for (int i = 0; i < fieldNames.Length; i++)
        for (int i = 0; i < fieldNames.Length; i++)
 lastValues[i] = sourceRow[fieldNames[i]];
            lastValues[i] = sourceRow[fieldNames[i]];
 }
    }


 }
}
 
In a project I'm doing I needed to basically do a DISTINCT on a couple fields in a DataTable that I had used earlier in code. Did some Google searching and came up with this MS KB on the subject. I was hoping to find it in the Framework, but hey, writing a little extra code never hurt anyway. Looking over the code though I found that I just didn't really like it. Biggest reason being it only took one field. I wanted to be able to pass in n number of fields, so I rewrote it and did a little tweaking to it. Here's what I came up with in both VB and C#
 using System;
using System; using System.Data;
using System.Data; using System.Configuration;
using System.Configuration; using System.Collections;
using System.Collections; using System.Web;
using System.Web; using System.Web.Security;
using System.Web.Security; using System.Web.UI;
using System.Web.UI; using System.Web.UI.WebControls;
using System.Web.UI.WebControls; using System.Web.UI.WebControls.WebParts;
using System.Web.UI.WebControls.WebParts; using System.Web.UI.HtmlControls;
using System.Web.UI.HtmlControls;
 public partial class SelectDistinctDataTable : System.Web.UI.Page
public partial class SelectDistinctDataTable : System.Web.UI.Page {
{ protected void Page_Load(object sender, EventArgs e)
    protected void Page_Load(object sender, EventArgs e) {
    { //Create table
        //Create table DataTable dt = new DataTable("items");
        DataTable dt = new DataTable("items"); //add two columns to datatable
        //add two columns to datatable DataColumn col;
        DataColumn col; col = new DataColumn("id", typeof(System.Int32));
        col = new DataColumn("id", typeof(System.Int32)); col.AutoIncrement = true;
        col.AutoIncrement = true; dt.Columns.Add(col);
        dt.Columns.Add(col); col = new DataColumn("item", typeof(System.String));
        col = new DataColumn("item", typeof(System.String)); dt.Columns.Add(col);
        dt.Columns.Add(col); //config AcceptChange
        //config AcceptChange dt.AcceptChanges();
        dt.AcceptChanges(); //set primary key
        //set primary key dt.PrimaryKey = new DataColumn[] { dt.Columns["id"] };
        dt.PrimaryKey = new DataColumn[] { dt.Columns["id"] };
 DataTable DesTbl = dt.Clone();
        DataTable DesTbl = dt.Clone();

 DataRow dr;
        DataRow dr; for (int i = 0; i < 10; i++)
        for (int i = 0; i < 10; i++) {
        { dr = dt.NewRow();
            dr = dt.NewRow(); dr[1] = "Roboth" + i;
            dr[1] = "Roboth" + i; dt.Rows.Add(dr);
            dt.Rows.Add(dr); }
        } for (int i = 0; i < 10; i++)
        for (int i = 0; i < 10; i++) {
        { dr = dt.NewRow();
            dr = dt.NewRow(); dr[1] = "Roboth" + i;
            dr[1] = "Roboth" + i; dt.Rows.Add(dr);
            dt.Rows.Add(dr); }
        } PrintValues(dt, "before select distinct");
        PrintValues(dt, "before select distinct"); DataTable distTbl = SelectDistinct(dt, new string[] {"item" });
        DataTable distTbl = SelectDistinct(dt, new string[] {"item" }); PrintValues(distTbl, "the result for select distinct datatable");
        PrintValues(distTbl, "the result for select distinct datatable");
 }
    } private static void PrintValues(DataTable table, string lbl)
    private static void PrintValues(DataTable table, string lbl) {
    { // Display the values in the supplied DataTable:
        // Display the values in the supplied DataTable:
 System.Web.HttpContext.Current.Response.Write(lbl + "<br />");
        System.Web.HttpContext.Current.Response.Write(lbl + "<br />"); foreach (DataRow row in table.Rows)
        foreach (DataRow row in table.Rows) {
        { foreach (DataColumn column in table.Columns)
            foreach (DataColumn column in table.Columns) {
            { System.Web.HttpContext.Current.Response.Write(row[column, DataRowVersion.Current]);
                System.Web.HttpContext.Current.Response.Write(row[column, DataRowVersion.Current]); }
            } System.Web.HttpContext.Current.Response.Write("<br />");
            System.Web.HttpContext.Current.Response.Write("<br />"); }
        } }
    } private static DataTable SelectDistinct(DataTable SourceTable, params string[] FieldNames)
    private static DataTable SelectDistinct(DataTable SourceTable, params string[] FieldNames) {
    { object[] lastValues;
        object[] lastValues; DataTable newTable;
        DataTable newTable; DataRow[] orderedRows;
        DataRow[] orderedRows;
 if (FieldNames == null || FieldNames.Length == 0)
        if (FieldNames == null || FieldNames.Length == 0) throw new ArgumentNullException("FieldNames");
            throw new ArgumentNullException("FieldNames");
 lastValues = new object[FieldNames.Length];
        lastValues = new object[FieldNames.Length]; newTable = new DataTable();
        newTable = new DataTable();
 foreach (string fieldName in FieldNames)
        foreach (string fieldName in FieldNames) newTable.Columns.Add(fieldName, SourceTable.Columns[fieldName].DataType);
            newTable.Columns.Add(fieldName, SourceTable.Columns[fieldName].DataType);
 orderedRows = SourceTable.Select("", string.Join(", ", FieldNames));
        orderedRows = SourceTable.Select("", string.Join(", ", FieldNames));
 foreach (DataRow row in orderedRows)
        foreach (DataRow row in orderedRows) {
        { if (!fieldValuesAreEqual(lastValues, row, FieldNames))
            if (!fieldValuesAreEqual(lastValues, row, FieldNames)) {
            { newTable.Rows.Add(createRowClone(row, newTable.NewRow(), FieldNames));
                newTable.Rows.Add(createRowClone(row, newTable.NewRow(), FieldNames));
 setLastValues(lastValues, row, FieldNames);
                setLastValues(lastValues, row, FieldNames); }
            } }
        }
 return newTable;
        return newTable; }
    }
 private static bool fieldValuesAreEqual(object[] lastValues, DataRow currentRow, string[] fieldNames)
    private static bool fieldValuesAreEqual(object[] lastValues, DataRow currentRow, string[] fieldNames) {
    { bool areEqual = true;
        bool areEqual = true; /*
        /* id username sex
         id username sex 0   1        2
         0   1        2 */
         */ for (int i = 0; i < fieldNames.Length; i++)
        for (int i = 0; i < fieldNames.Length; i++) {
        { if (lastValues[i] == null || !lastValues[i].Equals(currentRow[fieldNames[i]]))
            if (lastValues[i] == null || !lastValues[i].Equals(currentRow[fieldNames[i]])) {
            { areEqual = false;
                areEqual = false; break;
                break; }
            } }
        }
 return areEqual;
        return areEqual; }
    }
 private static DataRow createRowClone(DataRow sourceRow, DataRow newRow, string[] fieldNames)
    private static DataRow createRowClone(DataRow sourceRow, DataRow newRow, string[] fieldNames) {
    { //id username sex
        //id username sex // 0   1       2
        // 0   1       2 foreach (string field in fieldNames)
        foreach (string field in fieldNames) newRow[field] = sourceRow[field];
            newRow[field] = sourceRow[field];
 return newRow;
        return newRow; }
    }
 private static void setLastValues(object[] lastValues, DataRow sourceRow, string[] fieldNames)
    private static void setLastValues(object[] lastValues, DataRow sourceRow, string[] fieldNames) {
    { //id username,sex
        //id username,sex // 0   1        2
        // 0   1        2 for (int i = 0; i < fieldNames.Length; i++)
        for (int i = 0; i < fieldNames.Length; i++) lastValues[i] = sourceRow[fieldNames[i]];
            lastValues[i] = sourceRow[fieldNames[i]]; }
    }

 }
}
 
                    
                     
                    
                 
                    
                


 
     
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号