根据项目需求编写的代码。

适用场景:在网络地图上,比如天地图与谷歌地图,用户用鼠标在地图上拉一个矩形框,希望下载该矩形框内某一层级的瓦片数据,并将所有瓦片拼接成一个完整的,包含地理坐标的tif图像。

那么在下载瓦片与拼接瓦片之前,用户希望能看到待下载的瓦片数量与待拼接图像的像素尺寸,再决定是否拼接。

该java代码根据该矩形框的经纬度范围与用户指定的瓦片层级,计算需要下载的瓦片数量与待拼接结果图像的像素尺寸。

支持EPSG4326经纬度与EPSG3857谷歌全球墨卡托投影。经纬度瓦片切图规则与天地图相同,从第一层开始切,第一层包含两个瓦片。谷歌全球墨卡托从第0层开始切,第0层一个瓦片。

public class Main {

    private void LonLatToTile(double lon,double lat,int zoom,int[] txy)
{
double resFact = 180.0 / 256.0;
double[] pxy = new double[]{0.0,0.0};
double res = resFact / Math.pow(2,(double)zoom);
pxy[0] = (180.0 + lon) / res;
pxy[1] = (90.0 - lat) / res; txy[0] = (int)(Math.ceil(pxy[0]/256.0) - 1);
txy[1] = (int)(Math.ceil(pxy[1]/256.0) - 1);
} private void LatLonToMeters(double lon, double lat,double[] mxy)
{
double m_originShift = 2 * 3.141592653589793 * 6378137 / 2.0; mxy[0] = lon * m_originShift / 180.0;
mxy[1] = Math.log( Math.tan((90 + lat) * 3.141592653589793 / 360.0 )) / (3.141592653589793 / 180.0); mxy[1] = mxy[1] * m_originShift / 180.0;
} private void MetersToTile(double mx, double my, int zoom, int[] txy)
{
double m_initialResolution = 2 * 3.141592653589793 * 6378137 / 256;
double m_originShift = 2 * 3.141592653589793 * 6378137 / 2.0; double res = m_initialResolution / Math.pow(2,(double)zoom);
double px = (mx + m_originShift) / res;
double py = (m_originShift - my) / res; txy[0] = (int)( Math.ceil( px / (float)(256) ) - 1 );
txy[1] = (int)( Math.ceil( py / (float)(256) ) - 1 );
} // 计算经纬度输出瓦片数量与待拼接图像像素尺寸
public int getGeodeticSize(double minLon, double maxLon, double minLat, double maxLat, int zoom, int[] pixSize){ int[] tminxy = new int[]{0,0};
int[] tmaxxy = new int[]{0,0}; LonLatToTile(minLon,minLat,zoom-1,tminxy);
LonLatToTile(maxLon,maxLat,zoom-1,tmaxxy); pixSize[0] = (1+Math.abs(tmaxxy[0]-tminxy[0])) * 256;
pixSize[1] = (1+Math.abs(tmaxxy[1]-tminxy[1])) * 256; int tnum = (1+Math.abs(tmaxxy[0]-tminxy[0])) * (1+Math.abs(tmaxxy[1]-tminxy[1])); return tnum;
}
// 计算谷歌投影输出瓦片数量与待拼接图像像素尺寸
public int getMercatorSize(double minLon, double maxLon, double minLat, double maxLat, int zoom, int[] pixSize){ double[] oULxy = new double[]{0,0};
double[] oDRxy = new double[]{0,0};
LatLonToMeters(minLon,maxLat,oULxy);
LatLonToMeters(maxLon,minLat,oDRxy);
double ominx = oULxy[0];
double omaxx = oDRxy[0];
double ominy = oDRxy[1];
double omaxy = oULxy[1]; int[] tminxy= new int[]{0,0};
int[] tmaxxy = new int[]{0,0};
MetersToTile(ominx,ominy, zoom, tminxy);
MetersToTile(omaxx,omaxy, zoom, tmaxxy); pixSize[0] = (1+Math.abs(tmaxxy[0]-tminxy[0])) * 256;
pixSize[1] = (1+Math.abs(tmaxxy[1]-tminxy[1])) * 256; int tnum = (1+Math.abs(tmaxxy[0]-tminxy[0])) * (1+Math.abs(tmaxxy[1]-tminxy[1])); return tnum;
} public static void main(String[] args) {
System.out.println("Hello World!");
double minLon = 119.54384371341310;
double maxLon = 119.93413672204591;
double minLat = 33.068895415323247;
double maxLat = 33.433168890047206;
int zoom = 13; Main e=new Main(); int[] pixSize= new int[]{0,0};
int tnum;
tnum = e.getGeodeticSize(minLon, maxLon, minLat, maxLat, zoom, pixSize);
System.out.println("经纬度数据瓦片数:" + tnum + " 图像尺寸:" + pixSize[0] + "*" + pixSize[1]);
tnum = e.getMercatorSize(minLon, maxLon, minLat, maxLat, zoom, pixSize);
System.out.println("谷歌数据瓦片数:" + tnum + " 图像尺寸:" + pixSize[0] + "*" + pixSize[1]); } }
运行结果:

												

JAVA代码根据经纬度范围计算WGS84与谷歌全球墨卡托包含的切片数目与拼接图像像素尺寸的更多相关文章

  1. 维吉尼亚密码java代码实现根据密钥长度计算IC值过程

    package cn.longxuzi; import java.util.Scanner; import org.junit.Test; public class ICUtils { /** * @ ...

  2. 200行Java代码搞定计算器程序

    发现了大学时候写的计算器小程序,还有个图形界面,能够图形化展示表达式语法树,哈哈;) 只有200行Java代码,不但能够计算加减乘除,还能够匹配小括号~ 代码点评: 从朴素的界面配色到简单易懂错误提示 ...

  3. 在java代码中执行js脚本,实现计算出字符串“(1+2)*(1+3)”的结果

            今天在公司项目中,发现一个计算运费的妙招.由于运费规则各种各样,因此写一个公式存到数据库.下次需要计算运费时,直接取出这个公式,把公式的未知变量给替换掉,然后计算出结果就是ok了. 一 ...

  4. 【JAVA】两点经纬度直线距离的计算

    来自谷歌地图的计算公式: 通过JAVA的Math类各种方法调用.实现上述公式 private static double EARTH_RADIUS = 6378.137;// 单位千米 /** * 角 ...

  5. Java代码计算运行时间

    突然想准确的测试一下Java代码的执行时间,在网上找了一会.发现基本有以下两种方法:第一种是以毫秒为单位计算的. Java代码 //伪代码 long startTime=System.currentT ...

  6. java 根据经纬度坐标计算两点的距离算法

    /** * @Desc 根据经纬度坐标计算两点的距离算法<br> * @Author yangzhenlong <br> * @Data 2018/5/9 18:38 */ p ...

  7. js代码--根据经纬度计算距离

    原网页地址:http://www.storyday.com/wp-content/uploads/2008/09/latlung_dis.html <!DOCTYPE html> < ...

  8. GeoHash核心原理解析及java代码实现(转)

    原文链接:http://blog.jobbole.com/80633/ 引子 机机是个好动又好学的孩子,平日里就喜欢拿着手机地图点点按按来查询一些好玩的东西.某一天机机到北海公园游玩,肚肚饿了,于是乎 ...

  9. 对一致性Hash算法,Java代码实现的深入研究

    一致性Hash算法 关于一致性Hash算法,在我之前的博文中已经有多次提到了,MemCache超详细解读一文中"一致性Hash算法"部分,对于为什么要使用一致性Hash算法.一致性 ...

随机推荐

  1. python之集合(set)学习

    集合(set) 集合是一个无序的不重复元素序列,使用大括号({}).set()函数创建集合, 注意:创建一个空集合必须用set()而不是{},因为{}是用来创建一个空字典. 集合是无序的.不重复的.没 ...

  2. 关于git 指令

    命令行操作(由于是Linux命令行下的普通用户,都是在$级别下操作): 一. 本机配置 添加用户 git config –global user.name “XX” git config –gloab ...

  3. Flume的各种类型的组件介绍

    1.   Source NetCat Source:绑定的端口(tcp.udp),将流经端口的每一个文本行数据作为Event输入: type:source的类型,必须是netcat. bind:要监听 ...

  4. IdentityServer4(7)- 使用客户端认证控制API访问(客户端授权模式)

    一.前言 本文已更新到 .NET Core 2.2 本文包括后续的Demo都会放在github:https://github.com/stulzq/IdentityServer4.Samples (Q ...

  5. redis 系列2 知识点概述

    一.概述 Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库.缓存和消息中间件. 它支持多种类型的数据结构,如 字符串(strings), 散列(hashes), 列表 ...

  6. Elasticsearch基本概念及核心配置文件详解

    Elasticsearch5.X,下列的是Elasticsearch2.X系类配置,其实很多配置都是相互兼容的 1. 配置文件 config/elasticsearch.yml 主配置文件 confi ...

  7. 《HelloGitHub月刊》第 01 期

    <HelloGitHub月刊> 因为现在这个项目只有我自己做,只敢叫"月刊",希望有志同道合者,快点加入到这个项目中来!同时,如果您有更好的建议或者意见,欢迎联系我.联 ...

  8. Java单元测试(Junit+Mock+代码覆盖率)

    微信公众号[程序员江湖] 作者黄小斜,斜杠青年,某985硕士,阿里 Java 研发工程师,于 2018 年秋招拿到 BAT 头条.网易.滴滴等 8 个大厂 offer,目前致力于分享这几年的学习经验. ...

  9. Chapter 4 Invitations——3

    Edward was never surrounded by crowds of curious by standers eager for his firsthand account. Edward ...

  10. Spring的后处理器-BeanPostProcessor跟BeanFactoryPostProcessors

    最近在重读spring源码(为什么要重读?因为不得不承认,去年跟着<深入解析sping源码>一书过了一遍spring的源码,除了满脑袋都是各种BeanFactory跟BeanDefinit ...