所以我在列表中有对象,我试图为每个人分配一个角度。目前我有两个名单:
List<double> BulletAngles = new List<double>();
List<OvalShape> Bullets = new List<OvalShape>();
我对列表如何工作的看法显然非常不合适。现在,子弹正在根据以下代码旅行:
foreach (OvalShape b in Bullets)
{
int i = Bullets.IndexOf(b);
if (b.Location.X > -10 && b.Location.X < (this.Width - 6) + 10
&& b.Location.Y > -10 && b.Location.Y < (this.Height - 30) + 10)
{
b.Left += Convert.ToInt32(bulletSpeed * Math.Cos(BulletAngles[i]));
b.Top += Convert.ToInt32(bulletSpeed * Math.Sin(BulletAngles[i]));
}
}
角度全部基于鼠标点击时的位置,但在测试时,所有项目符号都会朝第一次鼠标点击的方向移动。 这里也是设置角度和放置在子弹中的代码:
Point target = new Point(mousePos.X - origin.X, mousePos.Y - origin.Y);
OvalShape g = Bullet;
double y = Math.Atan2(target.Y, target.X);
g.Left = origin.X - (g.Width / 2);
g.Top = origin.Y - (g.Height / 2);
g.Visible = true;
BulletAngles.Add(y);
Bullets.Add(g);
答案 0 :(得分:0)
使用IndexOf方法可能不会返回您要查找的结果,因为此方法使用默认的相等比较器确定相等性。
虽然在两个单独的列表中具有与对象相关的属性是有问题的,并且可以通过将属性封装到单个对象中来更好地解决,如前所述。如果您在迭代列表时尝试获取对象的索引,则最好使用for语句而不是foreach语句。
for (int index = 0; index < Bullets.Count; index++)
{
var b = Bullets[index];
if (b.Location.X > -10 && b.Location.X < (this.Width - 6) + 10
&& b.Location.Y > -10 && b.Location.Y < (this.Height - 30) + 10)
{
b.Left += Convert.ToInt32(bulletSpeed*Math.Cos(BulletAngles[index]));
b.Top += Convert.ToInt32(bulletSpeed * Math.Sin(BulletAngles[index]));
}
}
以下是使用封装的示例:
public class OvalShapeBullet
{
private readonly OvalShape _ovalShape;
private Point _location;
private readonly double _angle;
private double _distance;
public OvalShapeBullet(Point orgin, Point target, OvalShape ovalShape)
{
_ovalShape = ovalShape;
_location = new Point(orgin.X - ((int)ovalShape.Width / 2), orgin.Y - ((int)ovalShape.Height / 2));
_distance = Math.Sqrt((orgin.X - target.X) ^ 2 + (orgin.Y - target.Y) ^ 2);
_angle = Math.Atan2(target.Y, target.X);
}
public Point Location { get { return _location; } }
public bool Visible
{
get { return _ovalShape.Visible; }
set { _ovalShape.Visible = value; }
}
/// <summary>
/// Attempts to move bullet to target
/// </summary>
/// <param name="distance">The distance the bullt will travel</param>
/// <returns>True if bullet connects with target</returns>
public bool TryMoveToTarget(float distance)
{
if (_distance <= 0)
return true; // check if bullet has travelled far enough
_location.X += (int)(distance * Math.Cos(_angle));
_location.Y += (int)(distance * Math.Sin(_angle));
_distance -= distance;
return _distance <= 0; // check if bullet has traveled far enough
}
}
随着原始语句成为单个列表声明
,您的代码将变得更加清晰List<OvalShapeBullet> Bullets = new List<OvalShapeBullet>();
创建新对象的处理
Point target = new Point(mousePos.X - origin.X, mousePos.Y - origin.Y);
OvalShapeBullet g = new OvalShapeBullet(mousePos, target, Bullet);
g.Visible = true;
Bullets.Add(g);
管理子弹的移动并检查目标连接
foreach (OvalShapeBullet b in Bullets)
{
if (b.TryMoveToTarget(bulletSpeed))
{
// handle bullet has hit the target
Bullets.Remove(b);
}
}