Geometry.php 5.5 KB

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