我已经搜索 SO 和 Unity Answers 数周了,但无法解决这个问题。我在 Unity 中有一个 3d 对象,它是根据用户当前的纬度/经度(比如 41.23423,-122.87978)实例化的,然后我想在对象的位置发生变化时将该对象移动到新的纬度/经度。
所以 41.23423, -122,87978 在我的 3D 环境中变为 0,0,0 并且我应用相对于原始位置的更改。
我在 C# 中编码,当位置更改触发 Action 时,我比较两个纬度/经度并计算距离和角度,然后进行极坐标到笛卡尔的转换并将对象移动到新位置。
我的问题是更新正在移动对象,但产生了一些不稳定的结果,我猜这是我的代码中的一个问题。例如,如果我在两个位置之间切换,第一个更改将移至 (131.6, 0.0, 0.0) 但随后将其切换回移动不会移动对象。或者我也曾在测试场合看到,第一次更改会产生意外坐标,但任何后续更改(在两点之间切换)都会为每个位置产生相同的相对值。
我不能很好地处理我在这里做错了什么,所以任何帮助将不胜感激,如果这个问题太模糊或重复的问题(我的代码中的问题或者可能我没有正确处理问题)
这是我的类的代码,它根据位置移动对象。
using UnityEngine;
using System.Collections;
using System;
public class CameraMover : MonoBehaviour {
const double PIx = 3.141592653589793;
const double RADIO = 20925721.8;
public CoreLocationData lastLocationData;
public int firstUpdate = 0;
private Vector3 currentVector;
private Vector3 newVector;
// Use this for initialization
void OnEnable () {
CoreLocationManager.locationServicesDidUpdate += locationServicesDidUpdate;
}
// Update is called once per frame
void Update () {
}
public void locationServicesDidUpdate( CoreLocationData locationData )
{
if (firstUpdate == 0)
lastLocationData = locationData;
double newDistance = DistanceBetweenPlaces(lastLocationData.longitude, lastLocationData.latitude, locationData.longitude, locationData.latitude);
double angleToMove = Angle(lastLocationData.latitude, lastLocationData.longitude, locationData.latitude, locationData.longitude);
double newX = (newDistance * Math.Cos(angleToMove));
double newZ = (newDistance * Math.Sin(angleToMove));
float newXfloat = System.Convert.ToSingle(newX);
float newZfloat = System.Convert.ToSingle(newZ);
currentVector = this.gameObject.transform.position;
newVector = new Vector3(newXfloat, 0.0F, newZfloat);
this.gameObject.transform.position = Vector3.Lerp(currentVector, newVector, 1);
Debug.Log(string.Format("Distance To Move is {0}, angle is {1}", newDistance, angleToMove));
Debug.Log(string.Format("Moving from {0} to {1}", currentVector, newVector));
lastLocationData = locationData;
firstUpdate = 1;
}
public static double Radians(double x)
{
return x * PIx / 180;
}
public static double DistanceBetweenPlaces(
double lon1,
double lat1,
double lon2,
double lat2)
{
double dlon = Radians(lon2 - lon1);
double dlat = Radians(lat2 - lat1);
double a = (Math.Sin(dlat / 2) * Math.Sin(dlat / 2)) + Math.Cos(Radians(lat1)) * Math.Cos(Radians(lat2)) * (Math.Sin(dlon / 2) * Math.Sin(dlon / 2));
double angle = 2 * Math.Atan2(Math.Sqrt(a), Math.Sqrt(1 - a));
return angle * RADIO;
}
public double Angle(double px1, double py1, double px2, double py2)
{
// Negate X and Y values
double pxRes = px2 - px1;
double pyRes = py2 - py1;
double angle = 0.0;
// Calculate the angle
if (pxRes == 0.0)
{
if (pxRes == 0.0)
angle = 0.0;
else if (pyRes > 0.0) angle = System.Math.PI / 2.0;
else
angle = System.Math.PI * 3.0 / 2.0;
}
else if (pyRes == 0.0)
{
if (pxRes > 0.0)
angle = 0.0;
else
angle = System.Math.PI;
}
else
{
if (pxRes < 0.0)
angle = System.Math.Atan(pyRes / pxRes) + System.Math.PI;
else if (pyRes < 0.0) angle = System.Math.Atan(pyRes / pxRes) + (2 * System.Math.PI);
else
angle = System.Math.Atan(pyRes / pxRes);
}
// Convert to degrees
angle = angle * 180 / System.Math.PI;return angle;
}
}
编辑
正确实现:
using UnityEngine;
using System.Collections;
using System;
public class CameraMover : MonoBehaviour
{
const double RATIO = 20902231.0029;
public CoreLocationData lastLocationData;
public int firstUpdate = 0;
private Vector3 currentVector;
private Vector3 newVector;
// Use this for initialization
void OnEnable ()
{
CoreLocationManager.locationServicesDidUpdate += locationServicesDidUpdate;
}
// Update is called once per frame
void Update () { }
public void locationServicesDidUpdate( CoreLocationData locationData )
{
if (firstUpdate == 0)
{
lastLocationData = locationData;
}
double newDistance = DistanceBetweenPlaces(lastLocationData.longitude, lastLocationData.latitude, locationData.longitude, locationData.latitude);
double angleToMove = AngleBetweenPlaces(lastLocationData.latitude, lastLocationData.longitude, locationData.latitude, locationData.longitude);
double angleToMoveRadians = Deg2Rad(angleToMove);
double newX = (newDistance * Math.Sin(angleToMoveRadians));
double newZ = (newDistance * Math.Cos(angleToMoveRadians));
float newXfloat = System.Convert.ToSingle(newX) * -1;
float newZfloat = System.Convert.ToSingle(newZ) * -1;
Debug.Log(string.Format("new x coordinate should be {0} and z should be {1}", newXfloat, newZfloat));
currentVector = this.gameObject.transform.position;
this.gameObject.transform.position = new Vector3((currentVector.x + newXfloat),0, (currentVector.z + newZfloat));
lastLocationData = locationData;
firstUpdate = 1;
}
public static double DistanceBetweenPlaces(
double lon1,
double lat1,
double lon2,
double lat2)
{
double phi1 = Deg2Rad(lat1);
double phi2 = Deg2Rad(lat2);
double deltaLamdba = Deg2Rad(lon2 - lon1);
double d = Math.Acos(
(Math.Sin(phi1) * Math.Sin(phi2)) +
(Math.Cos(phi1) * Math.Cos(phi2) * Math.Cos(deltaLamdba)));
return d * RATIO;
}
public static double Deg2Rad(double degrees)
{
return degrees * (Math.PI / 180.0);
}
public static double Rad2Deg(double radians)
{
return radians * (180.0 / Math.PI);
}
public double AngleBetweenPlaces(double lat1, double lon1, double lat2, double lon2)
{
var newLat1 = Deg2Rad(lat1);
var newLat2 = Deg2Rad(lat2);
var dLon = Deg2Rad(lon2) - Deg2Rad(lon1);
var y = Math.Sin(dLon) * Math.Cos(newLat2);
var x = Math.Cos(newLat1) * Math.Sin(newLat2) - Math.Sin(newLat1) * Math.Cos(newLat2) * Math.Cos(dLon);
var brng = Math.Atan2(y, x);
return (Rad2Deg(brng) + 360) % 360;
}
}
最佳答案
有时您使用 System.Math.PI,有时您使用 PIx = 3.141592653589793 - 这是什么原因?另外,RADIO 是什么,根据它的使用,我认为它是地球的平均半径(以公里为单位),但该值已经过时了。
DistanceBetweenPlaces 方法也有一些错误,无论如何,请尝试以下操作,如果您对准确性有疑问,那么我会使用 Vincenty's formulae ,否则坚持使用简单的余弦球面定律。
// Earths Mean Radius in Kilometres?
public const double RADIO = 6371;
public static double DistanceBetweenPlaces(
double lon1,
double lat1,
double lon2,
double lat2)
{
double phi1 = Deg2Rad(lat1);
double phi2 = Deg2Rad(lat2);
double deltaLamdba = ConvertDegreesToRadians(lon2 - lon1);
double d = Math.Acos((Math.Sin(phi1) * Math.Sin(phi2)) + (Math.Cos(phi1) * Math.Cos(phi2) * Math.Cos(deltaLamdba)));
return d * RADIO;
}
public static double Deg2Rad(double degrees)
{
return degrees == 0 ? degrees : (degrees * Math.PI / 180.0);
}
另外,查看这个用于处理测地线数据的重要资源
关于ios - 位置变化时极地到笛卡尔的转换,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9271042/
我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]
这道题是thisquestion的逆题.给定一个散列,每个键都有一个数组,例如{[:a,:b,:c]=>1,[:a,:b,:d]=>2,[:a,:e]=>3,[:f]=>4,}将其转换为嵌套哈希的最佳方法是什么{:a=>{:b=>{:c=>1,:d=>2},:e=>3,},:f=>4,} 最佳答案 这是一个迭代的解决方案,递归的解决方案留给读者作为练习:defconvert(h={})ret={}h.eachdo|k,v|node=retk[0..-2].each{|x|node[x]||={};node=node[x]}node[
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
我正在使用Rails构建一个简单的聊天应用程序。当用户输入url时,我希望将其输出为html链接(即“url”)。我想知道在Ruby中是否有任何库或众所周知的方法可以做到这一点。如果没有,我有一些不错的正则表达式示例代码可以使用... 最佳答案 查看auto_linkRails提供的辅助方法。这会将所有URL和电子邮件地址变成可点击的链接(htmlanchor标记)。这是文档中的代码示例。auto_link("Gotohttp://www.rubyonrails.organdsayhellotodavid@loudthinking.
我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的
我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。
在启用Rack::Deflater来gzip我的响应主体时偶然发现了一些奇怪的东西。也许我遗漏了一些东西,但启用此功能后,响应被压缩,但是资源的ETag在每个请求上都会发生变化。这会强制应用程序每次都响应,而不是发送304。这在没有启用Rack::Deflater的情况下有效,我已经验证页面源没有改变。我正在运行一个使用thin作为Web服务器的Rails应用程序。Gemfile.lockhttps://gist.github.com/2510816有没有什么方法可以让我从Rack中间件获得更多的输出,这样我就可以看到发生了什么?提前致谢。 最佳答案
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里