我正在尝试返回一个函数的值来告诉我某个位置是否在该位置有熔岩或水。
问题是即使它有熔岩或水,它仍然会返回真实。 水,熔岩和苍蝇的值来自玩家生物的列表。如果该生物具有水,熔岩或能够飞行的能力,那就是它。我只是将它们标记为真/假。位置是TPOINT。它将所有这些加载到一个函数中,只要函数CheckLocation返回true然后它的ok位置
if FMyPlayers.Player[i].Values['water']='yes' then
canwater := true;
if FMyPlayers.Player[i].Values['Lava']='yes' then
canlava := true;
if FMyPlayers.Player[i].Values['fly']='yes' then
canfly := true;
cansnow := true;
if checkLocation(position,cansnow,canlava,canwater,canfly) = False then
exit;
function TBaseGameForm.checkLocation(position :Tpoint;Cansnow,canlava,canwater,canfly:bool):Bool;
begin
RESULT := True;
if canfly = true then
RESULT := true
else begin
if FGamePlay.waterLocations.IndexOfName('x'+inttostr(Position.x)+inttostr(Position.Y)) <> -1 then begin //Check location
Showmessage('Cant move there due to water');
RESULT := FALSE;
end;
if FGamePlay.LavaLocations.IndexOfName('x'+inttostr(Position.x)+inttostr(Position.Y)) <> -1 then begin //Check location
Showmessage('Cant move there due to lava');
RESULT := False;
end;
end;
end;
运行时我检查WaterLocations中的值
[0] x47
[1] y47
[2] x58
[3] y58
Position.x和Position.y的值是(4,7)因此它应该返回false原因x47在列表中。
答案 0 :(得分:8)
就像@SertacAkyuz所说,问题是你使用IndexOfName()
。您需要改为使用IndexOf()
。
您的CheckLocation()
功能还有另一个错误。您忽略了canfly
以外的所有输入参数。如果canwater
或canlava
为True,则您的函数将在任何水/熔岩位置返回False。您需要检查位置是否与玩家的能力实际匹配。如果canwater
为真,则无需检查位置是否为水。与熔岩一样。
请改为尝试:
canwater := (FMyPlayers.Player[i].Values['water'] = 'yes');
canlava := (FMyPlayers.Player[i].Values['Lava'] = 'yes');
canfly := (FMyPlayers.Player[i].Values['fly'] = 'yes');
cansnow := true;
if not CheckLocation(position, cansnow, canlava, canwater, canfly) then
Exit;
function TBaseGameForm.CheckLocation(position: TPoint; cansnow, canlava, canwater, canfly: Boolean): Boolean;
var
loc: String;
begin
Result := True;
if (not canfly) and ((not canwater) or (not canlava)) then
begin
loc := Format('x%d%d', [Position.X, Position.Y]);
if (not canwater) and (FGamePlay.waterLocations.IndexOf(loc) <> -1) then
begin
Showmessage('Cant move there due to water');
Result := False;
Exit;
end;
if (not canlava) and (FGamePlay.LavaLocations.IndexOf(loc) <> -1) then
begin
Showmessage('Cant move there due to lava');
Result := False;
Exit;
end;
end;
end;
话虽如此,我同意@sarnold你的坐标系需要一些调整。只要您的x / y坐标都是单个数字,它就可以了。但如果它们是多位数,它将无法工作。至少,您应该在Y坐标前加上前缀,例如:
[0] x4y7
[1] x5y8
loc := Format('x%dy%d', [Position.X, Position.Y]);
就个人而言,我根本不会使用TStrings
来保存这样的整数值。我会使用TList
个TPoint
实例,例如:
waterLocations: TList;
function FindLocation(List: TList; Position: TPoint): Integer;
begin
for Result := 0 to List.Coun-1 do
begin
with PPoint(List[Result])^ do
begin
if (X = Position X) and (Y = Position.Y) then Exit;
end;
end;
Result := -1;
end;
if (not canwater) and (FindLocation(FGamePlay.waterLocations, Position) <> -1) then
begin
Showmessage('Cant move there due to water');
Result := False;
Exit;
end;
或者,如果您使用的是支持泛型的现代Delphi版本,则TList<TPoint>
:
waterLocations: TList<TPoint>;
function FindLocation(List: TList<TPoint>; Position: TPoint): Integer;
begin
for Result := 0 to List.Coun-1 do
begin
with List[Result] do
begin
if (X = Position X) and (Y = Position.Y) then Exit;
end;
end;
Result := -1;
end;
if (not canwater) and (FindLocation(FGamePlay.waterLocations, Position) <> -1) then
begin
Showmessage('Cant move there due to water');
Result := False;
Exit;
end;
甚至一个TDictionary
存储任何给定X / Y坐标指定的位置类型:
type
locType = (locLand, locWater, locLava, locSnow);
locations: TDictionary<TPoint, locType>;
function TBaseGameForm.CheckLocation(position: TPoint; cansnow, canlava, canwater, canfly: Boolean): Boolean;
var
loc: locType;
begin
Result := True;
if not canfly then
begin
locations.TryGetValue(Position, loc);
case loc of
locWater: begin
if not canwater then
begin
Showmessage('Cant move there due to water');
Result := False;
end;
end;
locLava: begin
if not canlava then
begin
Showmessage('Cant move there due to lava');
Result := False;
Exit;
end;
end;
end;
end;
end;