zoukankan      html  css  js  c++  java
  • geo-经纬度计算

    经纬度计算, 本质上是球面三角函数的应用, 将数学公式转换为代码的过程, 

    站在前人的肩膀上, 自己又补充了一点: 

    package com.iwhere.easy.travel.tool;
    
    public class GeoTools {
        
        public static String getScopeByGeoLevel(Integer geolevel) {
            return ((16 - geolevel) * 5) + "公里";
        }
    
        public static double getAngle(double lat1, double lng1, double lat2, double lng2) {
            double x1 = lng1;
            double y1 = lat1;
            double x2 = lng2;
            double y2 = lat2;
            double pi = Math.PI;
            double w1 = y1 / 180 * pi;
            double j1 = x1 / 180 * pi;
            double w2 = y2 / 180 * pi;
            double j2 = x2 / 180 * pi;
            double ret;
            if (j1 == j2) {
                if (w1 > w2)
                    return 270; // 北半球的情况,南半球忽略
                else if (w1 < w2)
                    return 90;
                else
                    return -1;// 位置完全相同
            }
            ret = 4 * Math.pow(Math.sin((w1 - w2) / 2), 2)
                    - Math.pow(Math.sin((j1 - j2) / 2) * (Math.cos(w1) - Math.cos(w2)), 2);
            ret = Math.sqrt(ret);
            double temp = (Math.sin(Math.abs(j1 - j2) / 2) * (Math.cos(w1) + Math.cos(w2)));
            ret = ret / temp;
            ret = Math.atan(ret) / pi * 180;
            if (j1 > j2) { // 1为参考点坐标
                if (w1 > w2)
                    ret += 180;
                else
                    ret = 180 - ret;
            } else if (w1 > w2)
                ret = 360 - ret;
            return ret;
        }
    
        /**
         * @param lat1
         *            纬度1,参考点
         * @param lng1
         *            经度1,参考点
         * @param lat2
         *            纬度2
         * @param lng2
         *            经度2
         * @return 方向
         */
        public static String getDirection(double lat1, double lng1, double lat2, double lng2) {
            double jiaodu = getAngle(lat1, lng1, lat2, lng2);
            if ((jiaodu <= 10) || (jiaodu > 350))
                return "正东方";
            if ((jiaodu > 10) && (jiaodu <= 80))
                return "东北方";
            if ((jiaodu > 80) && (jiaodu <= 100))
                return "正北方";
            if ((jiaodu > 100) && (jiaodu <= 170))
                return "西北方";
            if ((jiaodu > 170) && (jiaodu <= 190))
                return "正西方";
            if ((jiaodu > 190) && (jiaodu <= 260))
                return "西南方";
            if ((jiaodu > 260) && (jiaodu <= 280))
                return "正南方";
            if ((jiaodu > 280) && (jiaodu <= 350))
                return "东南方";
            return "";
        }
    
        /**
         * 获取方位角描述
         * 前一个点
         * 参照点
         * 目标点
         * @author wenbronk
         */
        public static String getAzimuth(double preLng, double preLat, double referLng, double referLat, double poiLng,
                double poiLat) {
            double jiaodu1 = getAngle(preLat, preLng, referLat, referLng);
            double jiaodu2 = getAngle(referLat, referLng, poiLat, poiLng);
            double jiaodu = jiaodu2 - jiaodu1;
            if (((jiaodu <=10 && jiaodu > 0) || jiaodu > 350) || ((jiaodu >= -10 && jiaodu < 0) || jiaodu < -350)) {
                return "正前方";
            }else if ((jiaodu <= 80 && jiaodu > 10) || (jiaodu >= -350 && jiaodu < -280)) {
                return "左前方";
            }else if ((jiaodu <= 100 && jiaodu > 80) || (jiaodu >= -280 && jiaodu < -260)) {
                return "左侧";
            }else if ((jiaodu <= 170 && jiaodu > 100) || (jiaodu >= -260 && jiaodu < -190)) {
                return "左后方";
            }else if ((jiaodu <= 190 && jiaodu > 170) || (jiaodu >= -190 && jiaodu < -170)) {
                return "后方";
            }else if ((jiaodu <=260 && jiaodu > 190) || (jiaodu >= -170 && jiaodu < -100)) {
                return "右后方";
            }else if ((jiaodu <=280 && jiaodu > 260) || (jiaodu >= -100 && jiaodu < -80)) {
                return "右侧";
            }else if ((jiaodu <= 350 && jiaodu > 280) || (jiaodu >= -80 && jiaodu < -10)) {
                return "右前方";
            }
            return "";
        }
        
        /**
         * 判断是否在后方
         * @author wenbronk
         */
        public static Boolean isBehine(double preLng, double preLat, double referLng, double referLat, 
                double poiLng, double poiLat) {
            double x1 = referLng - preLng;
            double y1 = referLat - preLat;
            double x2 = poiLng - referLng;
            double y2 = poiLat - referLat;
            
            // 求 (x1, y1), (x2, y2) 向量的夹角的cos值
            double cos = (x1 * x2 + y1 * y2) / 
                    (Math.sqrt(Math.pow(x1, 2) + Math.pow(y1, 2)) + Math.sqrt(Math.pow(x2, 2) + Math.pow(y2, 2)));
            if (cos >= 0) {
                return false;
            }
            return true;
        }
        
        public static void main(String[] args) {
            String str = getDirection(0, 0, -40, -34);
            System.out.println(str);
        }
    }
  • 相关阅读:
    使用ajax和window.history.pushState无刷新改变页面内容和地址栏URL
    URL中“#” “?” &“”号的作用
    JavaScript中事件捕获(Event capturing)-------------->由外向内,事件冒泡(Event bubblin)---------->由内向外
    单页面应用程序案例
    【344】Jupyter relevant problems
    【343】MathJax、LaTex、Mathml 数学公式
    【342】Linear Regression by Python
    【341】Numpy 相关应用
    【340】GIS related knowledge
    java中通过反射获取方法并且调用(getMethod和invoke深入)实践
  • 原文地址:https://www.cnblogs.com/wenbronk/p/7273871.html
Copyright © 2011-2022 走看看