GPS를 이용한 두지점간 거리계산 방법입니다.
안드로이드에서 기본적으로 제공해주는 직선거리 구하는 방법
지구는 타원으로 되어있기에 가까운지역인 경우 오차가 없지만,
거리가 멀어지면 오차가 발생하는걸 알 수 있습니다.
방법1.
double distance;
Location locationA = new Location("point A");
locationA.setLatitude(latA);
locationA.setLongitude(lngA);
Location locationB = new Location("point B");
locationB.setLatitude(latB);
LocationB.setLongitude(lngB);
distance = locationA.distanceTo(locationB);
Location locationA = new Location("point A");
locationA.setLatitude(latA);
locationA.setLongitude(lngA);
Location locationB = new Location("point B");
locationB.setLatitude(latB);
LocationB.setLongitude(lngB);
distance = locationA.distanceTo(locationB);
방법2.
float[] results = new float[3]; Location location = new Location("start"); location.distanceBetween( startLatitude, startLongitude, endLatitude, endLongitude, results); Sring mes = Float.toString(results[0]); =============================================================================== The computed distance is stored in results[0].
If results has length 2 or greater, the initial bearing is stored in results[1].
If results has length 3 or greater, the final bearing is stored in results[2].
국토지리원에서 공개한 거리계산 함수 사용하기.
지구는 완전한 구라는 가정하에 지구를 GRS80타원체로 봤을경우
두지점간 거리와 방위각 구하는 소스
//거리 구하는 부분
public double distance(double P1_latitude, double P1_longitude, double P2_latitude, double P2_longitude) { if ((P1_latitude == P2_latitude) && (P1_longitude == P2_longitude)) { return 0; }
double e10 = P1_latitude * Math.PI / 180; double e11 = P1_longitude * Math.PI / 180; double e12 = P2_latitude * Math.PI / 180; double e13 = P2_longitude * Math.PI / 180;
/* 타원체 GRS80 */ double c16 = 6356752.314140910; double c15 = 6378137.000000000; double c17 = 0.0033528107;
double f15 = c17 + c17 * c17; double f16 = f15 / 2; double f17 = c17 * c17 / 2; double f18 = c17 * c17 / 8; double f19 = c17 * c17 / 16;
double c18 = e13 - e11; double c20 = (1 - c17) * Math.tan(e10); double c21 = Math.atan(c20); double c22 = Math.sin(c21); double c23 = Math.cos(c21); double c24 = (1 - c17) * Math.tan(e12); double c25 = Math.atan(c24); double c26 = Math.sin(c25); double c27 = Math.cos(c25);
double c29 = c18; double c31 = (c27 * Math.sin(c29) * c27 * Math.sin(c29)) + (c23 * c26 - c22 * c27 * Math.cos(c29)) * (c23 * c26 - c22 * c27 * Math.cos(c29)); double c33 = (c22 * c26) + (c23 * c27 * Math.cos(c29)); double c35 = Math.sqrt(c31) / c33; double c36 = Math.atan(c35); double c38 = 0; if (c31 == 0) { c38 = 0; } else { c38 = c23 * c27 * Math.sin(c29) / Math.sqrt(c31); }
double c40 = 0; if ((Math.cos(Math.asin(c38)) * Math.cos(Math.asin(c38))) == 0) { c40 = 0; } else { c40 = c33 - 2 * c22 * c26 / (Math.cos(Math.asin(c38)) * Math.cos(Math.asin(c38))); }
double c41 = Math.cos(Math.asin(c38)) * Math.cos(Math.asin(c38)) * (c15 * c15 - c16 * c16) / (c16 * c16); double c43 = 1 + c41 / 16384 * (4096 + c41 * (-768 + c41 * (320 - 175 * c41))); double c45 = c41 / 1024 * (256 + c41 * (-128 + c41 * (74 - 47 * c41))); double c47 = c45 * Math.sqrt(c31) * (c40 + c45 / 4 * (c33 * (-1 + 2 * c40 * c40) - c45 / 6 * c40 * (-3 + 4 * c31) * (-3 + 4 * c40 * c40))); double c50 = c17 / 16 * Math.cos(Math.asin(c38)) * Math.cos(Math.asin(c38)) * (4 + c17 * (4 - 3 * Math.cos(Math.asin(c38)) * Math.cos(Math.asin(c38)))); double c52 = c18 + (1 - c50) * c17 * c38 * (Math.acos(c33) + c50 * Math.sin(Math.acos(c33)) * (c40 + c50 * c33 * (-1 + 2 * c40 * c40)));
double c54 = c16 * c43 * (Math.atan(c35) - c47);
// return distance in meter return c54; }
//방위각 구하는 부분
public short bearingP1toP2(double P1_latitude, double P1_longitude, double P2_latitude, double P2_longitude) { // 현재 위치 : 위도나 경도는 지구 중심을 기반으로 하는 각도이기 때문에
//라디안 각도로 변환한다. double Cur_Lat_radian = P1_latitude * (3.141592 / 180); double Cur_Lon_radian = P1_longitude * (3.141592 / 180);
// 목표 위치 : 위도나 경도는 지구 중심을 기반으로 하는 각도이기 때문에
// 라디안 각도로 변환한다. double Dest_Lat_radian = P2_latitude * (3.141592 / 180); double Dest_Lon_radian = P2_longitude * (3.141592 / 180);
// radian distance double radian_distance = 0; radian_distance = Math.acos(Math.sin(Cur_Lat_radian) * Math.sin(Dest_Lat_radian) + Math.cos(Cur_Lat_radian) * Math.cos(Dest_Lat_radian) * Math.cos(Cur_Lon_radian - Dest_Lon_radian));
// 목적지 이동 방향을 구한다.(현재 좌표에서 다음 좌표로 이동하기 위해서는
//방향을 설정해야 한다. 라디안값이다. double radian_bearing = Math.acos((Math.sin(Dest_Lat_radian) - Math .sin(Cur_Lat_radian) * Math.cos(radian_distance)) / (Math.cos(Cur_Lat_radian) * Math.sin(radian_distance)));
// acos의 인수로 주어지는 x는 360분법의 각도가 아닌 radian(호도)값이다.
double true_bearing = 0; if (Math.sin(Dest_Lon_radian - Cur_Lon_radian) < 0) { true_bearing = radian_bearing * (180 / 3.141592); true_bearing = 360 - true_bearing; } else { true_bearing = radian_bearing * (180 / 3.141592); }
return (short) true_bearing; }
이외 거리 구하는 참고자료는 첨부파일 참고해주세요~
댓글 없음:
댓글 쓰기