使用linq找到锯齿状数组中最左边和最右边的点?

时间:2015-01-17 06:05:36

标签: c# linq jagged-arrays

我确定这是某个地方,但我找不到它。

我有一个标准

StrokeCollection strokes

集合中的每个笔划由一系列点组成。因此,这是锯齿状的点数,Point [] []

我需要在StrokeCollection中找到最左边和最右边的点。如何用linq完成?

这基本上就是我要做的事情(没有成功):

var x = from stroke in strokes
           from point in stroke
         where min(point.x)
         select point

非常感谢任何帮助。

编辑:显然直接使用墨水手写笔点有一些问题,所以让我们

 Point[][] myStrokeCollection = new Point[strokes.Count][];
            for (int i = 0; i < strokes.Count; i++)
            {
                // creating a one-dimensional array of StylusPoints for the i row of myStrokeCollection.
                myStrokeCollection[i] = new Point[strokes[i].StylusPoints.Count];
                for (int j = 0; j < strokes[i].StylusPoints.Count; j++)
                {
                    myStrokeCollection[i][j] = new Point();
                    myStrokeCollection[i][j].X = strokes[i].StylusPoints[j].X;
                    myStrokeCollection[i][j].Y = strokes[i].StylusPoints[j].Y;
                }
            }

对于myStrokeCollection,最左边的点(x,y)和最右边的点(x,y)是否可以通过linq(一次通过?)获得? (希望没有定义其他类型)。

3 个答案:

答案 0 :(得分:2)

Here is what needs to be done

    var leftMostVal = strokeCollection.Min(p => p.Point.Min(q => q.Min(r => r.X)));
    var rightMostVal = strokeCollection.Max(p => p.Point.Max(q => q.Max(r => r.X)));

Here is fiddle https://dotnetfiddle.net/y5hZ7w

Or 

       public class StrokeCollection : List<Stroke>
        {
        }

        public class Stroke : List<Point>
        {
            public Point[][] Points { get; set; }
        }
        class Program
        {
            static void Main(string[] args)
            {
                StrokeCollection lst = new StrokeCollection();


                Random rnd1 = new Random(3);
                Random rnd2 = new Random(13);


                for (int i = 0; i < 10; i++)
                {
                    int x = rnd1.Next(1, 5);
                    var s = new Stroke { Points = new Point[x][] };
                    foreach (var r in Enumerable.Range(0, x))
                    {
                        int y = rnd1.Next(1, 5);
                        s.Points[r] = new Point[y];

                        for (int j = 0; j < y; j++)
                            s.Points[r][j] = new Point(rnd2.Next(-50, 80), rnd2.Next(-20, 20));
                    }
                    lst.Add(s);
                }

                Point leftMostPoint = Point.Empty;
                Point rightMostPoint = Point.Empty;

                lst.ForEach(p1 =>
                {
                    foreach (Point[] p2 in p1.Points)
                    {
                        foreach (var p3 in p2)
                        {
                            if (leftMostPoint == Point.Empty)
                                leftMostPoint = p3;

                            if (rightMostPoint == Point.Empty)
                                rightMostPoint = p3;

                            if (p3.X < leftMostPoint.X)
                                leftMostPoint = p3;

                            if (p3.X > rightMostPoint.X)
                                rightMostPoint = p3;
                        }
                    }
                });

                Console.WriteLine("Leftmost point " + leftMostPoint.ToString());
                Console.WriteLine("Rightmost point " + rightMostPoint.ToString());
           }
       }

编辑:更新这是您最终需要的

    System.Windows.Ink.StrokeCollection strokeCollection; //Set it to whatever    resultant you want collection to set 

    var l1 = strokeCollection.SelectMany(p => p.StylusPoints) 
                             .GroupBy(p => p.X) 
                             .OrderBy(p => p.Key) 
                             .ToList(); 

StylusPoint leftPoint = l1.First().FirstOrDefault(); 
StylusPoint rightPoint = l1.Last().FirstOrDefault();

答案 1 :(得分:1)

(from stroke in strokeCollection
from point in stroke
group point by 1 into g
select new { Left = g.Min (p => p.x), Right = g.Max (p => p.x)}).FirstOrDefault();

https://dotnetfiddle.net/QONEqu

答案 2 :(得分:1)

来自Sangram的聊天,这很有效:

var l1 = strokeCollection.SelectMany(p => p.StylusPoints) 
.GroupBy(p => p.X) 
.OrderBy(p => p.Key) 
.ToList(); 

StylusPoint leftPoint = l1.First().FirstOrDefault(); 
StylusPoint rightPoint = l1.Last().FirstOrDefault();