遗传算法案例一:基于感官的移动

首先,看一下我们最终训练出来的结果是这样的(这个gif图片我截得非常不满意,后面有一张比较完整的训练动图,大家可以先拉到后方看一下,这篇文章可是非常长的。。。)


遗传算法结果.gif

第一步:在场景里新建一个Cube(黑色)和一个Plane(蓝色),并改变改变他们的缩放和位置,实现如下图所示的场景;然后将Cube命名为Dead,Plane命名为Ground。


基本场景搭建.png

第二步:新建一个胶囊体和一个立方体,将立方体设为胶囊体的子物体,然后将胶囊体命名为People,立方体命名为Eye,并且调整他们的位置和旋转角度成如下图所示:(请务必保持Cube的轴向如图所示(60,0,0),因为到时我们会通过Cube的蓝色轴方向发射射线,来检测地面)
物体的轴向.png

物体的层级关系.png

第三步:
1.选中People,给他添加Rigibody组件,并冻结刚体xyz轴的旋转。
2.修改People的层,将layer改为bot(需要自己先创建)。


物体需要更改的属性参数.png

第四步:
1.新建一个空物体,并拖拽到蓝色地面的正上方,以后他将会作为我们People的出生点。
2.将People变成预制体,并删掉场景里的People。

第五步:新建3个脚本,分别命名为MyDNA,MyBrain,MyPopulationManager

MyDNA脚本编写以下代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class MyDNA
{
    //这个列表里存储的数值,是为了影响物体的行动
    List<int> genes = new List<int>();
    int dnaLength = 0;
    int maxValues = 0;
        
    /// <summary>
    /// 自己编写构造函数来实现这个类实例化时,进行一些初始的赋值
    /// </summary>
    /// <param name="l">基因列表的长度</param>
    /// <param name="v">基因列表中影响物体行动的随机整数的最大值</param>
    public MyDNA (int l,int v)
    {
        dnaLength = l;
        maxValues = v;
        SetRandom();
    }
        
    /// <summary>
    /// 对基因列表进行随机数赋值(因为我们的目标是实现优胜劣汰,所以要有随机数产生各种
    /// 各样的“性状”,来进行训练)
    /// </summary>
    private void SetRandom()
    {
        genes.Clear();
        for (int i = 0; i < dnaLength; i++)
        {
            genes.Add(Random.Range(0, maxValues));
        }
    }
        
        
    /// <summary>
    /// 这个函数会将传进来的父亲和母亲的DNA序列进行重组(各取一半,取d1的基因列表
    /// 前半部分,取d2的基因列表后半部分)
    /// </summary>
    /// <param name="d1">父亲的DNA</param>
    /// <param name="d2">母亲的DNA</param>
    public void Combine(MyDNA d1,MyDNA d2)
    {
        for (int i = 0; i < dnaLength; i++)
        {
            if (i < dnaLength / 2.0)
            {
                int c = d1.genes[i];
                genes[i] = c;
            }
            else
            {
                int c = d2.genes[i];
                genes[i] = c;
            }
        }
    }
    
    /// <summary>
    /// 对基因列表的某个随机的元素进行赋一个随机的"行为"值(因为列表里的元素的值
    /// 是可以影响物体的运动行为)
    /// </summary>
    public void Mutate()
    {
        genes[Random.Range(0, dnaLength)] = Random.Range(0, maxValues);
    }
    
    /// <summary>
    /// 返回基因列表里的某个元素的值
    /// </summary>
    /// <param name="pos">元素的下标位置</param>
    /// <returns></returns>
    public int GetGene(int pos)
    {
            return genes[pos];
    }
}

MyBrain脚本编写如下代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class MyBrain : MonoBehaviour
{
    int DNALength = 2;
    //用来记录存活时间,
    public float timeAlive;
    public float timeWalking;
    public MyDNA dna;
    public GameObject eyes;
    bool alive = true;
    bool seeGround = true;
        
    
    /// <summary>
    /// 这个是unity的回调函数,当两个碰撞体/刚体相撞时,unity会自动调用此函数
    /// </summary>
    /// <param name="collision">碰撞到的物体</param>
    private void OnCollisionEnter(Collision collision)
    {
        if (collision.gameObject.name == "Dead")
        {
            alive = false;
            timeWalking = 0;
            timeAlive = 0;
        }
    }

    /// <summary>
    /// Brain进行初始化赋值的函数,每个挂在这个脚本的物体,都会进行初始化赋值
    /// </summary>
    public void Init()
    {
        //列表长度为2,每个元素存在3种动作状态,实例化后将随机赋予一种
        //实例化后,每个数字的意义:0 向前走 1左转 2右转
        //dna.genes[0]看到地面,dna.genes[1]看不到地面
        dna = new MyDNA(DNALength, 3);
        timeAlive = 0;
        alive = true;
    }
    
    void Update()
    {
        if (!alive) return;
        Debug.DrawRay(eyes.transform.position, eyes.transform.forward * 10, Color.red, 10);
        seeGround = false;
        RaycastHit hit;
        if (Physics.Raycast(eyes.transform.position, eyes.transform.forward * 10, out hit))
        {
            if (hit.collider.gameObject.name == "Ground")
            {
                seeGround = true;
            }
        }
        //不能让他们一直无限制的运行下去,即使是最优秀的基因,他们最多存活时间也不可能超过极限。
        timeAlive = MyPopulationManager.elapsed;

        float turn = 0;
        float move = 0;
        if (seeGround)
        {
            if (dna.GetGene(0) == 0) { move = 1; timeWalking += 1; }
            else if (dna.GetGene(0) == 1) turn = -90;
            else if (dna.GetGene(0) == 2) turn = 90;
        }
        else
        {
            if (dna.GetGene(1) == 0) { move = 1; timeWalking += 1; }
            else if (dna.GetGene(1) == 1) turn = -90;
            else if (dna.GetGene(1) == 2) turn = 90;
        }

        this.transform.Translate(0, 0, move * 0.1f);
        this.transform.Rotate(0, turn, 0);
    }
}

MyPopulationManager编写如下代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System.Linq;
    
public class MyPopulationManager : MonoBehaviour
{
    public GameObject botPrefab;
    public int populationSize = 50;
    //用来储存实例化出来的"人"
    List<GameObject> population = new List<GameObject>();
    //可以看做是人类寿命。
    public static float elapsed = 0;
    public float trailTime = 5;
    int generation = 1;

    GUIStyle guiStyle = new GUIStyle();      
    /// <summary>
    /// 通过UI的方式向屏幕输出一些信息,方便查看当前进化状态
    /// </summary>
    private void OnGUI()
    {
        guiStyle.fontSize = 25;
        guiStyle.normal.textColor = Color.white;
        GUI.BeginGroup(new Rect(10, 10, 250, 150));
        GUI.Box(new Rect(0, 0, 140, 140), "Stats", guiStyle);
        GUI.Label(new Rect(10, 25, 200, 30), "Gen:" + generation, guiStyle);
        GUI.Label(new Rect(10, 50, 200, 30), string.Format("Time:{0:0.00}", elapsed), guiStyle);
        GUI.Label(new Rect(10, 75, 200, 30), "Population:" + population.Count, guiStyle);
        GUI.EndGroup();
    }
    

    void Start()
    {  
        for (int i = 0; i < populationSize; i++)
        {
            Vector3 startingPos = new Vector3(this.transform.position.x + Random.Range(-2, 2),
            this.transform.position.y,
            this.transform.position.z + Random.Range(-2, 2));
            GameObject b = Instantiate(botPrefab, startingPos, this.transform.rotation);
            b.GetComponent<MyBrain>().Init();
            population.Add(b);
        }
    }
    
    /// <summary>
    /// 这个函数的核心就是调用了MyDNA脚本里的Combine函数,实现父母基因的重组功能。
    /// 此外还有百分之一的概率会调用MyDNA脚本里的Mutate函数,实现自然界中的基因突变功能。
    /// </summary>
    /// <param name="parent1">父亲</param>
    /// <param name="parent2">母亲</param>
    /// <returns></returns>
    GameObject Breed(GameObject parent1, GameObject parent2)
    {
        Vector3 startingPos = new Vector3(this.transform.position.x + Random.Range(-2, 2),this.transform.position.y,this.transform.position.z + Random.Range(-2, 2));
        GameObject offspring = Instantiate(botPrefab, startingPos, this.transform.rotation);
        MyBrain b = offspring.GetComponent<MyBrain>();
        //模仿自然界中的变异,有百分之一的概率,在DNA序列中的随机位置产生随机的数字。
        //否则的话,会将父亲的基因和母亲的基因进行分离重组。
        //在MyDNA脚本里有注释解释了Combine函数的作用。
        if (Random.Range(0, 100) == 1)
        {
            b.Init();
            b.dna.Mutate();
        }
        else
        {
            b.Init();
            b.dna.Combine(parent1.GetComponent<MyBrain>().dna,parent2.GetComponent<MyBrain>().dna);
        }
        return offspring;
    }
    
    /// <summary>
    /// 这个脚本的核心
    /// </summary>
    void BreedNewPopulation()
    {
        //这条语句的作用就是实现:根据物体的存活时间和行走时间的长短,来对元素重新排序(升序)
        //因为我们认为行走比存活更重要一些,所以对行走进行加权。
        //查询表达式 
        //List<GameObject> sortedList = (from o in population
        //                               orderby o.GetComponent<MyBrain>().timeAlive+
        //                                       o.GetComponent<MyBrain>().timeWalking*5
        //                               select o).ToList();
            
        //用Lambda表达式
        List<GameObject> sortedList = population.OrderBy(o => (o.GetComponent<MyBrain>().timeWalking * 5 +o.GetComponent<MyBrain>().timeAlive)).ToList();
    
        //把上一代的物体清除掉,用来储存新的一代的物体。
        population.Clear();
        for (int i = (int)(sortedList.Count / 2.0f) - 1; i < sortedList.Count - 1; i++)
        {
            //假设有两个基因良好的物体1,2。
            //首先让1当父亲,2当母亲来组成一个新的基因
            //然后让2当父亲,1当母亲来组成一个新的基因
            //目的1.是加快产生优良基因的速度2.是为了能让这些存活下来排名较高的基因能填满人口列表
            population.Add(Breed(sortedList[i], sortedList[i + 1]));
            population.Add(Breed(sortedList[i + 1], sortedList[i]));
        }
        //销毁掉存放在列表的排序好的所有的上一代物体。
        for (int i = 0; i < sortedList.Count; i++)
        {
            Destroy(sortedList[i]);
        }
        generation++;
    }
    
    
    void Update()
    {
        elapsed += Time.deltaTime;
        //当场景里的物体存活时间超过极限时间,强制结束这一代,重新开始产生新人
        if (elapsed >= trailTime)
        {
            BreedNewPopulation();
            elapsed = 0;
        }
    }
}

第六步:将MyBrain脚本添加到People预制体上,并将Eye拖到空开的字段上。


场景环境配置1.png

第七步:将MyPopulationManager脚本添加到PopulationManager物体上并将People预制体拖动到空开的字段上。


场景环境配置2.png

第八步:点击菜单栏Edit—>ProjectSettings—>Physics,关闭bot层碰撞。(因为People上面有刚体,我们的People相互碰撞,会影响我们的一些数据,所以我们要关闭掉它)


场景环境配置3.png

第九步:运行,等待结果。


遗传算法训练过程和结果.gif

第十步:因为篇幅所限,我没有教大家如何调试及优化,直接把项目最终形式的代码都写好了。其实做项目不要想着一上来就做的完美,要先完成一个小功能,在一点点的进行调试完善。

最后也希望大家水平越来越高。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容