2

以前我问过关于将列插入数据集的问题。我现在有一个类似的问题......即将两列或多列合并为一列。

假设我有以下数据集:

DataSet ds = new DataSet();
ds.Tables.Add(new DataTable());
ds.Tables[0].Columns.Add("id", typeof(int));
ds.Tables[0].Columns.Add("firstname", typeof(string));
ds.Tables[0].Columns.Add("lastname", typeof(string));

我需要将“名字”和“姓氏”列组合成一个名为“名称”的列。

我最好创建一种将两列合并在一起的方法,还是创建一种可用于将多列合并在一起的更通用的方法?

我的想法是创建一个类似于以下的通用方法:

MergeColumns(字符串格式,字符串mergeColumn,DataTable dt,params string[] columnsToMerge)

用户提供如下格式:“{0} {1}”

mergeColumn 是新列的名称...但它必须能够与将要合并的列之一相同,因为在我的真实世界案例中,我将“name”和“given_names”合并为“name” ...但是如果我需要将“街道”、“城市”、“州”、“邮政编码”等合并到一个名为“地址”的列中,我仍然希望能够使用它。

我认为这将被使用的方式如下:

MergeColumns("{0} {1}", "name", dataTable, "firstname", "lastname");

鉴于上述数据集,我希望得到的数据集如下所示:

DataSet ds = new DataSet();
ds.Tables.Add(new DataTable());
ds.Tables[0].Columns.Add("id", typeof(int));
ds.Tables[0].Columns.Add("name", typeof(string));

这看起来是一种合理的方法吗?这样的方法是否已经存在?我不想重新发明轮子。另外,我是否正在创建一种比我现在实际需要的方法更多的方法?

4

8 回答 8

5
public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();


        DataSet ds = new DataSet();
        ds.Tables.Add(new DataTable());
        ds.Tables[0].Columns.Add("id", typeof(int));
        ds.Tables[0].Columns.Add("firstname", typeof(string));
        ds.Tables[0].Columns.Add("lastname", typeof(string));


        ds.Tables[0].Rows.Add(1,"torvalds", "linus");
        ds.Tables[0].Rows.Add(2,"lennon", "john");            


        ds.Tables[0].MergeColumns("name", "lastname", "firstname");

        foreach (DataRow dr in ds.Tables[0].Rows)
            MessageBox.Show(dr["name"].ToString());


    }


}

public static class Helper
{
    public static void MergeColumns(this DataTable t, string newColumn, params string[] columnsToMerge)
    {
        t.Columns.Add(newColumn, typeof(string));

        var sb = new StringBuilder();

        sb.Append("{0}, ");
        for (int i = 1; i < columnsToMerge.Length; ++i)
            sb.Append("{" + i.ToString() + "}");

        string format = sb.ToString();

        foreach(DataRow r in t.Rows)
            r[newColumn] = string.Format(format, columnsToMerge.Select(col => r[col]).ToArray() );
    }


}
于 2009-01-19T05:53:49.203 回答
1

如果数据集足够具体,那么 MergeName() 方法就可以了。如果您认为您不会在不止一次的地方使用它,则尤其如此。通用方法将需要更多的工作,但如果您正在进行大量合并,则可能是值得的。

我能想到的一些事情是你必须处理的:

  • 您可能会遇到的一个问题是类型检查。如果用户想要合并日期字段或数字字段会发生什么。
  • 旧田地怎么办?它们会保留还是被删除?如果您将其用作直接数据源,您可能希望删除剩余的列,因此某种RemoveOldColumns标志可能是个好主意。
于 2009-01-19T05:09:47.043 回答
1

在我看来,这应该发生在数据层而不是逻辑层。据推测,数据集消费者知道它需要什么数据,并且可以明确地从数据存储中请求它,而无需操作数据集。

在数据库场景中,我本质上是在提倡一种多存储过程方法,并且数据集使用者将从适当的方法中检索数据。

于 2009-01-19T05:11:33.220 回答
1

或者,您可以这样做:

    public Form1()
    {
        InitializeComponent();


        DataSet ds = new DataSet();
        ds.Tables.Add(new DataTable());
        ds.Tables[0].Columns.Add("id", typeof(int));
        ds.Tables[0].Columns.Add("firstname", typeof(string));
        ds.Tables[0].Columns.Add("lastname", typeof(string));


        ds.Tables[0].Rows.Add(1,"torvalds", "linus");
        ds.Tables[0].Rows.Add(2,"lennon", "john");    


        ds.Tables[0].Columns.Add("name", typeof(string), "lastname + ', ' + firstname"); 


        foreach (DataRow dr in ds.Tables[0].Rows)
            MessageBox.Show(dr["name"].ToString());


    }
于 2009-01-19T05:56:39.447 回答
0
        DataSet ds = new DataSet();
        ds.Tables.Add(new DataTable());
        ds.Tables[0].Columns.Add("id", typeof(int));
        ds.Tables[0].Columns.Add("firstname", typeof(string));
        ds.Tables[0].Columns.Add("lastname", typeof(string));


        ds.Tables[0].Rows.Add(1,"torvalds", "linus");
        ds.Tables[0].Rows.Add(2,"lennon", "john");


        ds.Tables[0].Columns.Add("name", typeof(string));
        foreach (DataRow dr in ds.Tables[0].Rows) dr["name"] = dr["lastname"] + ", " + dr["firstname"];


        foreach (DataRow dr in ds.Tables[0].Rows)
            MessageBox.Show(dr["name"]);
于 2009-01-19T05:34:10.193 回答
0

这正是我的要求,经过大量搜索后,我最终决定最好创建自己的数据表并将其用作 ListBox 或 DropDownList 的数据源,无论哪个需要。

DataTable dt = new DataTable();
dt.Columns.Add("Test",typeof(String));
foreach (DataRow r in ds.Tables[0].Rows)
{
  DataRow dr = dt.NewRow();
  dr["Test"] = r["LastName"] + " ," + r["firstName"];
  dt.Rows.Add(dr);
}

ds是包含该过程的主要返回表的数据集。

于 2011-11-15T11:50:47.447 回答
0
string result = string.Join(",", dtTotalQuery.AsEnumerable().Select(row => dtTotalQuery.Rows[0].ItemArray[0] + " " + dtTotalQuery.Rows[0].ItemArray[1]));
于 2015-07-03T05:29:55.207 回答
0

扩展方法带有可选的分隔符位置

public static class Helper
{
    public static void MergeColumns(this DataTable t, string newColumn, params string[] columnsToMerge)
    {
        t.Columns.Add(newColumn, typeof(string));

        var sb = new StringBuilder();

        sb.Append("{0}, ");
        for (int i = 1; i < columnsToMerge.Length; ++i)
            sb.Append("{" + i.ToString() + "}");

        string format = sb.ToString();

        foreach(DataRow r in t.Rows)
            r[newColumn] = string.Format(format, columnsToMerge.Select(col => r[col]).ToArray() );
    }
}

用法

//DataTable dt
dt.MergeColumns("name", 0, " ", "lastname", "firstname");
于 2018-09-17T09:54:48.490 回答