2

在这个简单的游戏中,有一个职业战士,其目的是让两个战士战斗。将生命值降至 0 以下的人将输掉比赛。

为了战斗,有一个静态方法战斗(..),它迭代直到一个战士赢得比赛,由另一个非静态方法攻击(..)支持

object Fighter 的健康应该随着两个对象在游戏中使用 Fight(...) 和 Attack(...) 方法进行战斗而改变。问题是它总是打印相同的战斗机健康状况,并且游戏永远不会结束。我看不出问题出在哪里

class ViewController: UIViewController {
     override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.

        let david = Fighter(name: "David", health: 100, damage: 30, defense: 10, initiative: 80)
        let goliath = Fighter(name: "Goliath", health: 300, damage: 60, defense: 14, initiative: 90)


        let myFight1 = Fighter.fight(fighter1: david, fighter2: goliath) // always executing same Fighters
        print(myFight1)

    }
}

import Foundation


struct Fighter {
    var name: String
    var health: Double
    var damage: Int
    var defense: Int
    var initiative: Int

    init (name: String, health: Double, damage: Int, defense: Int, initiative: Int) {
        self.name = name
        self.health = health
        self.damage = damage
        self.defense = defense
        self.initiative = initiative
    }

     init (name: String, health: Double, damage: Int, defense: Int) {
        self.name = name
        self.health = health
        self.damage = damage
        self.defense = defense
        self.initiative = 0
    }

    static func fight(fighter1: Fighter, fighter2: Fighter) -> Fighter {
        let f1 = fighter1
        let f2 = fighter2

        if f1.health == f2.health {
            return f1
        }

        if f2.initiative > f1.initiative {
           f2.attack(f: f1)
        }

        var i = 0

        while f1.health > 0 {
            i += 1
            print("--> i: \(i)")
            f1.attack(f: f2 )

            if f2.health <= 0 {
                return f1
            }
        f2.attack(f: f1)
            }
        return f2
        }

    func attack(f: Fighter) -> Void {
        var g = f
        g.health = g.health - Double(g.damage * (1 - g.defense / 100))
        print(g)
    }        
}
4

4 回答 4

3

您正在使用structfor Fighterwhich 是 Swift 中的类型。

值类型最基本的区别特征是复制——赋值、初始化和参数传递的效果——创建一个独立的实例,它有自己唯一的数据副本

解决方案:更改Fighter为 aclass就可以了。

打印语句的输出:(第二个打印语句更改为print(g.name, g.health)

大卫 70.0
--> i: 1
歌利亚 240.0
大卫 40.0
--> i: 2
歌利亚 180.0
大卫 10.0
--> i: 3
歌利亚 120.0
大卫 -20.0


更多阅读:值和引用类型

于 2018-10-03T12:47:07.267 回答
1

func attack(f: Fighter) -> Void每次调用该方法后,Fighter被攻击者的属性都没有更新。所以while循环不会在任何时候中断。

请替换下面的代码。

static func fight(fighter1: Fighter, fighter2: Fighter) -> Fighter {
        var f1 = fighter1
        var f2 = fighter2

        if f1.health == f2.health {
            return f1
        }

        if f2.initiative > f1.initiative {
            f1 = f2.attack(f: f1)
        }

        var i = 0

        while f1.health > 0 {
            i += 1
            print("--> i: \(i)")
            f2 = f1.attack(f: f2 )

            if f2.health <= 0 {
                return f1
            }
            f1 = f2.attack(f: f1)
        }
        return f2
    }

    func attack( f: Fighter) -> Fighter {
        var g = f
        g.health = g.health - Double(g.damage * (1 - g.defense / 100))
        print(g)
        return g
    }
于 2018-10-03T13:57:12.487 回答
1

当你说...

    var g = f

...您实际上是在创建该对象的副本,而不是引用。因此,当您更改“健康”属性时,您将在副本中更改它。有2个简单的解决方案:

1)将结构更改为类,因为类被引用,不像结构,它只是复制。

2)用修改后的副本替换原始对象(g)

于 2018-10-03T14:26:29.257 回答
0

正如 Rakesha 所说,结构是值类型,因此您的attack代码实际上并没有修改任何内容:

func attack(f: Fighter) -> Void {
    var g = f   // Make a mutable copy of `f` called `g`
    g.health = g.health - Double(g.damage * (1 - g.defense / 100)) // Modify g
    print(g) // Print g
    // Throw g away
}

(旁注:我认为g.damage这里不正确;我想您可能是说self.damage.)

那里没有任何实际修改f。有几种方法可以解决这个问题。一种是使用类,它引入了微妙的可变状态。我不认为我会在这里这样做。“微妙的可变状态”是指您希望attack修改f,但签名中没有任何内容表明它会这样做,因此调用者可能会感到惊讶。

相反,您有几种方法可以实现这一点,使您的突变在结构上显式。您可以attack显式修改f

func attack(f: inout Fighter) {
    f.health = f.health - Double(damage * (1 - f.defense / 100))
}

或者你可以在被别人攻击时转过身来修改自己:

mutating func attackedBy(f: Fighter) {
    health = health - Double(f.damage * (1 - defense / 100)
}
于 2018-10-03T14:38:18.270 回答