Geometry.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. <?php
  2. declare(strict_types=1);
  3. namespace algorithm;
  4. /**
  5. * 地理算法
  6. */
  7. class Geometry
  8. {
  9. /**
  10. * 火星坐标Gcj02 转 WGS84
  11. *
  12. * @param double $lat 纬度
  13. * @param double $lng 经度
  14. */
  15. public static function gcj02ToWgs84($lat, $lng)
  16. {
  17. $pi = 3.14159265358979324;
  18. $a = 6378245.0;
  19. $ee = 0.00669342162296594323;
  20. $dlat = self::wgsTOgcjTransformLat($lng - 105.0, $lat - 35.0);
  21. $dlng = self::wgsTOgcjTransformLon($lng - 105.0, $lat - 35.0);
  22. $radlat = $lat / 180.0 * $pi;
  23. $magic = sin($radlat);
  24. $magic = 1 - $ee * $magic * $magic;
  25. $sqrtmagic = sqrt($magic);
  26. $dlat = ($dlat * 180.0) / (($a * (1 - $ee)) / ($magic * $sqrtmagic) * $pi);
  27. $dlng = ($dlng * 180.0) / ($a / $sqrtmagic * cos($radlat) * $pi);
  28. $mglat = $lat + $dlat;
  29. $mglng = $lng + $dlng;
  30. return [
  31. 'lng' =>round($lng * 2 - $mglng,6),
  32. 'lat' =>round($lat * 2 - $mglat,6),
  33. ];
  34. }
  35. /**
  36. * 火星坐标Gcj02 转 百度地图坐标BD09
  37. * 腾讯地图用的也是GCJ02坐标
  38. * @param double $lat 纬度
  39. * @param double $lng 经度
  40. */
  41. public static function convertGcj02ToBd09($lat, $lng)
  42. {
  43. $lat = floatval($lat);
  44. $lng = floatval($lng);
  45. $x_pi = 3.14159265358979324 * 3000.0 / 180.0;
  46. $x = $lng;
  47. $y = $lat;
  48. $z =sqrt($x * $x + $y * $y) + 0.00002 * sin($y * $x_pi);
  49. $theta = atan2($y, $x) + 0.000003 * cos($x * $x_pi);
  50. $lng = $z * cos($theta) + 0.0065;
  51. $lat = $z * sin($theta) + 0.006;
  52. $lng = round($lng,6);
  53. $lat = round($lat,6);
  54. return array('lng'=>$lng,'lat'=>$lat);
  55. }
  56. /**
  57. * 百度地图坐标Bd09 转 火星坐标Gcj02
  58. * @param double $lat 纬度
  59. * @param double $lng 经度
  60. */
  61. public static function convertBd09ToGcj02($lat, $lng)
  62. {
  63. $lat = floatval($lat);
  64. $lng = floatval($lng);
  65. $x_pi = 3.14159265358979324 * 3000.0 / 180.0;
  66. $x = $lng - 0.0065;
  67. $y = $lat - 0.006;
  68. $z = sqrt($x * $x + $y * $y) - 0.00002 * sin($y * $x_pi);
  69. $theta = atan2($y, $x) - 0.000003 * cos($x * $x_pi);
  70. $lng = $z * cos($theta);
  71. $lat = $z * sin($theta);
  72. $lng = round($lng,6);
  73. $lat = round($lat,6);
  74. return array('lng'=>$lng,'lat'=>$lat);
  75. }
  76. /**
  77. * wgs84坐标转国测局坐标
  78. * @param double $lat 纬度
  79. * @param double $lng 经度
  80. */
  81. public static function wgsTOgcj($lat, $lng)
  82. {
  83. $pi = 3.14159265358979324;
  84. $a = 6378245.0;
  85. $ee = 0.00669342162296594323;
  86. $wgLat = $lat;
  87. $wgLon = $lng;
  88. if (self::outOfChina($wgLat, $wgLon)){
  89. return array('lat'=>$wgLat, 'lng'=>$wgLon);
  90. }
  91. $dLat = self::wgsTOgcjTransformLat($wgLon - 105.0, $wgLat - 35.0);
  92. $dLon = self::wgsTOgcjTransformLon($wgLon - 105.0, $wgLat - 35.0);
  93. $radLat = $wgLat / 180.0 * $pi;
  94. $magic = sin($radLat);
  95. $magic = 1 - $ee * $magic * $magic;
  96. $sqrtMagic = sqrt($magic);
  97. $dLat = ($dLat * 180.0) / (($a * (1 - $ee)) / ($magic * $sqrtMagic) * $pi);
  98. $dLon = ($dLon * 180.0) / ($a / $sqrtMagic * cos($radLat) * $pi);
  99. $mgLat = $wgLat + $dLat;
  100. $mgLon = $wgLon + $dLon;
  101. return array('lat'=>$mgLat, 'lng'=>$mgLon);
  102. }
  103. /**
  104. * 经纬度是否超出中国范围
  105. * @param double $lat 纬度
  106. * @param double $lng 经度
  107. */
  108. public static function outOfChina($lat, $lng)
  109. {
  110. if ($lng < 72.004 || $lng > 137.8347)
  111. return true;
  112. if ($lat < 0.8293 || $lat > 55.8271)
  113. return true;
  114. return false;
  115. }
  116. /**
  117. * wgsTOgcj转换纬度
  118. * @param double $lat 纬度
  119. * @param double $lng 经度
  120. */
  121. private static function wgsTOgcjTransformLat( $x, $y ){
  122. $pi = 3.14159265358979324;
  123. $ret = -100.0 + 2.0 * $x + 3.0 * $y + 0.2 * $y * $y + 0.1 * $x * $y + 0.2 * sqrt(abs($x));
  124. $ret += (20.0 * sin(6.0 * $x * $pi) + 20.0 * sin(2.0 * $x * $pi)) * 2.0 / 3.0;
  125. $ret += (20.0 * sin($y * $pi) + 40.0 * sin($y / 3.0 * $pi)) * 2.0 / 3.0;
  126. $ret += (160.0 * sin($y / 12.0 * $pi) + 320 * sin($y * $pi / 30.0)) * 2.0 / 3.0;
  127. return $ret;
  128. }
  129. /**
  130. * wgsTOgcj转换经度
  131. * @param double $lat 纬度
  132. * @param double $lng 经度
  133. */
  134. private static function wgsTOgcjTransformLon( $x, $y ){
  135. $pi = 3.14159265358979324;
  136. $ret = 300.0 + $x + 2.0 * $y + 0.1 * $x * $x + 0.1 * $x * $y + 0.1 * sqrt(abs($x));
  137. $ret += (20.0 * sin(6.0 * $x * $pi) + 20.0 * sin(2.0 * $x * $pi)) * 2.0 / 3.0;
  138. $ret += (20.0 * sin($x * $pi) + 40.0 * sin($x / 3.0 * $pi)) * 2.0 / 3.0;
  139. $ret += (150.0 * sin($x / 12.0 * $pi) + 300.0 * sin($x / 30.0 * $pi)) * 2.0 / 3.0;
  140. return $ret;
  141. }
  142. /**
  143. * 获取百度2点间距离(单位:km)
  144. * @param $point1 坐标1
  145. * @param $point2 坐标2
  146. */
  147. public static function distanceBetween2BdPoints($point1, $point2){
  148. $PI = 3.1415926535898;
  149. $earthRadius = 6371.393;//单位:km
  150. $radLat1 = $point1['lat'] * ($PI / 180);
  151. $radLat2 = $point2['lat'] * ($PI / 180);
  152. $a = $radLat1 - $radLat2;
  153. $b = ($point1['lng'] * ($PI / 180)) - ($point2['lng'] * ($PI / 180));
  154. $s = 2 * asin(sqrt(pow(sin($a/2),2) + cos($radLat1)*cos($radLat2)*pow(sin($b/2),2)));
  155. $s = $s * $earthRadius;
  156. $s = round($s * 10000) / 10000;
  157. return $s;
  158. }
  159. /**
  160. *坐标点是否在多边形范围内
  161. * @param $polygon 多边形点数组
  162. * @param $lnglat 经纬度数组
  163. */
  164. public static function isInPolygon($polygon, $lnglat){
  165. $count = count($polygon);
  166. $px = $lnglat['lat'];
  167. $py = $lnglat['lng'];
  168. $flag = FALSE;
  169. for ($i = 0, $j = $count - 1; $i < $count; $j = $i, $i++) {
  170. $sy = $polygon[$i]['lng'];
  171. $sx = $polygon[$i]['lat'];
  172. $ty = $polygon[$j]['lng'];
  173. $tx = $polygon[$j]['lat'];
  174. if ($px == $sx && $py == $sy || $px == $tx && $py == $ty)
  175. return TRUE;
  176. if ($sy < $py && $ty >= $py || $sy >= $py && $ty < $py) {
  177. $x = $sx + ($py - $sy) * ($tx - $sx) / ($ty - $sy);
  178. if ($x == $px) return TRUE;
  179. if ($x > $px)
  180. $flag = !$flag;
  181. }
  182. }
  183. return $flag;
  184. }
  185. }