Javascript中的十进制度到度分和秒
Decimal Degrees to Degrees Minutes And Seconds in Javascript
我正试图编写一个函数,该函数使用我的十进制度数(lat或long(,并将其转换为DMS度数(分秒(。我知道我要把小数点乘以60,然后它又是十进制的。但我是个傻瓜。我可以平分号码吗?
function ConvertDDToDMS(DD) {
eg. DD =-42.4
D= 42;
M= 4*60;
S= .M * 60;
var DMS =
return DMS //append Direction (N, S, E, W);
}
我走对了吗?
function ConvertDDToDMS(D, lng) {
return {
dir: D < 0 ? (lng ? "W" : "S") : lng ? "E" : "N",
deg: 0 | (D < 0 ? (D = -D) : D),
min: 0 | (((D += 1e-9) % 1) * 60),
sec: (0 | (((D * 60) % 1) * 6000)) / 100,
};
}
上面给出了一个对象{deg, min, sec, dir}
,其中sec被截断为两位数字(例如3.14
(,dir是N
、E
、S
和W
之一,具体取决于是否将lng
(经度(参数设置为true。例如:
ConvertDDToDMS(-18.213, true) == {
deg : 18,
min : 12,
sec : 46.79,
dir : 'W'
}
或者,如果你只想要基本字符串:
function ConvertDDToDMS(D){
return [0|D, 'd ', 0|(D=(D<0?-D:D)+1e-4)%1*60, "' ", 0|D*60%1*60, '"'].join('');
}
ConvertDDToDMS(-18.213) == `-18d 12' 47"`
[编辑2019年6月]-修复了一个8年前的错误,该错误有时会导致在转换精确分钟(例如ConvertDDToDMS(4 + 20/60)
(时由于浮点运算而导致结果延迟1分钟。
【编辑2021年12月】——哇。修复#2。回到原始代码,将1e-9
添加到值中,a(将任何稍低的浮点误差提升到下一个最高值,b(小于.01
秒,因此对输出没有影响。将CCD_ 11添加到";字符串";版本,它是相同的修复,但也舍入秒(接近1/2秒(。
不清楚您需要如何输出。这是一个将所有3个值作为字符串返回的版本:
function ConvertDDToDMS(dd)
{
var deg = dd | 0; // truncate dd to get degrees
var frac = Math.abs(dd - deg); // get fractional part
var min = (frac * 60) | 0; // multiply fraction by 60 and truncate
var sec = frac * 3600 - min * 60;
return deg + "d " + min + "' " + sec + "'"";
}
更新:我删除了没有任何意义的部分(感谢cwolves!(。
这里还有另一个实现。它不会像以前的那样简短和高效,但希望更容易理解。
要做到正确,首先需要了解计算是如何完成的,然后才能尝试实现它们。为此,伪代码是一个很好的选择,因为你可以用通俗易懂的英语或简化语法写下步骤,然后将其翻译成所选的编程语言。
我希望它有用!
/* This is the pseudocode you need to follow:
* It's a modified version from
* http://en.wikipedia.org/wiki/Geographic_coordinate_conversion#Conversion_from_Decimal_Degree_to_DMS
function deg_to_dms ( degfloat )
Compute degrees, minutes and seconds:
deg ← integerpart ( degfloat )
minfloat ← 60 * ( degfloat - deg )
min ← integerpart ( minfloat )
secfloat ← 60 * ( minfloat - min )
Round seconds to desired accuracy:
secfloat ← round( secfloat, digits )
After rounding, the seconds might become 60. These two
if-tests are not necessary if no rounding is done.
if secfloat = 60
min ← min + 1
secfloat ← 0
end if
if min = 60
deg ← deg + 1
min ← 0
end if
Return output:
return ( deg, min, secfloat )
end function
*/
function deg_to_dms (deg) {
var d = Math.floor (deg);
var minfloat = (deg-d)*60;
var m = Math.floor(minfloat);
var secfloat = (minfloat-m)*60;
var s = Math.round(secfloat);
// After rounding, the seconds might become 60. These two
// if-tests are not necessary if no rounding is done.
if (s==60) {
m++;
s=0;
}
if (m==60) {
d++;
m=0;
}
return ("" + d + ":" + m + ":" + s);
}
试试这个完美的工作!!!
function truncate(n) {
return n > 0 ? Math.floor(n) : Math.ceil(n);
}
function getDMS(dd, longOrLat) {
let hemisphere = /^[WE]|(?:lon)/i.test(longOrLat)
? dd < 0
? "W"
: "E"
: dd < 0
? "S"
: "N";
const absDD = Math.abs(dd);
const degrees = truncate(absDD);
const minutes = truncate((absDD - degrees) * 60);
const seconds = ((absDD - degrees - minutes / 60) * Math.pow(60, 2)).toFixed(2);
let dmsArray = [degrees, minutes, seconds, hemisphere];
return `${dmsArray[0]}°${dmsArray[1]}'${dmsArray[2]}" ${dmsArray[3]}`;
}
var lat = 13.041107;
var lon = 80.233232;
var latDMS = getDMS(lat, 'lat');
var lonDMS = getDMS(lon, 'long');
console.log('latDMS: '+ latDMS);
console.log('lonDMS: '+ lonDMS);
Output:
latDMS: 13°2'27.99" N
lonDMS: 80°13'59.64" E
一个解决方案,具有指定输出秒中小数位数的选项,并可校正由于舍入秒和分钟而引起的任何边缘情况。
// @ input {deg} Numeric; degrees number to convert
// @ input {dplaces} Decimal places to use for output seconds
// Default 0 places
// @ return {DMS} string degrees (°) minutes (') seconds (")
//
function degToDMS (deg, dplaces=0) {
var d = Math.floor (deg); // make degrees
var m = Math.floor((deg-d)*60); // make minutes
var s = Math.round(((deg-d)*60-m)*60*Math.pow(10,dplaces))/Math.pow(10,dplaces); // Make sec rounded
s == 60 && (m++, s=0 ); // if seconds rounds to 60 then increment minutes, reset seconds
m == 60 && (d++, m=0 ); // if minutes rounds to 60 then increment degress, reset minutes
return (d + "° " + m + "' " + s+'"'); // create output DMS string
}
// ----- tests ------
console.log(degToDMS(55.23456)); // 55° 14' 4"
console.log(degToDMS(55.23456 ,3)); // 55° 14' 4.416"
console.log(degToDMS(4 + 20/60 ,2)); // 4° 20' 0"
console.log(degToDMS(89.64789 ,2)); // 89° 38' 52.4"
console.log(degToDMS(-23.1234567,3)); // -24° 52' 35.556"
这个在TypeScript:中工作%100
ConvertDDToDMS(deg: number, lng: boolean): string {
var d = parseInt(deg.toString());
var minfloat = Math.abs((deg - d) * 60);
var m = Math.floor(minfloat);
var secfloat = (minfloat - m) * 60;
var s = Math.round((secfloat + Number.EPSILON) * 100) / 100
d = Math.abs(d);
if (s == 60) {
m++;
s = 0;
}
if (m == 60) {
d++;
m = 0;
}
let dms = {
dir: deg < 0 ? lng ? 'W' : 'S' : lng ? 'E' : 'N',
deg: d,
min: m,
sec: s
};
return `${dms.deg}'u00B0 ${dms.min}' ${dms.sec}" ${dms.dir}`
}
private static DecimalFormat DecimalFormat = new DecimalFormat(".##");
public static void main(String[] args){
double decimal_degrees = 22.4229541515;
System.out.println(getDMS(decimal_degrees));
}
public static String getDMS(double decimal_degrees) {
double degree = Math.floor(decimal_degrees);
double minutes = ((decimal_degrees - Math.floor(decimal_degrees)) * 60.0);
double seconds = (minutes - Math.floor(minutes)) * 60.0;
return ((int)degree)+":"+((int)minutes)+":"+decimalFormat.format(seconds);
}
输入:22.4229541515输出:22:25:22.63
基于上面的答案,我将它们写成了javascript和php风格。
JS-
function convertDDToDMS(deg, lng){
var d = parseInt(deg);
var minfloat = Math.abs((deg-d) * 60);
var m = Math.floor(minfloat);
var secfloat = (minfloat-m)*60;
var s = Math.round(secfloat);
d = Math.abs(d);
if (s==60) {
m++;
s=0;
}
if (m==60) {
d++;
m=0;
}
return {
dir : deg<0?lng?'W':'S':lng?'E':'N',
deg : d,
min : m,
sec : s
};
}
PHP-
function convertDDtoDMS($deg, $lng){
$dd = intval($deg);
$minfloat = abs(($deg - $dd) * 60);
$mm = floor($minfloat);
$secfloat = ($minfloat - $mm) * 60;
$ss = round($secfloat);
$dd = abs($dd);
if($ss == 60){
$mm++;
$ss = 0;
}
if($mm == 60){
$dd++;
$mm = 0;
}
$dd = array(
'dir' => $deg < 0 ? ($lng ? 'W' : 'S') : ($lng ? 'E' : 'N'),
'deg' => abs($dd),
'min' => $mm,
'sec' => $ss,
);
return $dd;
}
经过一段时间后,无法使上面的脚本正常工作;只需将dms交给脚本
function ConvertDMSToDEG(dms) {
var dms_Array = dms.split(/[^'d'w'.]+/);
var degrees = dms_Array[0];
var minutes = dms_Array[1];
var seconds = dms_Array[2];
var direction = dms_Array[3];
var deg = (Number(degrees) + Number(minutes)/60 + Number(seconds)/3600).toFixed(6);
if (direction == "S" || direction == "W") {
deg = deg * -1;
} // Don't do anything for N or E
return deg;
}
反之亦然,只给脚本提供度数,lat(纬度(为true或false
function ConvertDEGToDMS(deg, lat) {
var absolute = Math.abs(deg);
var degrees = Math.floor(absolute);
var minutesNotTruncated = (absolute - degrees) * 60;
var minutes = Math.floor(minutesNotTruncated);
var seconds = ((minutesNotTruncated - minutes) * 60).toFixed(2);
if (lat) {
var direction = deg >= 0 ? "N" : "S";
} else {
var direction = deg >= 0 ? "E" : "W";
}
return degrees + "°" + minutes + "'" + seconds + "'"" + direction;
}
希望这能帮助人们。。
我很惊讶所有的解决方案都使用了一些额外的逻辑来处理"舍入到60〃;案例(如果他们知道的话(,但没有人想过用另一种方式,从(四舍五入(秒开始,然后使用mod和intdiv,不必担心所有这些:
function coordToStr(coord)
{
let seconds = Math.round(Math.abs(coord) * 3600)
let sec = Math.floor(seconds % 60)
let minutes = Math.floor(seconds / 60)
let min = minutes % 60
let deg = Math.floor(minutes / 60)
return deg + "°" + ((min < 10) ? "0" : "") + min + "'" + ((sec < 10) ? "0" : "") + sec
}
对不起,这没有N/S,E/W部分,需要一些额外的方法来调用它。
如果你想要第二个分数,你可以使用这个:
function coordToStrWithDecimals(coord)
{
let centiSecs = Math.round(Math.abs(coord) * 360000)
let frac = Math.floor(centiSecs % 100)
let seconds = Math.floor(centiSecs / 100)
let sec = Math.floor(seconds % 60)
let minutes = Math.floor(seconds / 60)
let min = minutes % 60
let deg = Math.floor(minutes / 60)
return deg + "°" + ((min < 10) ? "0" : "") + min + "'" + ((sec < 10) ? "0" : "") + sec + "." + ((frac < 10) ? "0" : "") + frac + '"'
}
这里的答案很好,但要获得标准格式的DMS。
function padZero(num, targetLength) {
return String(num).padStart(targetLength, "0");
}
function ddToDms(deg, latOrlon) {
var absolute = Math.abs(deg);
var degrees = Math.floor(absolute);
var minNt = (absolute - degrees) * 60;
var minutes = Math.floor(minNt);
var seconds = ((minNt - minutes) * 60).toFixed(2);
var secs = Math.floor(seconds);
// Get cardinal direction
if (latOrlon == "lat") {
var direction = deg >= 0 ? "N" : "S";
} else if (latOrlon == "lon") {
var direction = deg >= 0 ? "E" : "W";
}
// Ensure 60 minutes add 1 to degree and 60 seconds add 1 to minutes
if (seconds == 60) {
minutes++;
seconds = 0;
}
if (minutes == 60) {
degrees++;
minutes = 0;
}
// Pad with zero
if (
(degrees < 10 && latOrlon == "lat") ||
(degrees > 10 && degrees < 100 && latOrlon == "lon")
) {
var degrees = padZero(degrees, String(degrees).length + 1);
} else if (degrees < 10 && latOrlon == "lon") {
var degrees = padZero(degrees, String(degrees).length + 2);
}
if (minutes < 10) {
var minutes = padZero(minutes, String(minutes).length + 1);
}
if (secs < 10) {
var seconds = padZero(seconds, String(seconds).length + 1);
}
// Validate lat and lon
if (deg > 90 && latOrlon == "lat") {
alert("LATITUDE CANNOT BE MORE THAN 90");
} else if (deg > 180 && latOrlon == "lon") {
alert("LONGITUDE CANNOT BE MORE THAN 180");
} else {
return degrees + "°" + minutes + "'" + seconds + '"' + direction;
}
}
// Example
var lat = 9.129438;
var lon = -4.233587;
var latDMS = ddToDms(lat, 'lat');
var lonDMS = ddToDms(lon, 'lon');
console.log('latDMS: '+ latDMS);
console.log('lonDMS: '+ lonDMS);
/*
Output
latDMS: 09°07'45.98"N
lonDMS: 004°14'00.91"W
*/
仅用于备注,答案
function ConvertDDToDMS(D){
return [0|D, 'd ', 0|(D<0?D=-D:D)%1*60, "' ", 0|D*60%1*60, '"'].join('');
}
对于-1°和0°之间的角度是否无效。运气不好!hc
- 在Javascript和PHP中,十进制到RGB
- 在javascript中接受十进制数字的Regex
- 如何使用javascript验证在表单输入中避免负值、浮点值和十进制值
- JavaScript basis 是 math.pow 函数中的十进制数
- 如何使用 JavaScript 自动设置十进制值
- Javascript - 具有非线性刻度的负十进制输入范围
- 如何使用 javascript 将浮点数格式化为十进制数(7)
- 循环 += 十进制变量的 javascript 有问题
- 有人可以详细解释这个 JavaScript 十进制舍入函数吗?
- Javascript:我怎么能强迫一个有十进制值的字符串使用点而不是逗号呢
- Javascript 十进制计算
- Javascript 十进制到二进制 - 64 位
- Javascript十进制减法
- 节点JavaScript-十进制字符串到整数
- Javascript-十进制格式
- 神秘的2出现在我的JavaScript十进制到二进制转换器的输出中
- Javascript十进制到十六进制
- Javascript十进制(低16位)到十进制
- 简单的javascript十进制加法会产生不正确的结果
- JavaScript十进制数字