将两个数据表中的数值相加,生成一个新的数据表。

3

我有两个强类型的数据表 (dt1):

|FirstName|LastName|Val1|Val2|
|Tony     |Stark   |34  |35  |
|Steve    |Rogers  |12  |23  |
|Natasha  |Romanoff|2   |100 |

和第二个 (dt2)

|FirstName|LastName|Val1|Val2|
|Tony     |Stark   |16  |5   |
|Bruce    |Banner  |2   |1   |
|Steve    |Rogers  |54  |40  |

我将尝试创建一个新的数据表,其中我将为人员的价值相加。我需要进行外部连接,因为我需要所有人员,并且第二个表中的值减半。
因此,结果应该如下所示:
|FirstName|LastName|Val1|Val2|
|Tony     |Stark   |42  |37.5|
|Steve    |Rogers  |39  |43  |
|Natasha  |Romanoff|2   |100 |
|Bruce    |Banner  |1   |0.5 |

我的方法是使用LINQ:

Dim query = 
from a in ds1.Table1
Join b in ds2.Table2
On a.FirstName + a.LastName Equals b.FirstName + b.Lastname
Select New With {
.FirstName = a.FirstName,
.LastName = a.LastName,
.Val1 = a.Val1 + b.Val1 *0.5,
.Val2 = a.Val2 + b.Val2 *0.5
}

但我不太理解所有人都采用这种方法。我也尝试过。
Dim query = 
From a in ds1.Table1
From b in ds2.Table2
Select New With{
Key .KeyName = a.FirstName + a.LastName = b.FirstName + b.FirstName,
.Val1 = a.Val1 + b.Val1 *0.5,
.Val2 = a.Val2 + b.Val2 * 0.5
}

现在,每个人都有很多条目。有没有人能帮我完成这个任务?我不知道是否有其他方法可以解决,而不需要使用Linq。


2
你需要使用LINQ进行全外连接(Full Outer JOIN) - Habib
我会使用for循环和字典。如果你真的想用LINQ,可以看看这个:在LINQ中进行左外连接。关键是.DefaultIfEmpty(),你需要先进行左外连接再进行右外连接,这样就能得到全连接了。 - Victor Zakharov
使用Group Join子句执行左外连接。 - Tim Schmelter
另一个选项是使用group by进行联合。 - the_lotus
你尝试过将解决方案应用到你的数据表中吗?它是否起作用? - Vland
谢谢你的回答!我会在周一尝试,今天我们放假,我无法访问数据库。不用担心,我不会忘记这个问题! - ruedi
2个回答

0

我现在明白了。我的解决方案是:

Dim qLeft = From a in dt1.Getdt1Info
                Group Join b in dt2.Getdt2Info
                On a.FirstName + a.LastName Equals b.FirstName + b.LastName  
                Into NewGroup = Group
                From c in NewGroup.DefaultIfEmpty
                Select New With{
                .Name = a.FirstName + ", " + a.LastName,
                .Value1 = a.Val1 + If (c is nothing, 0D, c.Val1) * 0.5,
                .Value2 = a.Val2 + If (c is nothing, 0D, c.Val2) * 0.5      
                }

Dim qRight =From a in dt2.Getdt2Info
                Group Join b in dt1.Getdt1Info  
                On a.FirstName + a.LastName Equals b.FirstName + b.LastName
                Into NewGroup = Group
                From c in NewGroup.DefaultIfEmpty
                Select New With{
                .Name = a.FirstName + ", " + a.LastName,
                .Value1 = a.val1 * 0.5 + If (c Is Nothing, 0D, c.Val1),
                .Value2 = a.val2 * 0.5 + If (c Is Nothing, 0d, c.Val2)}

Dim qFull = qleft.Concat(qRight).GroupBy(function(x) x.Name).Select(function(x) x.First)

我还不知道为什么qleft.Union(qRight)不能消除重复项。我用函数(参见代码)解决了这个问题。


0

使用group by的示例。需要考虑的是,在执行group bySum操作之前,第二个表中的所有值都必须除以2。我会在.Concat之前处理它。

var dt1 = new List<MyClass>();
dt1.Add(new MyClass { FirstName = "Tony", LastName = "Stark", Val1 = 34, Val2 = 35});
dt1.Add(new MyClass { FirstName = "Steve", LastName = "Rogers", Val1 = 12, Val2 = 23});
dt1.Add(new MyClass { FirstName = "Natasha", LastName = "Romanoff", Val1 = 2, Val2 = 100 });


var dt2 = new List<MyClass>();
dt2.Add(new MyClass { FirstName = "Tony", LastName = "Stark", Val1 = 16, Val2 = 5 });
dt2.Add(new MyClass { FirstName = "Bruce", LastName = "Banner", Val1 = 2, Val2 = 1 });
dt2.Add(new MyClass { FirstName = "Steve", LastName = "Rogers", Val1 = 54, Val2 = 40 });


var q = from a in dt1
                .Concat(
                    from b in dt2 
                    select new MyClass 
                        {
                            FirstName = b.FirstName,
                            LastName = b.LastName,
                            Val1 = b.Val1 * 0.5m,
                            Val2 = b.Val2 * 0.5m
                        })
        group a by new {a.FirstName, a.LastName}
        into g
        select new
            {
                g.First().FirstName,
                g.First().LastName,
                Val1 = g.Sum(x => x.Val1),
                Val2 = g.Sum(x => x.Val2),
            };


foreach (var s in q)
{
    Console.WriteLine("{0} {1} {2} {3}", s.FirstName,s.LastName,s.Val1,s.Val2);
}

结果

Tony Stark 42,0 37,5
Steve Rogers 39,0 43,0
Natasha Romanoff 2 100
Bruce Banner 1,0 0,5

嗨!你的代码运行得很完美。我还在寻找一个左/右->全连接的解决方案。感谢你的帮助! - ruedi
我不在乎那些愚蠢的分数。你得到了它们。只是,所有上面的评论都让我想到了一个完整连接的解决方案(请跟随链接)。我想...好吧,试试看。顺便说一句,我不会因为检查自己的答案而得到分数,但我会因为检查你的答案而得到分数。 - ruedi
@ruedi 好的,抱歉我不知道。评论员给了一些提示但没有实际解决方案。你查询的问题在于值被减半,所以我认为在将值减少到50%之前,你不能进行组合连接(至少不能在单个查询中)。这就是为什么我认为这是一个非常有趣的问题。 - Vland

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接