溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何在Unity中檢測死循環(huán)和卡死

發(fā)布時間:2020-09-03 10:04:37 來源:腳本之家 閱讀:492 作者:Qing''''s Blog 欄目:開發(fā)技術(shù)

當(dāng)游戲在手機(jī)/模擬器上卡死,logcat沒有日志輸出,也沒有卡死堆棧信息或者bugly也沒有捕獲到異常,你是否很焦急?本文介紹一下我們項(xiàng)目中檢測Unity卡死的方法,也許適合你使用。

實(shí)現(xiàn)原理

在絕大多數(shù)情況下我們可以認(rèn)為Unity是單線程的,基于這點(diǎn)我們在Unity的系統(tǒng)函數(shù)FixedUpdate中統(tǒng)計游戲運(yùn)行期間的總幀數(shù),如果Unity沒有卡死,那么TotalFrame是會一直累加的,如果在某一段時間內(nèi)TotalFrame都不會變化了,則可以認(rèn)為Unity已經(jīng)卡死了

既然Unity的主線程已經(jīng)卡死了,我們就需要用另一個線程用來定時檢查unity主線程中的TotalFrame是否不會變化了

示例代碼

using System;
using System.Threading;
using UnityEngine;

namespace KEngine
{
 /// <summary>
 /// 開另外一個線程檢測unity是否被卡死
 /// </summary>
 public static class UnityThreadDetect
 {
  public static Thread _MainThread = System.Threading.Thread.CurrentThread;//獲取unity線程
  private static int check_interval = 3000;//檢測間隔

  public static void Start()
  {
   new Thread(CheckMainThread).Start();
  }
  
  static void CheckMainThread()
  {
   long frame = 0;
   while(!AppEngine.IsApplicationQuit)
   {
    frame = AppEngine.TotalFrame;
    Thread.Sleep(check_interval);
    if (frame == AppEngine.TotalFrame)
    {
     Log.LogToFile("unity thread dead,ThreadState:{0}",_MainThread.ThreadState);
     if (AppEngine.IsApplicationFocus)
     {
      //todo report error
     }
    }
   }
  }
 }
}

捕獲卡死的方法名

在我們的游戲中一般出現(xiàn)卡死的情況都是在定時器里面,我們的定時器是通過在Unity的Update驅(qū)動定時器列表,當(dāng)卡死時,在另一個線程中打印出定時器中正在執(zhí)行的函數(shù)就可以定位到卡死的函數(shù)了。定時器可參考:UnityTimer中的Timer.cs

同時在Unity的Update進(jìn)行派發(fā)多個事件,比如PreUpdate,Update,以便出問題更容易定位到卡在那兒

舉例說明問題

下面舉例我們遇到的出現(xiàn)卡死的問題

死循環(huán)

下面這個死循環(huán)在Unity中會卡死,而在.NET中不會,.NET中當(dāng)i超過byte的最大值255時i會從0開始

public static void TesBadCode()
{
	byte i = 0; 
	while (true)
	{
		i++;
	}
}

目前我們遇到的絕大多數(shù)情況都是邏輯代碼中寫了where(true) do xxx 然后里面某些情況不會break,導(dǎo)致循環(huán)永遠(yuǎn)退不出來

屏蔽了事件系統(tǒng)

在某些系統(tǒng)中屏蔽掉了UGUI的事件系統(tǒng),導(dǎo)致無法接受用戶輸入,這個問題不應(yīng)該歸類為Unity卡死,但用戶反饋來看就是卡死了,無法操作。

重復(fù)添加定時器

起因是底層沒有對同名定時器進(jìn)行限制,在某些邏輯中誤使用,出現(xiàn)每秒添加一個定時器,而定時器中的邏輯很大且長時間不退出的,當(dāng)不斷添加重復(fù)定時器就導(dǎo)致游戲運(yùn)行越來越慢

重復(fù)注冊事件

在一些界面的刷新函數(shù)和控制器函數(shù),被頻繁重復(fù)注冊了事件,導(dǎo)致在拋出事件時,同一個函數(shù)被調(diào)用了N次,這個問題在Unity的Profiler中可以清晰看到函數(shù)的調(diào)用次數(shù)

擴(kuò)展

遞歸調(diào)用

遞歸調(diào)用,會報stack overflow,不會讓unity卡死

為什么無限循環(huán)遞歸調(diào)用不會卡死Unity?

這是因?yàn)槊總€方法的方法調(diào)用棧容量是有限的,當(dāng)超出之后就會跳出報stack overflow,不會讓應(yīng)用程序卡死

public static void TesBadCode()
{
	while (true)
	{
		TesBadCode();
	}
}

總結(jié)

到此這篇關(guān)于在Unity中檢測死循環(huán)和卡死的文章就介紹到這了,更多相關(guān)Unity檢測死循環(huán)和卡死內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI