Sprite Kit 游戏在 tvOS 9.1 和 iOS 9.2 上的 Game Over 中崩溃

Posted

技术标签:

【中文标题】Sprite Kit 游戏在 tvOS 9.1 和 iOS 9.2 上的 Game Over 中崩溃【英文标题】:Sprite Kit game crashes on Game Over on tvOS 9.1 and iOS 9.2 【发布时间】:2015-12-20 06:28:12 【问题描述】:

我的 Sprite Kit 游戏在游戏结束时崩溃。这发生在 tvOS 9.1 和 ios 9.2 上。之前,我让它在 iOS 9.1 上运行而没有崩溃。

这似乎是一个 OpenGL 的问题,但是当我在 Xcode 中搜索该函数时,它什么也没找到。

不幸的是,错误在控制台中出现并不一致。这是错误:

Jet: draw_indexed: indexType must be 'unsigned_int' or 'unsigned_short'
Assertion failed: (indexType == jet_component_type_unsigned_int || indexType
== jet_component_type_unsigned_short), function draw_indexed, file
/BuildRoot/Library/Caches/com.apple.xbs/Sources/Jet_Sim/Jet- 
1.50.1/Jet/jet_context_OpenGL.mm, line 1426.

Xcode 在崩溃时也指向 AppDelate 类:

class AppDelegate: UIResponder, UIApplicationDelegate

启用断点异常后,Xcode 会指向这一行。这是一个声音文件的常数:

 let soundHitLava = SKAction.playSoundFileNamed("DrownFireBug.mp3",
waitForCompletion: false)

来自异常断点的更多信息:

SpriteKit`+[SKAction(SKActions) playSoundFileNamed:waitForCompletion:]:
  0x10b95e738 <+0>:  pushq  %rbp
  0x10b95e739 <+1>:  movq   %rsp, %rbp
  0x10b95e73c <+4>:  movq   0x15bf8d(%rip), %rdi      ; (void   *)0x000000010babd020: SKPlaySound
  0x10b95e743 <+11>: movq   0x1618b6(%rip), %rax      ; (void   *)0x000000010b412be8: CGPointZero
  0x10b95e74a <+18>: movsd  (%rax), %xmm0
  0x10b95e74e <+22>: movsd  0x8(%rax), %xmm1
  0x10b95e753 <+27>: movq   0x159a46(%rip), %rsi      ; "playSoundFileNamed:atPosition:waitForCompletion:"
  0x10b95e75a <+34>: movzbl %cl, %ecx
  0x10b95e75d <+37>: callq  *0x1619a5(%rip)           ; (void *)0x000000010d09e800: objc_msgSend
  0x10b95e763 <+43>: movq   %rax, %rdi
  0x10b95e766 <+46>: callq  0x10ba192e8               ; symbol stub for:   objc_retainAutoreleasedReturnValue
  0x10b95e76b <+51>: movq   %rax, %rdi  // **Thread 1: Breakpoint 1.2
  0x10b95e76e <+54>: popq   %rbp
  0x10b95e76f <+55>: jmp    0x10ba1929a               ; symbol stub for:   objc_autoreleaseReturnValue

这是 GameScene 中的 GameOver 函数:

 override func touchesBegan(touches: Set<UITouch>, withEvent event: UIEvent?) 

    #if os(tvOS) // tvOS
    if startTouchX == 0.0 
        startTouchX = (touches.first?.locationInNode(self).x)!
    
    #endif

   switch gameState.currentState 
    case is WaitingForTap:
    gameState.enterState(WaitingForBomb)
    // Switch to playing state
    self.runAction(SKAction.waitForDuration(2.0),
      completion:
      self.gameState.enterState(Playing)
    )

    case is GameOver:
     let newScene = GameScene(fileNamed:"GameScene")
     newScene!.scaleMode = .AspectFill
     let reveal = SKTransition.flipHorizontalWithDuration(0.5)
     self.view?.presentScene(newScene!, transition: reveal)

     self.saveHighScore("com.prismstudios.jumpingcarl.leaderboard", score: GameState.sharedInstance.highScore)

     GameState.sharedInstance.highScore = 0
     GameState.sharedInstance.coins = 0

     default:
       break
     
    

这里是 GameOver 类:

import SpriteKit
import GameplayKit

class GameOver: GKState 
  unowned let scene: GameScene

init(scene: SKScene) 
self.scene = scene as! GameScene
super.init()


override func didEnterWithPreviousState(previousState: GKState?) 
if previousState is Playing 

  scene.playBackgroundMusic("SpaceGame.caf")
  let gameOver = SKSpriteNode(imageNamed: "GameOver")
  gameOver.position = scene.getCameraPosition()
  gameOver.zPosition = 10
  scene.addChild(gameOver)

  let explosion = scene.explosion(6.0)
  explosion.position = gameOver.position
  explosion.zPosition = 11
  scene.addChild(explosion)
  scene.runAction(scene.soundExplosions[3])
  scene.screenShakeByAmt(200)

  scene.addChild(button)
   
 

 override func isValidNextState(stateClass: AnyClass) -> Bool 
   return stateClass is WaitingForTap.Type
 

【问题讨论】:

我们可能需要更多代码。如果您认为这是一个框架错误,请提交雷达:radar.apple.com 谢谢。但我不确定代码在哪里。它在 Game Over 时崩溃,所以我将在我的问题中添加该功能。 您是否启用了异常中断?如果没有,请转到菜单 Debug > Breakpoints > Create Exception Breakpoint... @0x141E 是的,当我启用异常断点时,它会在声音文件常量let soundHitLava = SKAction.playSoundFileNamed("DrownFireBug.mp3", waitForCompletion: false) 上崩溃我不知道为什么它会在那里崩溃,因为文件在捆绑包中。我会将错误添加到我的问题中。 它看起来和this 一样的错误发生在 OP 向场景中添加发射器时。我猜这是一个操作系统错误。 【参考方案1】:

我发现的答案是降低 SKEmitterNode 出生率和发射粒子,因为这会导致游戏崩溃。

在 GameScene.swift 中:

func explosion(intensity: CGFloat) -> SKEmitterNode 
   let emitter = SKEmitterNode()
   let particleTexture = SKTexture(imageNamed: "spark")

   emitter.zPosition = 2
   emitter.particleTexture = particleTexture

   //LOWER the particleBirthRate and numParticlesToEmit
   emitter.particleBirthRate = 50; //4000 * intensity
   emitter.numParticlesToEmit = 50; //Int(400 * intensity)

   ....
 

一旦分别降低到50,那么游戏就不会崩溃。我不知道为什么会发生这种情况,但 Apple 目前正在解决这个问题,并且被认为是一个错误。

【讨论】:

以上是关于Sprite Kit 游戏在 tvOS 9.1 和 iOS 9.2 上的 Game Over 中崩溃的主要内容,如果未能解决你的问题,请参考以下文章

使用 Sprite Kit 和 Swift 在游戏中显示当前分数

游戏中的三角学——Sprite Kit 和 Swift 教程

游戏中的三角学——Sprite Kit 和 Swift 教程

如何在我的 Sprite Kit 游戏应用中实现 Facebook 和 Twitter 发布?

用Swift语言和Sprite Kit复制微信飞机大战游戏

在我的 Sprite-Kit 游戏中实现这个动作的正确方法是啥?