是这样的,这两天在写一个智熄打地鼠的代码,在生成地鼠的部分,采用了协程的方式,保证间隔一段时间生成一个地鼠(也有用InvokeRepeating试过,基于我第一版的代码,InvokeRepeating可能更方便一些,但是我最后还是用了协程,毕竟协程这种挂起的方式还蛮节省开销的
ok言归正传, 这是我第一版的代码
private void Start(){
StartCoroutine(RandomAppearRoutine());
}
IEnumerator RandomAppearRoutine(){
while(true){
GameObject mole = RandomMoles();
yield return new WaitForSeconds(3);
}
}
后面想说通过控制按钮来确定开始生成和停止生成, 于是改成这样。 原先是在对象池里面进行控制的, 现在就转移到了GameManager,就看上去逻辑更清晰。这个时候,有一个start的按钮和一个stop的按钮来控制游戏的进行。
public void RandomAppear(){
StartCoroutine(RandomAppearRoutine());
}
IEnumerator RandomAppearRoutine(){
while(GameManager.Instance.IsGameLoading()){
GameObject mole = RandomMoles();
yield return new WaitForSeconds(3);
}
}
现在有个问题是它在我按下按钮的一瞬间就开始生成,这种措手不及的状态我们肯定是不希望看到的,于是我们需要一个缓冲时间。一开始是这么改的,欸对调一下。但是新的bug就出来了,在地鼠没出现的状态按下stop按钮时,我们已经经过了isGameLoading()
的判断,也就是说挂起时间结束后,后面的生成代码还会继续执行,所以这个就行不通啦。而且我的ZEN模式中,生成时间间隔大,缓冲区也设置那么大就不太好。
IEnumerator RandomAppearRoutine(){
while(GameManager.Instance.IsGameLoading()){
yield return new WaitForSeconds(3);
GameObject mole = RandomMoles();
}
}
那后面解决方法很简单,其实就要记住yield return的性质然后妥善利用(我开始还不太确定,写了注释里面的那个ver1(x 就其实没必要啦
IEnumerator RandomAppearRoutine(){
// buff time
/* buff time ver1 int count = 0; if(count == 0){ yield return new WaitForSeconds(1); count ++; }*/
yield return new WaitForSeconds(1); //buff time ver2
// zen ver
while(GameManager.Instance.IsGameLoading()){
GameObject mole = RandomMoles();
yield return new WaitForSeconds(3);
}
}
解决方式真的很简单粗暴,yield return new WaitForSeconds()
的作用就是让协程挂起一段时间,挂起时间结束以后继续执行下面的。在这边我们就是挂起1s后进入循环语句中,生成地鼠,再挂起3秒,挂完后判断看看是不是还满足循环条件,满足的话就继续生成挂起,不满足就直接进程结束这样子。
然后写到这边,我真的是谢天谢地没有用InvokeRepeating,那玩意哪里有协程方便啦…
文章评论