4.5 神经网络的编码(Encoding the Network ) 在本书的开始几章中,你已经看到过怎样用各种各样的方法为遗传算法编码。但当时我并没有向你介绍过一个用实数编码的具体例子,因为我知道我要留在这里向你介绍。我曾经讲到,为了设计一个前馈型神经网络,编码是很容易的。我们从左到右读每一层神经细胞的权重,读完第一个隐藏层,再向上读它的下一层,把所读到的数据依次保存到一个向量中,这样就实现了网络的编码。因此,如果我们有图14所示的网络,则它的权重编码向量将为: a4 |" e: Z, t" L& w; b' Z
0.3, -O.8, -O.2, 0.6, O.1, -0.l, 0.4, 0.5 / S/ s; n& N4 v( P! W3 D
在这一网络中,为了简单,我没有把偏移值的权重包括进去。但在实际实现编码时,你必须包含偏移值这个权重,否则你肯定无法获得你所需要的结果。 ( s; E* \ b% W* o6 W% r
! y: U0 j5 w3 v6 {
! P" U+ M' b* j3 A& y" d' Y5 ]
2 K$ f9 [5 H/ K& K3 {图14 为权重编码。 4 k3 r# h# X4 d; x4 c" u6 P" v+ e
在此之前讲的事情你都懂了吗?好极了,那下面就让我们转来考虑,怎样用遗传算法来操纵已编码的基因吧。 / O4 z& [. U9 ? v- ?
0 f. Z- I4 c3 D0 ~) w2 x
4.6 遗传算法(The Genetic Algorithm ) 到此,所有的权重已经象二进制编码的基因组那样,形成了一个串,我们就可以象本书早先讨论过的那样来应用遗传算法了。遗传算法(GA)是在扫雷机已被允许按照用户指定的帧数(为了某种缘故, 我下面更喜欢将帧数称作滴答数,英文是ticks)运转后执行的。你可以在ini文件中找到这个滴答数(iNumTicks)的设置。下面是基因组结构体的代码。这些对于你应该是十分面熟的东西了。
4 L6 O! X6 x! c- P0 }Struct SGenome { vector <double> vecWeights; 0 a2 {5 M* ^6 p M' t
double dFitness; ; F6 i" ]1 h+ [
SGenome():dFitness(0) {} , E& P8 h! t$ b9 M* X1 R) r5 O
SGenome(vector <double> w, double f):vecWeights(w),dFitness(f){}
" X* T; o) c. C$ G9 j3 U+ { //重载'<'的排序方法 friend bool operator<(const SGenome& lhs, const SGenome& rhs) { return (lhs.dFitness < rhs.dFitness); } };
7 F; R( \1 B }7 U 从上面的代码你可看出,这一SGenome结构和我们在本书所有其他地方见到的SGenome结构几乎完全一致,唯一的差别就是这里的染色体是一个双精度向量std::vector。因此,可以和通常一样来应用杂交操作和选择操作。但突变操作则稍微有些不同,这里的权重值是用一个最大值为dMaxPerturbation的随机数来搔扰的。这一参数dMaxPerturbation在ini文件中已作了声明。另外,作为浮点数遗传算法,突变率也被设定得更高些。在本工程中,它被设成为0.1。 下面就是扫雷机工程遗传算法类中所见到的突变函数的形式:
' ?# B6 x/ h- Y2 E! A$ `6 C% a3 Tvoid CGenAlg::Mutate(vector<double> &chromo) { // 遍历权重向量,按突变率将每一个权重进行突变 for (int i=0; i<chromo.size(); ++i) { // 我们要骚扰这个权重吗? if (RandFloat() < m_dMutationRate) { // 为权重增加或减小一个小的数量 chromo += (RandomClamped() * CParams::dMaxPerturbatlon); } } }
. \. C% s: M$ y9 | h! X8 Q! { 如同以前的工程那样,我已为v1.0版本的Smart Minesweepers工程保留了一个非常简单的遗传算法。这样就能给你留下许多余地,可让你利用以前学到的技术来改进它。就象大多数别的工程一样,v1.O版只用轮盘赌方式选精英,并采用单点式杂交。
' J+ F0 y1 o4 B注意: 当程序运行时,权重可以被演化成为任意的大小,它们不受任何形式的限制。 ( {, ?3 c" D' P" u$ |! v4 w- B8 U6 J/ i
$ U/ W/ n/ _1 O4.7 扫雷机类(The CMinesweeper Class ) - ~. j1 I$ T) Q
这一个类用来定义一个扫雷机。就象上一章描述的登月艇类一样,扫雷机类中有一个包含了扫雷机位置、速度、以及如何转换方向等数据的纪录。类中还包含扫雷机的视线向量(look-at vector);它的2个分量被用来作为神经网络的2个输入。这是一个规范化的向量,它是在每一帧中根据扫雷机本身的转动角度计算出来的,它指示了扫雷机当前是朝着哪一个方向,如图11所示。 下面就是CMinesweeper扫雷机类的声明: : n2 K& r( a6 x, t6 i
class CMinesweeper { private: // 扫雷机的神经网络 CNeuralNet m_ItsBrain; & L5 D; Z L2 ~3 h) |
// 它在世界坐标里的位置 SVector2D m_vPosition;
0 o7 m4 I& V' J. B2 T) N+ b# _ // 扫雷机面对的方向 SVector2D m_vLookAt; 9 b$ h# w1 E$ v" C6 w/ [) j d) j
// 它的旋转(surprise surprise) double m_dRotation; / F$ u5 Q! F$ |+ _, q# L
double m_dSpeed; - b0 M1 f, R `0 B: H0 t) r
// 根据ANN保存输出 double m_lTrack, m_rTrack;
5 }9 j' Z9 Q4 i) Q- {* h% I' @ j( b, A4 r* P$ H8 v. ^" e" \
! t4 \, C# ?6 d. }9 `7 c( X
+ \3 ~7 G* d: g m_lTrack和m_rTrack根据网络保存当前帧的输出。 这些就是用来决定扫雷机的移动速率和转动角度的数值。 " ?: c) V- G1 r D# V* w
. T6 A7 S1 l3 Q4 J/ \2 x D) k7 ~& s& l2 Z
6 D8 h% O2 v7 R, p% I // 用于度量扫雷机适应性的分数 double m_dFitness;
7 V6 w$ F5 S/ ~9 O
A. b8 h r6 F$ V* }# C! \( @3 u
3 ^: d2 f" m7 T. @9 Z3 e
, S' A! G8 h2 g' D1 x 每当扫雷机找到一个地雷,它的适应性分数就要增加。
3 M4 J+ Q- E. ?& a, _$ Y6 L+ p9 {
. h* P4 F+ K& _0 [) ^: C
3 `: V! j) M; f
4 K/ A& C3 P5 r3 f+ \; H // 扫雷机画出来时的大小比例 double m_dScale; 4 e; L6 ?8 F X* l! J
// 扫雷机最邻近地雷的下标位置 int m_iClosestMine;
6 r/ }0 U: G. T- l$ ~8 T b7 Q' X- @# ?- t! @) T& Y# U
1 z+ W1 Y1 y) o9 Y. F% f! q3 H, r g! m; h, c* ?
在控制器类CControl1er中,有一个属于所有地雷的成员向量std::vector。而m_iClosestMine就是代表最靠近扫雷机的那个地雷在该向量中的位置的下标。
1 A$ h4 u0 k5 e. t* T. j! o3 z Q; J3 j m7 @0 I) K u
3 E* h4 P& H- y7 W' G8 M; l2 A5 p5 ], ^/ Q
public: 9 ~2 t+ i9 T9 s) p
CMinesweeper(); , K m2 E% B* e
// 利用从扫雷机环境得到的信息来更新人工神经网 bool Update(vector<SVector2D> &mines);
, t9 T; g5 u4 K+ s- U // 用来对扫雷机各个顶点进行变换,以便接着可以画它出来 void WorldTransform(vector<SPoint> &sweeper);
8 f9 I7 [# a8 i' I# @5 v // 返回一个向量到最邻近的地雷 5Vector2D GetClosestMine(vector<SVector2D> &objects);
/ w* _2 d( e1 w) Q0 @. b // 检查扫雷机看它是否已经发现地雷 int CheckForMine(vector<SVector2D> &mines, double size);
7 w; i* g. t; S) w& g7 g( s void Reset();
- m* Y1 V$ L/ W* c* u9 Z$ o; T8 q // ----------------- 定义各种供访问用的函数 SVector2D Position()const { return m_vPosition; } void IncrementFitness(double val) { m_dFitness += val; } double Fitness()const { return m_dFitness; } void PutWeights(vector<double> &w) { m_ItsBrain.PutWeights(w); } int GetNumberOfWeights()const { return m_ItsBrain.GetNumberOfWeights(); } };
; P+ F1 k; ]1 L) h6 p
# j4 J* ]1 `! r6 `7 j8 {! Y4.7.1 The CMinesweeper::Update Function (扫雷机更新函数) / O- z+ }9 A# _5 D( L& z
需要更详细地向你说明的CMinesweeper类的方法只有一个,这就是Update更新函数。该函数在每一帧中都要被调用,以更新扫雷机神经网络。让我们考察这函数的肚子里有些什么货色:
3 G& n1 C# P' x( |9 gbool CMinesweeper::Update(vector<SVector2D> &mines)
$ O! V3 h$ P: B- k{ //这一向量用来存放神经网络所有的输入 vector<double> inputs; ; y& Q# v; w+ c# Z! w
//计算从扫雷机到与其最接近的地雷(2个点)之间的向量 SVector2D vClosestMine = GetClosestMine(mines);
( I, t, J+ v& }( j: @% o //将该向量规范化 Vec2DNormalize(vClosestMine);
- y1 [& p$ p6 ~3 [( {, q, m( w/ _- V) G$ I" G4 p* K
首先,该函数计算了扫雷机到与其最靠近的地雷之间的向量,然后使它规范化。(记住,向量规范化后它的长度等于1。)但扫雷机的视线向量(look-at vector)这时不需要再作规范化,因为它的长度已经等于1了。由于两个向量都有效地化成了同样的大小范围,我们就可以认为输入已经是标准化了,这我前面已讲过了。
4 h! ]* r, A" ]. M) E/ s4 D' F; g //加入扫雷机->最近地雷之间的向量 Inputs.push_back(vClosestMine.x); Inputs.push_back(vCIosestMine.y);
- D' d9 w$ S) A% v //加入扫雷机的视线向量 Inputs.push_back(m_vLookAt.x); Inputs.push_back(m_vLookAt.y);
# i" Z' X, `% k" K3 ]5 L5 a- E //更新大脑,并从网络得到输出 vector<double> output = m_ItsBrain.Update(inputs); ) W- k( L" D" x8 g& V, p* K" g
6 z& F% z$ R, `2 k; h 然后把视线向量,以及扫雷机与它最接近的地雷之间的向量,都输入到神经网络。函数CNeuralNet::Update利用这些信息来更新扫雷机网络,并返回一个std::vector向量作为输出。 . Q+ w8 T; W/ j7 I# h1 P
//保证在输出的计算中没有发生错误 if (output.size() < CParams::iNumOutputs) { return false; } 0 u, n& X! ]# T+ L2 m {
// 把输出赋值到扫雷机的左、右轮轨 m_lTrack = output[0]; m_rTrack = output[1];
& G) S8 T0 {/ m6 A' u% i: e* y! ]( A: k9 A% V
在更新神经网络时,当检测到确实没有错误时,程序把输出赋给m_lTrack和m_rTrack。 这些值代表施加到扫雷机左、右履带轮轨上的力。
1 W3 ~# W D6 X1 g! d, Q" v // 计算驾驶的力 double RotForce = m_lTrack - m_rTrack;
5 Z6 I6 q( N% _9 F3 V! N // 进行左转或右转 Clamp(RotForce, -CParams::dMaxTurnRate, CParams::dMaxTurnRate); , o0 {- x. P" t2 A3 J9 P+ g. j5 J5 ^9 ^
m_dSpeed = (m_lTrack + m_rTrack);
4 m6 }9 r, T: Z 扫雷机车的转动力是利用施加到它左、右轮轨上的力之差来计算的。并规定,施加到左轨道上的力减去施加到右轨道上的力,就得到扫雷机车辆的转动力。然后就把此力施加给扫雷机车,使它实行不超过ini文件所规定的最大转动率的转动。而扫雷机车的行进速度不过就是它的左侧轮轨速度与它的右侧轮轨速度的和。既然我们知道了扫雷机的转动力和速度,它的位置和偏转角度也就都能更新了。 O7 }1 v' S3 L; H
//更新扫雷机左右转向的角度 m_dRotation += RotForce; 3 l" R+ M6 K: j4 G
// 更新视线角度 m_vLookAt.x = -sin(m_dRotation); m_vLookAt.y = cos(m_dRotation); - i# Y# n" k; X$ v$ ?* J
// 更新它的位置 m_vPosition += (m_vLookAt* m_dSpeed);
3 g6 P. s3 X" \# r+ x, h: Q+ S: f8 R // 如果扫雷机到达窗体四周,则让它实行环绕,使它不至于离开窗体而消失 If (m_vPosition.x > CParams::WindowWidth) m_vPosition.x = 0; If (m_vPosition.x < 0) m_vPosition.x = CParams::WindowWidth; If (m_vPosition.y > CParams::WindowHeight) m_vPosition.y = 0; If (m_vPosition.y < D) m_vPosition.y = CParams::WindowHeight; ! z) k) n2 v9 o! t( d
为了使事情尽可能简单,我已让扫雷机在碰到窗体边框时就环绕折回(wrap)。采用这种方法程序就不再需 要做任何碰撞-响应方面的工作。环绕一块空地打转对我们人来说是一桩非常不可思议的动作,但对扫雷机,这 就像池塘中的鸭子。
/ B0 q: E5 R5 B9 _; h Returen true; } 1 e3 ~8 `# P) x$ N, W, ~" {
) }7 T6 G9 i3 A1 c7 a% K
4.8 CController Class (控制器类) CController类是和一切都有联系的类。图15指出了其他的各个类和CController类的关系。 下面就是这个类的定义:
( y- k J2 N9 V: E class CController { private: // 基因组群体的动态存储器(一个向量) vector<SGenome> m_vecThePopulation; ( _6 w9 e G# c. A) i2 _2 g+ l. E( T
: d9 u* h7 @( _& L* g; `
, H# W/ ?( J5 ~ Z+ e) s8 f
% d' H9 \7 ?; k V2 P 图15 minesweeper工程的程序流程图
8 k7 z1 ?6 q% D3 u+ C. T" K* b& [! i+ s: m) B" j
// 保存扫雷机的向量 vector<CMinesweeper> m_vecSweepers;
3 E2 p8 v( X: V* [% c( b! \$ p1 p // 保存地雷的向量 vector<SVector2D> m_vecMines; : T( ]' n# C6 ?/ O
1 M0 l7 f8 O4 D3 v& j+ ~% e$ N // 指向遗传算法对象的指针 CGenAIg* m_pGA; ; ^; {- T. j {( @
int m_NumSweepers;
' c; a& T& |2 w# b& k) \! c4 h int m_NumMines; : n# I2 q* q1 J# Q0 s" _0 E
// 神经网络中使用的权重值的总数 int m_NumWeightsInNN; ; i V* S. y$ X+ i* M, m. T. `2 e
// 存放扫雷机形状各顶点的缓冲区 vector<SPoint> m_SweeperVB;
( V0 u4 j- S6 @! Y; p* N1 B5 f // 存放地雷形状各顶点的缓冲区 vector<SPoint> m_MineVB;
4 N9 K O% @% ]% W // 存放每一代的平均适应性分数,供绘图用 vector<double> m_vecAvFitness; ' T8 A. o; S8 d" t0 @9 v( V
// 存放每一代的最高适应性分 vector<double> m_vecBestFitness;
. h" P% D* i W1 h, j1 d [ // 我们使用的各种不同类型的画笔 HPEN m_RedPen; HPEN m_BluePen; HPEN m_GreenPen; HPEN m_OldPen; * j& r7 _3 j) a/ O
// 应用程序窗口的句柄 HWND m_hwndMain; % L! {" b3 f1 _+ v
// 切换扫雷机程序运行的速度 bool m_bFastRender; 5 w+ Y, [/ g9 A3 r/ E7 v c, B
// 每一代的帧数(滴答数) int m_iTicks;
0 h: @# m* }4 @- x% Y J // 代的计数 int m_iGenerations;
) U& w! s& H+ H/ N // 窗体客户区的大小 int cxClient,cyClient;
+ o/ y5 b7 `8 P# P4 E# a // 本函数在运行过程中画出具有平均-,和最优适应性值的图 void PlotStats(HDC surface);
! T* o+ |( }' Y% ]& Ppublic:
3 m' `/ d$ X/ I) P4 y# t CController(HWND hwndMain); ) o6 i. J3 y% x# j& i5 f
~CController(); ) J. \% `4 ?. V" N7 Y8 s
void Render(HDC surface);
6 h( ^7 e0 d" n1 n+ {, N1 V void WorldTransform(vector<SPoint> &VBuffer, SVector2D vPos);
& o8 o! S% h& Y, e6 h8 ?( h! V% m6 ] bool Update(); ) S2 ^4 m2 J' B1 v2 {$ i
// 几个公用的访问方法 bool FastRender() { return m_bFastRender; } void FastRender(bool arg){ m_bFastRender = arg; } void FastRenderToggle() { m_bFastRender = !m_bFastRender; } };
1 m% w' i: \ W0 M$ d6 ~, f/ Y. J/ c8 T1 T8 j1 @
当创建CController类的某个实例时,会有一系列的事情发生:
6 \* {! L! e# n*创建CMinesweeper对象。 *统计神经网络中所使用的权重的总数,然后此数字即被利用来初始化遗传算法类的一个实例。 *从遗传算法对象中随机提取染色体(权重)并(利用细心的脑外科手术)插入到扫雷机的经网络中。 *创建了大量的地雷并被随机地散播到各地。 *为绘图函数创建了所有需要用到的GDI画笔。 *为扫雷机和地雷的形状创建了顶点缓冲区。
* O( r2 y( j+ |& b. ^+ l; B 所有的一切现都已完成初始化,由此Update方法就能在每一帧中被调用来对扫雷机进行演化。 1 s0 B# j# R# u% k$ ^
0 U+ U# }% g9 C1 A7 P! j5 k2 p
4.8.1 CController::Update Method (控制器的更新方法) ) M4 S8 U3 n% o l( V. a0 D
控制器更新方法CController::Update方法(或函数)在每一帧中都要被调用。当调用update函数时,函数的前一半通过对所有扫雷机进行循环,如发现某一扫雷机找到了地雷,就update该扫雷机的适应性分数。由于m_vecThePopulation包含了所有基因组的拷贝,相关的适应性分数也要在这时进行调整。如果为完成一个代(generation)所需要的帧数均已通过,本方法就执行一个遗传算法时代(epoch)来产生新一代的权重。这些权重被用来代替扫雷机神经网络中原有的旧的权重,使扫雷机的每一个参数被重新设置,从而为进入新一generation做好准备。
( R' J4 @- i, B0 t9 p4 E! z bool CController::Update() { // 扫雷机运行总数为CParams::iNumTicks次的循环。在此循环周期中,扫雷机的神经网络 // 不断利用它周围特有的环境信息进行更新。而从神经网络得到的输出,使扫雷机实现所需的 // 动作。如果扫雷机遇见了一个地雷,则它的适应性将相应地被更新,且同样地更新了它对应 // 基因组的适应性。 if (m_iTicks++ < CParams::iNumTicks) { for (int i=O; i<m_NumSweepers; ++i) { //更新神经网络和位置 if (!m_vecSweepers.Update(m_vecMines)) { //处理神经网络时出现了错误,显示错误后退出 MessageBox(m_hwndMain, 'Wrong amount of NN inputs!", "Error", MB_OK); 3 r0 I: l f, L- F8 r( s- y
return false; }
& ?! \& T; Y6 t' a# @* x // 检查这一扫雷机是否已经发现地雷 int GrabHit = m_vecSweepers.CheckForMine(m_vecMines, CParams::dMineScale);
) ]- J# a# I/ V! A$ Y" T if (GrabHit >= 0) { // 扫雷机已找到了地雷,所以要增加它的适应性分数 m_vecSweepers.IncrementFitness();
7 r2 ^" H( A" S: V // 去掉被扫雷机找到的地雷,用在随机位置放置的一个新地雷来代替 m_vecMines[GrabHit] = SVector2D(RandFloat() * cxClient, RandFloat() * cyClient); }
2 r' S' r0 |- z# s+ M // 更新基因组的适应性值 m-vecThePopulation.dFitness = m_vecSweepers.Fitness(); } } // 一个代已被完成了。 // 进入运行遗传算法并用新的神经网络更新扫雷机的时期 else { // 更新用在我们状态窗口中状态 m_vecAvFitness.push_back(m_pGA->AverageFitness()); m_vecBestFitness.push_back(m_pGA->BestFitness()); ; b8 X1 b0 A/ K& K. c* e, m3 r4 U
// 增加代计数器的值 ++m_iGenerations;
. v& V- o D$ N: F& n8 b // 将帧计数器复位 m_iTicks = 0;
6 T9 P8 d. D/ e% J" y/ e+ \ // 运行GA创建一个新的群体 m-vecThePopulation = m_pGA->Epoch(m_vecThePopulation); ' C6 X6 H! N* ]$ E& d6 u
// 在各扫雷机中从新插入新的(有希望)被改进的大脑 // 并将它们的位置进行复位,等 for(int i=O; i<m_NumSweepers; ++i) {m_vecSweepers.m_ItsBrain.PutWeights(m_vecThePopulation.vecWeights); 7 z3 N7 d" {0 s) R1 H
m_vecSweepers.Reset(); } } returen true; } 6 n8 ?+ V' Y* T% a1 r
( y4 E% h% z# |5 l; R- [/ l8 e6 M
概括起来,程序为每一世代做的工作是:
# I0 N" {- [* J% v2 H l.为所有扫雷机和为iNumTicks个帧组织循环,调用Update函数并根据情况增加扫雷机适应值的得分。 2.从扫雷机神经网络提取权重向量。 3.用遗传算法去演化出一个新的网络权重群体。 4.把新的权重插入到扫雷机神经网络。 5.转到第1步进行重复,直到获得理想性能时为止。
8 b" U2 H7 |' w/ |% x 最后,表3列出了Smart Sweepers工程 v1.0版所有缺省参数的设置值。
# L9 l6 ?4 D) U4 @# s
# K4 E) ?9 I9 @4.9 运行此程序 (Running the Program )
# R/ B2 G' l3 \) u 当你运行程序时,“F”键用来切换2种不同的显示状态,一种是显示扫雷机怎样学习寻找地雷,一种是示在运行期中产生的最优的与平均的适当性分数的统计图表。 当显示图表时,程序将会加速运行。
4 S3 [/ T/ b4 {+ R+ d: J4 h- I5 X Q0 k5 i/ E
|