首页 > 代码库 > 关于康威生命游戏的学习

关于康威生命游戏的学习

 

       康威生命游戏,又称康威生命棋,是英国数学家约翰·何顿·康威1970年发明的细胞自动机它最初于1970年10月在《科学美国人》杂志上马丁·葛登能的“数学游戏”专栏出现。

         一个偶然机会发现了这个美妙的生命游戏。于是开始写程序来实现它,其实实现是很简单的,我现在还只能做到这个而已,不过还会继续深究下去的。先用Random来生成随机数,这样就可以在初始的时候在随机位置设定生死细胞了。输出的时候用2个for循环就是了。昨天做的时候遇到的问题是不知道最后该怎么判断了,因为判断一个改一个的话,可能改了一个就会对周围的产生影响了。最后导致的是世界里的细胞都活了。后来采用了随机产生一个坐标去判断然后刷新一次的办法,也算是稍微公平一点了。

       今天弄了2个二维数组,先把判断之后的赋值给第二个数组。然后新旧数组交替变换就好了,总之只用输出新数组就ok了。下面来贴个代码。

  #region 规则介绍(生命游戏)
    //1.当前细胞为死亡状态时,当周围有3个存活细胞时,该细胞变成存活状态。 (模拟繁殖)
    //2.当前细胞为存活状态时,当周围低于2个(不包含2个)存活细胞时, 该细胞变成死亡状态。(模拟生命数量稀少)
    //3.当前细胞为存活状态时,当周围有2个或3个存活细胞时, 该细胞保持原样。
    //4.当前细胞为存活状态时,当周围有3个以上的存活细胞时,该细胞变成死亡状态。(模拟生命数量过多)
    #endregion
    class Program
    {
        static void Main(string[] args)
        {
            #region 设定两个二维数组
            int sideLength = 20;
            string[,] background = new string[sideLength, sideLength];
            string[,] toBackground = new string[sideLength, sideLength];
            #endregion

            #region 模板
            for (int r=0;r<20;r++)
            {
                for(int c=0;c<20;c++)
                {
                    background[r, c] = "╋";
                }
            }
            Output(background);
            #endregion

            #region 设定初始的细胞
            for (int i=0;i<200;)
            {
                Random ran1 = new Random();
                int x = ran1.Next(20);
                int y = ran1.Next(20);
                if(background[x,y]!="●"&&background[x,y]!="〓")
                {
                    background[x, y] = "●";
                    i++;
                }
            }
            Console.Clear();
            Output(background);
            for (int i = 0; i < 200; )
            {
                Random ran1 = new Random();
                int x = ran1.Next(20);
                int y = ran1.Next(20);
                if (background[x, y] != "●" && background[x, y] != "〓")
                {
                    background[x, y] = "〓";
                    i++;
                }
            }
            Console.Clear();
            Output(background);
            #endregion

            #region 初始化第二个二维数组已保存更改
            for (int r = 0; r < 20; r++)
            {
                for (int c = 0; c < 20; c++)
                {
                    toBackground[r, c] = background[r, c];
                }
            }
            #endregion

            #region 按规则演变
            while (true)
            {
             
                for (int r = 0; r < 20; r++)
                {
                    for (int c = 0; c < 20; c++)
                    {
                        if (background[r, c] == "〓")
                        {
                            if (Ruler(r, c, background) >= 3)
                            {
                                toBackground[r, c] = "●";
                            }
                        }
                        if (background[r, c] == "●")
                        {
                            if (Ruler(r, c, background) < 2)
                            {
                                toBackground[r, c] = "〓";
                            }
                            if (Ruler(r, c, background) == 2 || Ruler(r, c, background) == 3)
                            {
                                toBackground[r, c] = "●";
                            }
                            if (Ruler(r, c, background) >3)
                            {
                                toBackground[r, c] = "〓";
                            }
                        }
                    }
                }
                Console.Clear();
                Output(toBackground);
                Thread.Sleep(1000);

                for (int r = 0; r < 20; r++)
                {
                    for (int c = 0; c < 20; c++)
                    {
                        background[r, c] = toBackground[r, c];
                    }
                }
            }
            #endregion
        }
        #region 打印输出
        static void Output(string[,] background)
        {
            for (int r = 0; r < background.GetLength(0); r++)
            {
                for (int c = 0; c < background.GetLength(1); c++)
                {
                    Console.Write(background[r, c]);
                }
                Console.WriteLine();
            }
        }
        #endregion

        #region 规则
        static int Ruler(int r,int c,string[,] background)
        {
            int count = 0;
            if (r > 0 && r < 19 && c > 0 && c < 19)
            {
                if (background[r - 1, c - 1] == "●")
                {
                    count += 1;
                }
                if (background[r - 1, c] == "●")
                {
                    count += 1;
                }
                if (background[r - 1, c + 1] == "●")
                {
                    count += 1;
                }
                if (background[r, c - 1] == "●")
                {
                    count += 1;
                }
                if (background[r, c] == "●")
                {
                    count += 1;
                }
                if (background[r, c + 1] == "●")
                {
                    count += 1;
                }
                if (background[r + 1, c - 1] == "●")
                {
                    count += 1;
                }
                if (background[r + 1, c] == "●")
                {
                    count += 1;
                }
                if (background[r + 1, c + 1] == "●")
                {
                    count += 1;
                }
                return count;
            }
            else
                return count;
        }
        #endregion
    }

哈哈,要考试了,考完试弄一个WPF的程序。下面我把规则什么的贴出来,看到这个博客的朋友们也可以看看。

规则

生命游戏中,对于任意细胞,规则如下:
每个细胞有两种状态-存活或死亡,每个细胞与以自身为中心的周围八格细胞产生互动。(如图,黑色为存活,白色为死亡)

  1. 当前细胞为死亡状态时,当周围有3个存活细胞时,该细胞变成存活状态。 (模拟繁殖)
  2. 当前细胞为存活状态时,当周围低于2个(不包含2个)存活细胞时, 该细胞变成死亡状态。(模拟生命数量稀少)
  3. 当前细胞为存活状态时,当周围有2个或3个存活细胞时, 该细胞保持原样。
  4. 当前细胞为存活状态时,当周围有3个以上的存活细胞时,该细胞变成死亡状态。(模拟生命数量过多)

可以把最初的细胞结构定义为种子,当所有在种子中的细胞同时被以上规则处理后, 可以得到第一代细胞图。按规则继续处理当前的细胞图,可以得到下一代的细胞图,周而复始。

概述

生命游戏是一个零玩家游戏。它包括一个二维矩形世界,这个世界中的每个方格居住着一个活着的或死了的细胞。一个细胞在下一个时刻生死取决于相邻八个方格中活着的或死了的细胞的数量。如果相邻方格活着的细胞数量过多,这个细胞会因为资源匮乏而在下一个时刻死去;相反,如果周围活细胞过少,这个细胞会因太孤单而死去。实际中,玩家可以设定周围活细胞的数目怎样时才适宜该细胞的生存。如果这个数目设定过高,世界中的大部分细胞会因为找不到太多的活的邻居而死去,直到整个世界都没有生命;如果这个数目设定过低,世界中又会被生命充满而没有什么变化。

实际中,这个数目一般选取2或者3;这样整个生命世界才不至于太过荒凉或拥挤,而是一种动态的平衡。这样的话,游戏的规则就是:当一个方格周围有2或3个活细胞时,方格中的活细胞在下一个时刻继续存活;即使这个时刻方格中没有活细胞,在下一个时刻也会“诞生”活细胞。

在这个游戏中,还可以设定一些更加复杂的规则,例如当前方格的状况不仅由父一代决定,而且还考虑祖父一代的情况。玩家还可以作为这个世界的“上帝”,随意设定某个方格细胞的死活,以观察对世界的影响。

在游戏的进行中,杂乱无序的细胞会逐渐演化出各种精致、有形的结构;这些结构往往有很好的对称性,而且每一代都在变化形状。一些形状已经锁定,不会逐代变化。有时,一些已经成形的结构会因为一些无序细胞的“入侵”而被破坏。但是形状和秩序经常能从杂乱中产生出来。

这个游戏被许多计算机程序实现了。Unix世界中的许多黑客喜欢玩这个游戏,他们用字符代表一个细胞,在一个计算机屏幕上进行演化。比较著名的例子是,GNU Emacs编辑器中就包括这样一个小游戏。

原理

细胞自动机(又称元胞自动机),名字虽然很深奥,但是它的行为却是非常美妙的。所有这些怎样实现的呢?我们可以把计算机中的宇宙想象成是一堆方格子构成的封闭空间,尺寸为N的空间就有N*N个格子。而每一个格子都可以看成是一个生命体,每个生命都有两种状态,如果该格子生就显示蓝色,死则显示白色。每一个格子旁边都有邻居格子存在,如果我们把3*3的9个格子构成的正方形看成一个基本单位的话,那么这个正方形中心的格子的邻居就是它旁边的8个格子。

每个格子的生死遵循下面的原则:

1. 如果一个细胞周围有3个细胞为生(一个细胞周围共有8个细胞),则该细胞为生(即该细胞若原先为死,则转为生,若原先为生,则保持不变) 。

2. 如果一个细胞周围有2个细胞为生,则该细胞的生死状态保持不变;

3. 在其它情况下,该细胞为死(即该细胞若原先为生,则转为死,若原先为死,则保持不变)

设定图像中每个像素的初始状态后依据上述的游戏规则演绎生命的变化,由于初始状态和迭代次数不同,将会得到令人叹服的优美图案。

这样就把这些若干个格子(生命体)构成了一个复杂的动态世界。运用简单的3条作用规则构成的群体会涌现出很多意想不到的复杂性为,这就是复杂性科学的研究焦点。

细胞自动机有一个通用的形式化的模型,每个格子(或细胞)的状态可以在一个有限的状态集合S中取值,格子的邻居范围 是一个半径r,也就是以这个格子为中心,在距离它r远的所有格子构成了这个格子的邻居集合,还要有一套演化规则,可以看成是一个与该格子当前状态以及邻居 状态相关的一个函数,可以写成f:S*S^((2r)^N-1)->S。这就是细胞自动机的一般数学模型。

最早研究细胞自动机的科学家是冯·诺伊曼,后来康韦发明了上面展示的这个最有趣的细胞自动机程序:《生命游戏》,而 wolfram则详尽的讨论了一维世界中的细胞自动机的所有情况,认为可以就演化规则f进行自动机的分类,而只有当f满足一定条件的时候,系统演化出来的 情况才是有活力的,否则不是因为演化规则太死板而导致生命的死亡,就是因为演化规则太复杂而使得随机性无法克服,系统乱成一锅粥,没有秩序。后来人工生命 之父克里斯·朗顿进一步发展了元胞自动机理论。并认为具有8个有限状态集合的自动机就能够涌现出生命体的自复制功能。他根据不同系统的演化函数f,找到了 一个参数lamda用以描述f的复杂性,得出了结论只有当lamda比混沌状态的lamda相差很小的时候,复杂的生命活系统才会诞生,因此,朗顿称生命 诞生于“混沌的边缘”!并从此开辟了“人工生命”这一新兴的交叉学科!

如今细胞自动机已经在地理学、经济学、计算机科学等领域得到了非常广泛的应用。

关于康威生命游戏的学习