- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我一直在研究让圆圈相互碰撞,这是我到目前为止想出的:
foreach(Circle circle in circles)
foreach(Circle circle2 in circles)
{
bool HasCollided = false;
if(circle.ID != circle2.ID)
{
double squareRoot =(Math.Pow(((int)circle.position.X - (int)circle2.position.X), 2) + Math.Pow((int)circle.position.Y - (int)circle2.position.Y, 2));
double distanceBetweenCentres = Math.Sqrt(squareRoot);
if(distanceBetweenCentres < (circle.radius + circle2.radius))HasCollided = true;
else if(distanceBetweenCentres> (circle.radius + circle2.radius)) HasCollided = false;
if(HasCollided == true)
{
double newVelX1 = ((circle.velocity.X * (circle.Mass - circle2.Mass) + (2 * circle2.Mass * circle2.velocity.X)) / (circle.Mass + circle2.Mass));
double newVelY1 = ((circle.velocity.Y * (circle.Mass - circle2.Mass) + (2 * circle2.Mass * circle2.velocity.Y)) / (circle.Mass + circle2.Mass));
double newVelX2 = ((circle2.velocity.X * (circle2.Mass - circle.Mass) + (2 * circle.Mass * circle.velocity.Y)) / (circle2.Mass + circle.Mass));
double newVelY2 = ((circle2.velocity.Y * (circle2.Mass - circle.Mass) + (2 * circle.Mass * circle.velocity.Y)) / (circle2.Mass + circle.Mass));
circle.velocity = new Vector2((int)newVelX1, (int)newVelY1);
circle2.velocity = new Vector2((int)newVelX2, (int)newVelY2);
}
}
}
}
代码在圆圈碰撞时确实有效,但是最终速度的方程式似乎使圆圈要么在屏幕上跳跃,要么慢慢地相互掉落,我尝试在网上找到解决方案,但我找不到它工作。关于如何修复它有什么建议吗?
新的代码和 GIF 来展示它的作用:
if(circle.ID < circle2.ID)
{
double squareRoot =(Math.Pow(((int)circle.position.X - (int)circle2.position.X), 2) + Math.Pow((int)circle.position.Y - (int)circle2.position.Y, 2));
double distanceBetweenCentres = Math.Sqrt(squareRoot);
if(distanceBetweenCentres < (circle.radius + circle2.radius))HasCollided = true;
else if(distanceBetweenCentres> (circle.radius + circle2.radius)) HasCollided = false;
if(HasCollided == true)
{
double newVelX1 = ((circle.velocity.X * (circle.Mass - circle2.Mass) + (2 * circle2.Mass * circle2.velocity.X)) / (circle.Mass + circle2.Mass));
double newVelY1 = ((circle.velocity.Y * (circle.Mass - circle2.Mass) + (2 * circle2.Mass * circle2.velocity.Y)) / (circle.Mass + circle2.Mass));
double newVelX2 = ((circle2.velocity.X * (circle2.Mass - circle.Mass) + (2 * circle.Mass * circle.velocity.Y)) / (circle2.Mass + circle.Mass));
double newVelY2 = ((circle2.velocity.Y * (circle2.Mass - circle.Mass) + (2 * circle.Mass * circle.velocity.Y)) / (circle2.Mass + circle.Mass));
circle.velocity = new Vector2((float)newVelX1, (float)newVelY1);
circle2.velocity = new Vector2((float)newVelX2, (float)newVelY2);
}
}
最佳答案
如果他们跳来跳去,那很可能是因为在您进行下一次碰撞检查时他们还没有分开。尝试将圆从撞击点移动半径。
以第一个圆为例:
Cp = Cp + Normalize(Ip - Cp) * Cr
其中 Ip 是冲击位置,Cp 是圆位置,Cr 是圆半径。如果它似乎将圆进一步推向碰撞,则在归一化之前交换 Ip 和 Cp。
编辑:自己运行代码后想出这个:
问题 1:在 newVelX2 中,您正在比较 CircleVelocity.Y 而不是 .X
问题 2:您没有检查以前评估过的对。我用下面的代码解决了这个问题:`
for (int c1Index = 0; c1Index < RigidBodies.Count; c1Index++)
{
for (int c2Index = c1Index + 1; c2Index < RigidBodies.Count; c2Index++)
{
HandleCollisions(RigidBodies[c1Index], RigidBodies[c2Index]);
}
}
这可以防止您对一对进行两次评估。我的意思是,如果按照您的代码,列表中有 3 个圆圈:
Outer Loop = Circle0
Inner Loop = Circle0 <-- C0 is skipped
Inner Loop = Circle1 <-- C0 and C1 pair evaluated
Inner Loop = Circle2 <-- C0 and C2 pair evaluated
Outer Loop = Circle1
Inner Loop = Circle0 <-- C1 and C0 pair evaluated *PROBLEM: We already did this pair*
Inner Loop = Circle1 <-- C1 is skipped
Inner Loop = Circle2 <-- C1 and C2 pair evaluated
Outer Loop = Circle2
Inner Loop = Circle0 <-- C2 and C0 pair evaluated *PROBLEM: We already did this pair*
Inner Loop = Circle1 <-- C2 and C1 pair evaluated *PROBLEM: We already did this pair*
Inner Loop = Circle2 <-- C2 is skipped
`
我的代码通过让内部循环从当前评估的圆圈之后的一个开始来解决这个问题,因为在它之前的任何事情都已经被处理过。另一种选择是维护已评估对的列表,这显然会占用更多内存,但可以完成工作。
对于我的演示,我不必先将圆圈分开。但是,如果您在彼此内部生成圆圈,它们就会像那样卡住。因此,如果您预计圈子有时会粘在一起,您可能希望保留我原始答案中的伪代码。
此外,下面的代码显示了我改编自 this link 的更准确的响应机制。 :
private void HandleCollisions(Circle Circle1, Circle Circle2)
{
if (!HaveCollided(Circle1, Circle2))
return;
Vector2D distance = Circle1.Position - Circle2.Position;
Vector2D normalVector = distance.Normalize();
Vector2D tangentVector = normalVector.GetTangentVector();
float c1VelocityAlongNormal = VectorMath.CalculateDotProduct(normalVector, Circle1.Velocity);
float c2VelocityAlongNormal = VectorMath.CalculateDotProduct(normalVector, Circle2.Velocity);
float c1VelocityAlongTan = VectorMath.CalculateDotProduct(tangentVector, Circle1.Velocity);
float c2VelocityAlongTan = VectorMath.CalculateDotProduct(tangentVector, Circle2.Velocity);
float massSum = Circle1.Mass + Circle2.Mass;
float c1Speed = (c1VelocityAlongNormal * (Circle1.Mass - Circle2.Mass) + (2.0f * Circle2.Mass * c2VelocityAlongNormal) / massSum);
float c2Speed = (c2VelocityAlongNormal * (Circle1.Mass - Circle2.Mass) + (2.0f * Circle1.Mass * c1VelocityAlongNormal) / massSum);
Vector2D c1FinalVelocity = normalVector * c1Speed;
Vector2D c2FinalVelocity = normalVector * c2Speed;
Vector2D c1Tangent = tangentVector * c1VelocityAlongTan;
Vector2D c2Tangent = tangentVector * c2VelocityAlongTan;
Circle1.Velocity = c1FinalVelocity + c1Tangent;
Circle2.Velocity = c2FinalVelocity + c2Tangent;
}
**FROM VECTOR2D CLASS**
class Vector2D
{
public float X, Y;
public Vector2D()
{
this.X = 0.0f;
this.Y = 0.0f;
}
public Vector2D(float X, float Y)
{
this.X = X;
this.Y = Y;
}
public Vector2D(Vector2D vectorToCopy)
{
this.X = vectorToCopy.X;
this.Y = vectorToCopy.Y;
}
public float CalculateMagnitudeSquared()
{
return (this.X * this.X) + (this.Y * this.Y);
}
public float CalculateMagnitude()
{
return (float)Math.Sqrt((this.X * this.X) + (this.Y * this.Y));
}
public Vector2D GetTangentVector()
{
return new Vector2D(-this.Y, this.X).Normalize();
}
public Vector2D Normalize()
{
return new Vector2D(this / CalculateMagnitude());
}
public static Vector2D operator+(Vector2D thisVector, Vector2D otherVector)
{
return new Vector2D(thisVector.X + otherVector.X, thisVector.Y + otherVector.Y);
}
public static Vector2D operator-(Vector2D thisVector, Vector2D otherVector)
{
return new Vector2D(thisVector.X - otherVector.X, thisVector.Y - otherVector.Y);
}
public static Vector2D operator*(Vector2D thisVector, Vector2D otherVector)
{
return new Vector2D(thisVector.X * otherVector.X, thisVector.Y * otherVector.Y);
}
public static Vector2D operator*(Vector2D thisVector, float scalar)
{
return new Vector2D(thisVector.X * scalar, thisVector.Y * scalar);
}
public static Vector2D operator/(Vector2D thisVector, Vector2D otherVector)
{
return new Vector2D(thisVector.X / otherVector.X, thisVector.Y / otherVector.Y);
}
public static Vector2D operator/(Vector2D thisVector, float scalar)
{
return new Vector2D(thisVector.X / scalar, thisVector.Y / scalar);
}
}
这不是最简洁的代码...只是将它放在一起来测试这些东西。
还有什么问题可以私信我
关于c# - XNA-如何计算二维圆相互碰撞后的速度?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36023170/
我可以将内容项目引用到非 XNA 项目(MonoGame,如果它很重要)? 最佳答案 不直接。 将内容项目引用添加到项目的功能仅适用于“XNA 游戏”和“XNA 游戏库”类型的项目。 当我想在 Win
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 4年前关闭。 Improve thi
您如何看待 xna 和大型多人游戏?例如服务器和 100 个客户端?有没有很好的教程如何在 xna 中创建客户端服务器应用程序? 最佳答案 当然有关于如何做到这一点的教程。我个人使用 Lidgren
好的,所以我正在开发这个游戏,我是 XNA 的新手(我使用的是 4.0),我想做的是每次产生敌人时都有不同的纹理。 所以我有这些图像“image1.png”、“image2.png”等等。我希望它在每
我最近开始检查XNA。一切进展顺利,我什至开始了一个工作样本。困扰我的一件事是,每当我将鼠标放在游戏窗口上时,它就会消失。我不知道这是否是默认行为,但我想更改此设置以便可以看到鼠标。任何和所有建议,不
我刚从 XNA 开始,有一个关于轮换的问题。当你在 XNA 中将一个向量乘以一个旋转矩阵时,它会逆时针旋转。这个我明白。 但是,让我举一个我不明白的例子。假设我将一个随机艺术 Assets 加载到管道
我正在使用 Visual Studio 2010 Ultimate,在谷歌上搜索 XNA 下载后,我找到了这个页面: http://blogs.msdn.com/b/xna/archive/2010/
我有一个字典来存储SoundEffects,例如: public static Dictionary Hangok = new Dictionary(); 我从文件(正常.wav格式)加载声音,例如:
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 10年前关闭。 Improve this
好吧,我正在尝试为 3D XNA 游戏创建自己的物理引擎,但我在计算我应该将物体移动多少时遇到重力问题。 XNA 的游戏计时器每 16 毫秒出现一次,所以经过一些计算,并使用 9.81m/s 作为我的
我正在尝试对以下粒子系统进行修改。 http://create.msdn.com/en-US/education/catalog/sample/particle_3d 我有一个功能,当我按下Space
As it currently stands, this question is not a good fit for our Q&A format. We expect answers to be
也许这更像是一个数学问题而不是一个编程问题,但我一直在尝试在 XNA 中实现旋转卡尺算法。 我已经使用维基百科上详述的单调链从我的点集中推导出了一个凸包。 现在,我正在尝试对我的算法进行建模,以在此处
我正在 XNA 中制作一个简单的 2 人游戏,并开始研究保存玩家的高分。 我希望游戏能在 XBox 360 和 Windows 上运行,所以我必须使用框架来保存数据。 您似乎将数据保存到特定用户的玩家
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我需要使用不同字体大小的 spritefont,我必须为每个大小创建新的 spritefont 吗? 最佳答案 基本上是的。 有一个overload的 SpriteBatch.DrawString这为
我对在我的游戏中存储 *.txt 级别文件的位置和方式感到非常困惑。 我想要在我的游戏中提供一些随游戏一起安装的“预烘焙”关卡的可用性,以及用户创建自己的关卡的能力。 所以我所做的是在 Visual
我有一个足够简单的着色器,它支持多个点光源。 灯光存储为一组灯光结构(最大大小),当事件灯光发生变化时,我会传入事件灯光的数量。 问题出在 PixelShader 函数中: 这是基本的东西,从纹理中获
我有 19 张图像,它们是我的播放器的动画帧 在下面,我创建了 Frog 纹理数组,这是我的播放器。并且有 19 个图像。如何对它们进行动画处理。 公共(public)类纹理 { 公共(public)
所以,我要直截了本地说,我正在尝试在我正在制作的 XNA 游戏中制作一个聊天系统。我可以很容易地找出正在按下的键但是我无法知道实际键入的是什么。在我看来,他们好像在说:OemShifthello bi
我是一名优秀的程序员,十分优秀!