在这里我们看到判断Lod的级别主要有三个条件:
 * 1、相机视角范围,视角范围越大,所包含的tileSize就越大
 * 2、相机与瓦片距离,距离越远,所包含的tileSize也就越大
 * 3、相机视锥与瓦片是否相交
 相对应我们可以把视角剔除方法理解成以下三步:
 * 1、根据视角范围,画个大圈,把大圈里的大瓦片全加进来
 * 2、根据相机与瓦片的距离进行细化瓦片,如果太远的瓦片仍然保持很大,而近处的瓦片需要进一步更新,计算子瓦片。
 * 3、每计算一个瓦片都需要判断它是否在视锥里,否则剔除身后看不到的瓦片
 在理解以上三个条件的前提下,需要理解的是视角、距离到底与tileSize有什么关系。
 对于不同级别影像或地形,ww中设置了相应的大小,根据一定的比例系数和经验参数寻找与距离和视角的关系。即通过调整参数可控制在怎样的距离下可以看到怎样级别的地形或影像。例如在地形中设置为3.0 2.9等常数,在影像中设置为TileDrawDistance和TileDrawSpread 等常数。只是反映了一定的经验参数而已。

http://blog.csdn.net/jk276993857/article/details/5937941

1.采用dstile制作WW瓦片,需要传人参数Level0的大小和建立金字塔的等级,这是一种自顶向下的方法。顶是指金字塔的顶部。

  创建瓦片的通用命令行格式如下:

  dstile.exe tile --lztsd tile_size --wwcache --overviews output_directory georeferenced_image_file(s)

  参数如下:

  tile_size - 0图层瓦片大小,十进制数,能被180正常(最好为180/(2^N))。

  output_directory - 瓦片金字塔的输出路径。(如何没有盘符只要文件夹名,是在FWTool安装目录下)

  georeferenced_image_file(s) - 影像数据的完整路径(如果影像数据在FWTool安装目录下的文件夹下,可以使用相对路径).一系列的多重影像可以自动组合成单一的瓦片金字塔。( A list of multiple image files can be supplied and will be automatically combined into a single set of tiles.)

举例:确定金字塔的顶部大小比如2.25度,确定了0级的瓦片数量160*160,根据地球的周长,确定了每个瓦片的长度范围,对于了256pixels的瓦片,则每个像素的分辨率可知。
  确定了level0的tileSize,也就间接确定了Level0显示的相机(CameraBase)高度(_altitude),通过viewRange可视范围、视域体和相机中心与瓦片球面夹角确定。

  制作WW瓦片时,确定了levlZeroTileSizeDegrees后,瓦片分多少级合适应该由影像的最高精度决定。 同样的相机高度,对于不同的瓦片数据集的levelZeroTileSizeDegrees是不一样的,所以在同样的高度存在多个瓦片等级,没有一个统一的值。

CameraBase的Update()方法中计算可视范围:

 // Old view range (used in quadtile logic)
double factor = (this._altitude) / this._worldRadius;
if(factor > )
viewRange = Angle.FromRadians(Math.PI);
else
viewRange = Angle.FromRadians(Math.Abs(Math.Asin((this._altitude) / this._worldRadius))*);

  在QuadTile和SurfaceTile的Update()方法中,一项任务就是:初始化可见瓦片的瓦片;将视点中心部分瓦片细化,外围不细化。首先初始化瓦片,

 if (!isInitialized)
{
if (DrawArgs.Camera.ViewRange * 0.5f < Angle.FromDegrees(QuadTileSet.TileDrawDistance * tileSize)&&
MathEngine.SphericalDistance(CenterLatitude, CenterLongitude, DrawArgs.Camera.Latitude, DrawArgs.Camera.Longitude) < Angle.FromDegrees(QuadTileSet.TileDrawSpread * tileSize * 1.25f) && DrawArgs.Camera.ViewFrustum.Intersects(BoundingBox)
)
Initialize();
}

  接着细化,对于超出可视范围的剔除:

 if (isInitialized && World.Settings.VerticalExaggeration != verticalExaggeration || m_CurrentOpacity != QuadTileSet.Opacity ||
QuadTileSet.RenderStruts != renderStruts)
{
CreateTileMesh();
} if (isInitialized)
{
if (DrawArgs.Camera.ViewRange < Angle.FromDegrees(QuadTileSet.TileDrawDistance * tileSize)
&& MathEngine.SphericalDistance(CenterLatitude, CenterLongitude,
DrawArgs.Camera.Latitude, DrawArgs.Camera.Longitude) < Angle.FromDegrees(QuadTileSet.TileDrawSpread * tileSize)
&& DrawArgs.Camera.ViewFrustum.Intersects(BoundingBox)
)
{
if (northEastChild == null || northWestChild == null || southEastChild == null || southWestChild == null)
{
ComputeChildren(drawArgs);
} if (northEastChild != null)
{
northEastChild.Update(drawArgs);
} if (northWestChild != null)
{
northWestChild.Update(drawArgs);
} if (southEastChild != null)
{
southEastChild.Update(drawArgs);
} if (southWestChild != null)
{
southWestChild.Update(drawArgs);
}
}
else
{
if (northWestChild != null)
{
northWestChild.Dispose();
northWestChild = null;
} if (northEastChild != null)
{
northEastChild.Dispose();
northEastChild = null;
} if (southEastChild != null)
{
southEastChild.Dispose();
southEastChild = null;
} if (southWestChild != null)
{
southWestChild.Dispose();
southWestChild = null;
}
}
} if (isInitialized)
{
if (DrawArgs.Camera.ViewRange / > Angle.FromDegrees(QuadTileSet.TileDrawDistance * tileSize * 1.5f)
|| MathEngine.SphericalDistance(CenterLatitude, CenterLongitude, DrawArgs.Camera.Latitude, DrawArgs.Camera.Longitude) > Angle.FromDegrees(QuadTileSet.TileDrawSpread * tileSize * 1.5f))
{
if (Level != || (Level == && !QuadTileSet.AlwaysRenderBaseTiles))
this.Dispose();
}
}

2.Globe Mapper中默认采用的方法是根据最高精度建立ww瓦片金字塔。另外一个选项就是传人Level0参数。

  WW适用SurfaceTile、QuadTile,VirtualEarth Tile,可以加载多个不同的瓦片集合QuadTileSet。而不同的QuadTileSet有不同的levelZeroTileSizeDegrees。

  BingMap的瓦片体系只是一种固定的瓦片层级。由于Bing地图是正射地图,所以根据屏幕像素可以计算比例尺。但是放在球上显示,需要经过纹理映射,世界坐标转换到屏幕坐标,因此与比例尺对应似乎存在问题。

  

3.WW中几个参数理解尝试:

  如图:AB为相机海拔高度,令AB=BC做B点切线,作CE//AB交地球与E,作EG垂直AB于G,则EG=BC=AB,若认为弧BE长度等于EG,则α的弧度值=viewRange/2。事实上α的弧度值是大于viewRange/2的。

  β为相机的FOV/2,β=22.5度。α对应弧BE,包括多少个瓦片?假设β只能看到 1个(对称分布)1/2个瓦片,则弧BE大概包括tan(PI/4)/tan(PI/8)=1/tan(PI/8)=1/0.392699=2.414个。

  所以QuadTileSet.TileDrawDistance=3.5 ,每个Level等级加载大约5或者(3+3)个瓦片。QuadTileSet.TileDrawSpread=2.9。

  初始化时候的范围α<QuadTileSet.TileDrawDistance*tileSize,QuadTileSet.TileDrawSpread*1.25=3.625。比细化的范围大,

  细化范围 2α<QuadTileSet.TileDrawDistance*tileSize,细化了(1+1)或者3个瓦片;

  超过某个范围是剔除,剔除的临界范围更大些α>QuadTileSet.TileDrawDistance*tileSize*1.5=5.25*tileSize,QuadTileSet.TileDrawSpread*1.5=4.35。

  回到原来的问题,通过高度如何确定显示范围的金字塔等级Level。首先需要明确当前显示的范围内金字塔等级有多个,我要中心的那个。

  我想大概中心瓦片大小sizeTileCenter=AB*tan(PI/8)/R,再根据levelZeroTileSizeDegrees求出Level。

--------------------------------------------------------------------------------------------------------------------------------

WW支持改变视域体,相当于改变了Frustrm。
R*sin(α)=Altitude
临界条件:ViewRange=2*α=2*3.5*TileSize;

地球半径R(m) 6378137  
赤道周长 =2*PI*R(m) 40075016.69  
     
Level 0瓦片 Tile0(度) 2.25 36
瓦片纹理大小(像素) 512 512
Level 0 Tile赤道分辨率(m) =2*PI*R/360*Tile0/512 489.196981 7827.152

  以上不知道分析的对不对!

[World Wind学习]22.相机高度和瓦片等级计算的更多相关文章

  1. [转]World Wind学习总结一

    WW的纹理,DEM数据,及LOD模型 以earth为例 1. 地形数据: 默认浏览器纹理数据存放在/Cache/Earth/Images/NASA Landsat Imagery/NLT Landsa ...

  2. 从零开始一起学习SLAM | 相机成像模型

    上一篇文章<从零开始一起学习SLAM | 为啥需要李群与李代数?>以小白和师兄的对话展开,受到了很多读者的好评.本文继续采用对话的方式来学习一下相机成像模型,这个是SLAM中极其重要的内容 ...

  3. Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第十三章:计算着色器(The Compute Shader)

    原文:Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第十三章:计算着色器(The Compute Shader) 代码工程 ...

  4. 人工智能范畴及深度学习主流框架,IBM Watson认知计算领域IntelligentBehavior介绍

    人工智能范畴及深度学习主流框架,IBM Watson认知计算领域IntelligentBehavior介绍 工业机器人,家用机器人这些只是人工智能的一个细分应用而已.图像识别,语音识别,推荐算法,NL ...

  5. Three.js学习(相机,场景,渲染,形状)

    相机分为透视相机和正交相机(还有第三人称相机不介绍). var camera = new THREE.PerspectiveCamera(75, window.innerWidth / window. ...

  6. Python学习--22 异步I/O

    在同步IO中,线程启动一个IO操作然后就立即进入等待状态,直到IO操作完成后才醒来继续执行.而异步IO方式中,线程发送一个IO请求到内核,然后继续处理其他的事情,内核完成IO请求后,将会通知线程IO操 ...

  7. swift学习 - tableView自适应高度1(xib autoLayout)

    tableView自适应高度 效果图: 源码: class ViewController: UIViewController,UITableViewDelegate,UITableViewDataSo ...

  8. python学习(22) 访问数据库

    原文链接:http://www.limerence2017.com/2018/01/11/python22/ 本文介绍python如何使用数据库方面的知识. SQLite SQLite是一种嵌入式数据 ...

  9. OpenCV学习(22) opencv中使用kmeans算法

    kmeans算法的原理参考:http://www.cnblogs.com/mikewolf2002/p/3368118.html 下面学习一下opencv中kmeans函数的使用.      首先我们 ...

随机推荐

  1. POSIX是什么?

    1.什么是POSIX? POSIX是可移植操作系统接口(Portable Operating System Interface for UNIX)的缩写,是IEEE为了在各种UNIX操作系统上运行软件 ...

  2. MBR和GPT分区表

    https://www.reneelab.com.cn/m/mbr-gpt-difference.html

  3. 第三章 Spring.Net 环境准备和搭建

    在前面一章我们介绍了依赖注入,控制反转的概念.接下来我们来真正动手搭建一下Spring.Net的环境,看一下Spring.Net 中的控制反转和依赖注入是什么样子. 3.1  Spring.Net 下 ...

  4. HttpServletResponse status对应的状态信息

    1xx - 信息提示   这些状态代码表示临时的响应.客户端在收到常规响应之前,应准备接收一个或多个 1xx 响应.    ·0 - 本地响应成功.   · 100 - Continue 初始的请求已 ...

  5. NPOI抓取WPS表格保存的EXCEL文件

    其实是可以抓取的,唯一不同就是Sheet的位置前进了一位.     var sheet1 = (HSSFSheet)hssfworkbook.GetSheetAt(1);     来自为知笔记(Wiz ...

  6. nutch爬取时Exception in thread “main” java.io.IOException: Job failed!

    用cygwin运行nutch 1.2爬取提示IOException: $ bin/nutch crawl urls -dir crawl -depth 3 -topN 10 crawl started ...

  7. C++11新特性之八——函数对象function

    详细请看<C++ Primer plus>(第六版中文版) http://www.cnblogs.com/lvpengms/archive/2011/02/21/1960078.html ...

  8. CSS3 属性组参考资料

    CSS 属性组: 动画 背景 边框和轮廓 盒(框) 颜色 内容分页媒体 定位 可伸缩框 字体 生成内容 网格 超链接 行框 列表 外边距 Marquee 多列 内边距 分页媒体 定位 打印 Ruby ...

  9. Servlet MVC 项目实战实例

    MVC的架构模式,一直是JavaEE开发中所遵循的标准,如今很多框架都已经很好的实现了MVC,像大家所熟知的Struts,SpringMVC,JSF等,但是如果没有任何框架的支持,仅仅通过JavaWe ...

  10. 【转】实现Ribbon风格的窗体

    随着office2007的兴起,微软让我们看到了Ribbon风格的窗体,现在很多软件也都开始使用Ribbon风格.那么我们如果要自己开发,应当怎么做呢?本文就是为大家解开这个疑团的. 首先,Delph ...