Знаменитая игра Джона Конвея «Жизнь», благодаря своей простоте, занимательности и поучительность, реализовывалась программистами так много раз, что уступает вероятно только пресловутой сортировке «пузырьком».

Приведу, тем не менее, еще один вариант исполнения этой замечательной игры, целиком основанный на технологии LINQ в среде .NET — простой, компактный, без циклов и многомерных массивов.


Сразу оговорюсь, что задача достижения максимальной эффективности не ставилась.

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;

namespace LinqLife
{
    public struct Cell
    {
        public readonly int X, Y;

        public Cell(int x, int y) { X = x; Y = y; }
    }

    public class Life : IEnumerable<Cell>
    {
        private List<Cell> _cells = new List<Cell>();

        private static readonly int[] Liveables = { 2, 3 };

        public bool Next()
        {
            var died = _cells
                .Where(cell => !Liveables.Contains(Count(cell)))    // Условие смерти
                .ToArray();

            var born = _cells
                .SelectMany(Ambit)                                  // Все окружающие ячейки
                .Distinct()                                         // ...без дубликатов
                .Except(_cells)                                     // ...пустые
                .Where(cell => Count(cell) == 3)                    // Условие рождения
                .ToArray();

            if (died.Length == 0 && born.Length == 0)
                return false; // Нет изменений

            _cells = _cells
                .Except(died)
                .Concat(born)
                .ToList();

            return _cells.Any(); // Не все еще умерли?
        }

        private int Count(Cell cell)
        {
            return Ambit(cell)
                .Intersect(_cells)
                .Count();
        }

        private static IEnumerable<Cell> Ambit(Cell cell)
        {
            return Enumerable.Range(-1, 3)
                .SelectMany(x => Enumerable.Range(-1, 3)
                    .Where(y => x != 0 || y != 0) 					// Кроме центральной клетки
                    .Select(y => new Cell(cell.X + x, cell.Y + y)));
        }

        public override string ToString()
        {
            if (_cells.Count == 0)
                return string.Empty;

            var xmin = _cells.Min(cell => cell.X);
            var xmax = _cells.Max(cell => cell.X);
            var ymin = _cells.Min(cell => cell.Y);
            var ymax = _cells.Max(cell => cell.Y);

            var matrix = Enumerable.Range(xmin, xmax - xmin + 1)
                .Select(x => Enumerable.Range(ymin, ymax - ymin + 1)
                    .Select(y => _cells.Contains(new Cell(x, y))));

            return string.Join(Environment.NewLine,
                matrix.Select(row =>
                    string.Join("",
                        row.Select(b => b ? "X" : "."))));
        }

        public IEnumerator<Cell> GetEnumerator()
        {
            return _cells.GetEnumerator();
        }

        IEnumerator IEnumerable.GetEnumerator()
        {
            return GetEnumerator();
        }

        public void Add(int x, int y)
        {
            _cells.Add(new Cell(x, y));
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var life = new Life { { 1, 1 }, { 2, 2 }, { 3, 3 }, { 1, 2 }, { 1, 3 } };
            
            var i = 0;
            do
            {
                Console.WriteLine("#{0}\r\n{1}", i++, life);

                Console.ReadLine();

            } while (life.Next());

            Console.WriteLine(life.Any() ? "* Stagnation!" : "* Extinction!");
        }
    }
}

Замечания к коду:


  1. Текущее состояние игры — список координат живых клеток, обновляемый на каждом шаге. В других реализациях обычно используется представление игрового поля в виде двумерного массива.
  2. Использование операций над множествами: объединение, пересечение, вычитание.
  3. Ни единого явного цикла (в классе Life).
  4. Использование struct для Cell позволяет не заботиться о сравнении ячеек и обеспечивает эффективное управление памятью (.NET нативно поддерживает ValueType в типизированных коллекциях).
  5. Практически неограниченный размер игрового поля. Причем «бесплатно».
  6. Возможна простая оптимизация: замена тип _cell с List на HashSet и вызовы Except()/Concat() в Next() на циклы с Remove()/Add() даст значительный прирост скорости (правда, в ущерб элегантности кода).
  7. Возможна простая многопоточная оптимизация: AsParallel().
  8. Наследование от IEnumerable и метод Add() добавлены исключительно ради изящества инициализации.


Полагаю, мало кто сомневается, что LINQ позволяет легко писать компактный, выразительный и эффективный код. Пусть сей нехитрый пример послужит еще одним свидетельством тому.

UPD: Если ставить целью исключительно компактность, код мог бы выглядеть примерно так:

using System;
using System.Collections.Generic;
using System.Linq;

namespace LinqLife
{
    using Cell = Tuple<int, int>;

    public class Life
    {
        public static List<Cell> Next(List<Cell> cells)
        {
            Func<Cell, IEnumerable<Cell>> ambit = 
                cell => Enumerable.Range(-1, 3).SelectMany(x => Enumerable.Range(-1, 3)
                        .Where(y => x != 0 || y != 0)
                        .Select(y => new Cell(cell.Item1 + x, cell.Item2 + y)));
            Func<Cell, int> count = cell => ambit(cell).Intersect(cells).Count();
            return cells.SelectMany(ambit)
                .Where(cell => count(cell) == 3)
                .Union(cells.Where(cell => count(cell) == 2))
                .ToList();
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var world = new List<Cell> { new Cell(1, 1), new Cell(2, 2), new Cell(3, 3), new Cell(1, 2), new Cell(1, 3) };
            do
            {
                Console.WriteLine(Dump(world));
                Console.ReadLine();
            } while ((world = Life.Next(world)).Any());
        }

        public static string Dump(List<Cell> cells)
        {
            int xmin = cells.Min(x => x.Item1), xmax = cells.Max(x => x.Item1), 
                ymin = cells.Min(x => x.Item2), ymax = cells.Max(x => x.Item2);
            var matrix = Enumerable.Range(xmin, xmax - xmin + 1)
                .Select(x => Enumerable.Range(ymin, ymax - ymin + 1)
                    .Select(y => cells.Contains(new Cell(x, y))));
            return string.Join(Environment.NewLine, matrix.Select(row => string.Join("", row.Select(b => b ? "X" : "."))));
        }
    }
}

Комментарии (6)


  1. a553
    26.07.2015 20:01
    +2

    Возможна простая оптимизация: замена тип _cell с List на HashSet и вызовы Except()/Concat() в Next() на циклы с Remove()/Add() даст значительный прирост скорости (правда, в ущерб элегантности кода).
    Есть ExceptWith и UnionWith.


    1. lasalas Автор
      26.07.2015 20:08
      +1

      +1


  1. stepik777
    26.07.2015 23:39
    +5

    Игра жизнь на 72 языках. Даже на SQL кто-то написал, а на Mathematica программа как всегда в одну строчку.


  1. Denxc
    27.07.2015 06:16
    +2

    Не ново. И кода у вас многовато, кажется.
    habrahabr.ru/company/skbkontur/blog/187448 В конце статьи игра жизнь на C# + Linq.


  1. injecto
    27.07.2015 10:18
    +2

    Действительно, многословно. Мне понравилась вот эта обобщенная реализация «Жизни» (на Clojure).


    1. lasalas Автор
      27.07.2015 13:35
      -1

      Добиться лаконичности Clojure на C# удастся вряд ли, но приблизиться можно — см. UPD.