我在执行游戏时随机弹出以下错误:
我正在随机的时间创建一个carNode的多个实例,并且所有这些实例都具有从场景中删除该节点的 Action 。在大多数情况下,这不是问题,但是在某些情况下,应用程序因照片中显示的错误而崩溃。在这些情况下,我似乎可以得到错误:
对象未初始化
对象已被释放
不太可能发生的其他事情
..除了第一个对象外,我非常确定自己不会提前释放该对象..那会是什么呢?
这是完整的代码:
func spawnCarAtPosition(position: SCNVector3) {
// Create a material using the model_texture.tga image
let carMaterial = SCNMaterial()
carMaterial.diffuse.contents = UIImage(named: "assets.scnassets/Textures/model_texture.tga")
carMaterial.locksAmbientWithDiffuse = false
// Create a clone of the Car node of the carScene - you need a clone because you need to add many cars
var carNode: SCNNode!
let randomNumb = AppDelegate().randRange(0, upper: beachCarArray.count - 1)
let selectedObject = beachCarArray[randomNumb]
carNode = selectedObject.objectscene.rootNode.childNodeWithName(selectedObject.objectname, recursively: false)!.clone() as SCNNode
carNode.name = selectedObject.objectname
carNode.position = position
// Set the material
carNode.geometry!.firstMaterial = carMaterial
// x = length, y = height, z = width
let xscale = 0.6
let yscale = 0.8
let zscale = 0.5
carNode.scale = SCNVector3(xscale, yscale, zscale)
// Create a physicsbody for collision detection
let boundingBox = AppDelegate().sizeOfBoundingBoxFromNode(carNode)
let carPhysicsBodyShape = SCNPhysicsShape(geometry: SCNBox(width: CGFloat(boundingBox.width * Float(xscale)), height: CGFloat(boundingBox.height * Float(yscale)), length: CGFloat(boundingBox.depth * Float(zscale)), chamferRadius: 0.0), options: nil)
carNode.physicsBody = SCNPhysicsBody(type: SCNPhysicsBodyType.Kinematic, shape: carPhysicsBodyShape)
carNode.physicsBody!.categoryBitMask = PhysicsCategory.Car
carNode.physicsBody!.collisionBitMask = PhysicsCategory.Player
carNode.physicsBody?.contactTestBitMask = PhysicsCategory.Player
// Move the car
let moveDirection: Float = position.x > 0.0 ? -1.0 : 1.0
let moveDistance = levelData.gameLevelWidth()
let moveAction = SCNAction.moveBy(SCNVector3(x: moveDistance * moveDirection, y: 0.0, z: 0.0), duration: Double(AppDelegate().randomBetweenNumbers(6.0, secondNum: 7.0)))
let removeAction = SCNAction.runBlock { node -> Void in
node.removeFromParentNode()
}
carNode.runAction(SCNAction.sequence([moveAction, removeAction]))
// Rotate the car to move it in the right direction
if moveDirection > 0.0 {
carNode.rotation = SCNVector4(x: 0.0, y: 1.0, z: 0.0, w: 3.1415)
}
// add node to screen
rootNode.addChildNode(carNode)
}
编辑:
崩溃前对象的打印:
<SCNNode: 0x1374e5ba0 'BeachCar_2' pos(1.500000 0.035000 -4.800000) rot(0.000000 1.000000 0.000000 3.141500) scale(0.600000 0.800000 0.500000) | geometry=<SCNGeometry: 0x135fe80c0 'BeachCar_2'> | no child>
使用SCNAnimation删除操作:
编辑2:
func createSpawnNode(gridPosition: SCNVector3, row: Int, parentNode: SCNNode) {
// Determine if the car should start from the left of the right
let startCol = row % 2 == 0 ? 0 : data.columnCount() - 1
let moveDirection : Float = row % 2 == 0 ? 1.0 : -1.0
// Determine the position of the node
var position = coordinatesForGridPosition(column: startCol, row: row)
position = SCNVector3(x: position.x, y: GameVariables.gamePlaneHeight / 2, z: position.z)
// Create node
let spawnNode = SCNNode()
spawnNode.position = position
var spawnAction: SCNAction!
var delayAction: SCNAction!
// Create an action to make the node spawn cars
spawnAction = SCNAction.runBlock({ node in
self.spawnDelegate!.spawnCarAtPosition(node.position)
})
delayAction = SCNAction.waitForDuration(3.0, withRange: 2.0)
spawnNode.runAction(SCNAction.repeatActionForever(SCNAction.sequence([spawnAction, delayAction])))
parentNode.addChildNode(spawnNode)
// record newly created node
recordNodeAtRow(spawnNode, row: row)
}
parentNode是场景的rootNode。
最佳答案
推测...也许内部存在对carNode
的引用,该引用持续了您的removeFromParentNode()
调用?似乎有一个保留周期:carNode
拥有对removeAction
的强引用,其闭包拥有对carNode
的(强?)引用。
不过,我认为removeFromParentNode
上存在SCNAction
是一个很大的提示。如果不需要它就不会在那里。代替基于闭包的调用,尝试
let removeAction = SCNAction.removeFromParentNode()
carNode.runAction(SCNAction.sequence([moveAction, removeAction]))
只是要确保:您是从主线程而不是后台线程中调用
spawnCarAtPosition()
,对吗?在您的编辑2之后进行编辑:
createSpawnNode()
读起来有点令人困惑,因为您重复使用了position
(两个let
定义而不是一个var
呢?)(另外两个var
defs会比let
,btw更直接)。我看不到任何会导致崩溃的信息。我不会复制源Car的
objectName
,而是给每个节点一个唯一的名称(也许是NSDate().description
),以查看是否可以检测到模式。我还将尝试将生成的内容从SCNAction中移出,并转入渲染回调(可能是
renderer(_:updateAtTime:)
,以防您在内部SceneKit错误中打钩。您正在另一个中调用一个SCNAction
。文档中没有说明是否允许这样做,但是由于您看到的行为不稳定,因此简化起来似乎是一件好事。我假设您已经尝试了打开僵尸并设置异常断点的标准技巧。
关于scenekit - EXC_BAD_ACCESS位于removeFromParentNode()SceneKit,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/35529750/