- 浏览: 7256598 次
- 性别:
- 来自: 上海
文章分类
- 全部博客 (1546)
- 企业中间件 (236)
- 企业应用面临的问题 (236)
- 小布Oracle学习笔记汇总 (36)
- Spring 开发应用 (54)
- IBatis开发应用 (16)
- Oracle基础学习 (23)
- struts2.0 (41)
- JVM&ClassLoader&GC (16)
- JQuery的开发应用 (17)
- WebService的开发应用 (21)
- Java&Socket (44)
- 开源组件的应用 (254)
- 常用Javascript的开发应用 (28)
- J2EE开发技术指南 (163)
- EJB3开发应用 (11)
- GIS&Mobile&MAP (36)
- SWT-GEF-RCP (52)
- 算法&数据结构 (6)
- Apache开源组件研究 (62)
- Hibernate 学习应用 (57)
- java并发编程 (59)
- MySQL&Mongodb&MS/SQL (15)
- Oracle数据库实验室 (55)
- 搜索引擎的开发应用 (34)
- 软件工程师笔试经典 (14)
- 其他杂项 (10)
- AndroidPn& MQTT&C2DM&推技术 (29)
- ActiveMQ学习和研究 (38)
- Google技术应用开发和API分析 (11)
- flex的学习总结 (59)
- 项目中一点总结 (20)
- java疑惑 java面向对象编程 (28)
- Android 开发学习 (133)
- linux和UNIX的总结 (37)
- Titanium学习总结 (20)
- JQueryMobile学习总结 (34)
- Phonegap学习总结 (32)
- HTML5学习总结 (41)
- JeeCMS研究和理解分析 (9)
最新评论
-
lgh1992314:
[u][i][b][flash=200,200][url][i ...
看看mybatis 源代码 -
尼古拉斯.fwp:
图片根本就不出来好吧。。。。。。
Android文件图片上传的详细讲解(一)HTTP multipart/form-data 上传报文格式实现手机端上传 -
ln94223:
第一个应该用排它网关吧 怎么是并行网关, 并行网关是所有exe ...
工作流Activiti的学习总结(八)Activiti自动执行的应用 -
ZY199266:
获取不到任何消息信息,请问这是什么原因呢?
ActiveMQ 通过JMX监控Connection,Queue,Topic的信息 -
xiaoyao霄:
DestinationSourceMonitor 报错 应该导 ...
ActiveMQ 通过JMX监控Connection,Queue,Topic的信息
转载自 http://blog.tigerlihao.cn/2010/01/geotools-based-web-map-service.html
暑假看OGC标准的时候做了一个简单的WMS(Web Map Service),用的是GeoTools工具包。其实做出来用处也不大,应为已经有GeoServer这个项目在做基于GeoTool的网络GIS应用,并且已经做的比较完善了。我这个纯粹是做着玩,顺便学习Java网络编程和GeoTools的。
OGC的WMS标准我就不多说了,可以直接去看标准文档。GeoTools搞开源GIS开发的应该也不陌生,是一个用Java语言编写的遵循OGC规范的开源GIS工具包,其功能涵盖了地理信息数据读写、处理、坐标转换、查询分析、格式化输出等多个方面,详细的情况请访问GeoTools的主页:www.geotools.org。下面主要介绍一下我的设计方案。
WMS服务器的整体架构
WMS服务器的整体架构主要包括:请求分发模块、数据读取模块、样式设置模块、图层加载模块、地图绘制模块属性查询模块。
首先请求分发模块根据客户端的请求参数判断出操作的种类,并分别调用相应的模块。数据读取模块负责加载地图数据文件。样式设置模块负责获取图层的渲染样式。图层加载模块负责将各个数据集和样式对应并按顺序排列,生成地图对象。地图绘制模块负责将地图对象渲染成为图像。查询模块则根据位置返回指定要素的属性信息。最终将具体操作的结果返回给客户端。
请求分发模块主要实现对用户请求参数的解析。在Servlet中一般通过Request对象的getParameter方法获取客户端的请求参数。在请求的解析中需要首先判断客户端要执行的是GetCapabilities、GetMap、GetFeatureInfo中的哪一种操作,然后根据每种操作的参数列表读取参数值,并对参数的有效性进行检验。对于非必须的请求参数需要设置缺省值。最后生成请求参数对象,传递给各个操作的具体实现方法。对于不符合要求的请求参数,需要返回给客户端错误信息,并停止后续的操作,以避免运行错误。这部分的代码如下:
public void doService(HttpServletRequest request,
HttpServletResponse response) throws IOException {
Map map = request.getParameterMap();
Map param = new HashMap();
for (String k : map.keySet()) {
String s1 = "";
if (param.containsKey(k.toUpperCase())) {
s1 = param.get(k.toUpperCase()) + ",";
}
String[] s2 = (String[]) map.get(k);
for (int i = 0; i < s2.length; i++) {
s1 += s2[i] + (i == 0 ? "" : ",");
}
param.put(k.toUpperCase(), s1);
}
if (!param.containsKey("REQUEST")) {
WMSException.exception(response);
} else {
String wmsRequest = param.get("REQUEST");
if (wmsRequest.equals("GetCapabilities")) {
GetCapabilitiesRequest gcr = new GetCapabilitiesRequest(param);
doGetCapabilities(gcr, response);
} else if (wmsRequest.equals("GetMap")) {
GetMapRequest gmr = new GetMapRequest(param);
doGetMap(gmr, response);
} else if (wmsRequest.equals("GetFeatureInfo")) {
GetFeatureInfoRequest gfr = new GetFeatureInfoRequest(param);
doGetFeatureInfo(gfr, response);
} else {
WMSException.exception(response);
}
}
}
数据读取模块主要是利用GeoTools提供的Shape file reader模块来读取指定位置的shp格式地图文件,并创建所需要的Data Store对象。地图样式是用SLD文件定义的。SLD是OGC制定的用于描述图层样式的XML文件格式,通过制定一系列的样式规则对指定的要素类型进行样式化,包括设置显示的符号、颜色、填充样式、线条样式、标注等。GeoTools中图层的管理是通过MapContext对象来实现。调用MapContext对象的addLayer方法,可将图层按照从最底层开始依次加载到地图中。这部分代码如下:
private static void addShapeLayer(String name) throws Exception {
File file = new File("C:\\data\\" + name + ".shp");
File sldFile = new File("C:\\data\\" + name + ".sld");
FileDataStore store = FileDataStoreFinder.getDataStore(file);
((ShapefileDataStore) store).setStringCharset(Charset.forName("GB2312"));
FeatureSource featureSource = store
.getFeatureSource();
Configuration config = new SLDConfiguration();
Parser parser = new Parser(config);
InputStream sld = new FileInputStream(sldFile);
StyledLayerDescriptor styleSLD = (StyledLayerDescriptor) parser.parse(sld);
Style style = SLD.defaultStyle(styleSLD);
map.addLayer(featureSource, style);
}
WMS服务器的GetMap操作需要根据客户端的请求将地图对象的指定区域进行渲染,并返回图像文件。首先需要根据用户的请求参数生成一个ReferencedEnvelope对象作为地图输出的范围,然后使用StreamingRenderer对象进行渲染,并将输出绘制在用户指定的大小和格式的图像文件中。最后将图像以二进制编码的形式通过Response对象返回给用户。这部分的代码如下:
private void doGetMap(GetMapRequest gmr, HttpServletResponse response)
throws IOException {
double x1, y1, x2, y2;
int width, height;
try {
x1 = Double.parseDouble(gmr.getBBOX()[0]);
y1 = Double.parseDouble(gmr.getBBOX()[1]);
x2 = Double.parseDouble(gmr.getBBOX()[2]);
y2 = Double.parseDouble(gmr.getBBOX()[3]);
width = Integer.parseInt(gmr.getWidth());
height = Integer.parseInt(gmr.getHeight());
} catch (Exception e) {
WMSException.exception(response);
return;
}
// 设置输出范围
ReferencedEnvelope mapArea = new ReferencedEnvelope(x1, x2, y1, y2, crs);
// 初始化渲染器
StreamingRenderer sr = new StreamingRenderer();
sr.setContext(map);
// 初始化输出图像
BufferedImage bi = new BufferedImage(width, height,
BufferedImage.TYPE_INT_ARGB);
Graphics g = bi.getGraphics();
((Graphics2D) g).setRenderingHint(RenderingHints.KEY_ANTIALIASING,
RenderingHints.VALUE_ANTIALIAS_ON);
((Graphics2D) g).setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
Rectangle rect = new Rectangle(0, 0, width, height);
// 绘制地图
sr.paint((Graphics2D) g, rect, mapArea);
// 编码图像
PNGEncodeParam encodeParam = PNGEncodeParam.getDefaultEncodeParam(bi);
if (encodeParam instanceof PNGEncodeParam.Palette) {
PNGEncodeParam.Palette p = (PNGEncodeParam.Palette) encodeParam;
byte[] b = new byte[] { -127 };
p.setPaletteTransparency(b);
}
//将图像数据输出到Servlet相应中
response.setContentType("image/png");
ServletOutputStream out = response.getOutputStream();
com.sun.media.jai.codec.ImageEncoder encoder = ImageCodec
.createImageEncoder("PNG", out, encodeParam);
encoder.encode(bi.getData(), bi.getColorModel());
bi.flush();
}
WMS服务器的GetFeatureInfo操作需要通过位置对要素集进行查询,返回指定要素的属性信息。由于用户所给的查询坐标是图像的像素坐标,因此需要将坐标转换地图要素所使用的实际坐标。知道了实际坐标,就可以创建一个Filter对象来描述查询的约束条件。然后调用要素集的查询方法,就可以获取符合要求的要素子集,最后按一定的格式返回各个要素的属性信息给客户端。这部分的代码如下:
private void doGetFeatureInfo(GetFeatureInfoRequest gfr,
HttpServletResponse response) throws IOException {
double x1, y1, x2, y2;
int width, height, i, j;
try {
x1 = Double.parseDouble(gfr.getBBOX()[0]);
y1 = Double.parseDouble(gfr.getBBOX()[1]);
x2 = Double.parseDouble(gfr.getBBOX()[2]);
y2 = Double.parseDouble(gfr.getBBOX()[3]);
width = Integer.parseInt(gfr.getWidth());
height = Integer.parseInt(gfr.getHeight());
i = Integer.parseInt(gfr.getI());
j = Integer.parseInt(gfr.getJ());
} catch (Exception e) {
WMSException.exception(response);
return;
}
// 计算点选范围的地图坐标
double cx1, cy1, cx2, cy2;
cx1 = x1 * (width - i + 0.5 + GET_FEATURE_INFO_BUFFUR) / width + x2
* (i - 0.5 - GET_FEATURE_INFO_BUFFUR) / width;
cx2 = x1 * (width - i + 0.5 - GET_FEATURE_INFO_BUFFUR) / width + x2
* (i - 0.5 + GET_FEATURE_INFO_BUFFUR) / width;
cy1 = y1 * (j - 0.5 + GET_FEATURE_INFO_BUFFUR) / height + y2
* (height - j + 0.5 - GET_FEATURE_INFO_BUFFUR) / height;
cy2 = y1 * (j - 0.5 - GET_FEATURE_INFO_BUFFUR) / height + y2
* (height - j + 0.5 + GET_FEATURE_INFO_BUFFUR) / height;
ReferencedEnvelope clickArea = new ReferencedEnvelope(cx1, cx2, cy1, cy2, crs);
MapLayer[] maplayers = map.getLayers();
FilterFactory2 ff = CommonFactoryFinder.getFilterFactory2(null);
response.setContentType("text/html");
response.setCharacterEncoding("GBK");
PrintWriter out = response.getWriter();
out.println("location: " + ((cx1 + cx2) / 2.0) + ", "
+ ((cy1 + cy2) / 2.0) + "<br/>");
// 分别在每个图层中查找点选范围内的对象
for (int k = 0; k < maplayers.length; k++) {
FeatureSource fs =
(FeatureSource) maplayers[k].getFeatureSource();
String geometryPropertyName = fs.getSchema().getGeometryDescriptor().getLocalName();
Filter filter = ff.bbox(ff.property(geometryPropertyName), clickArea);
FeatureCollection fc = fs.getFeatures(filter);
SimpleFeatureType schema = fc.getSchema();
FeatureIterator fi = fc.features();
if (fi.hasNext()) {
out.println("Selected feature(s) in layer ["+schema.getTypeName()+"]:<br/>");
while (fi.hasNext()) {
SimpleFeature f = fi.next();
out.println("id:" + f.getID() + "<br/>");
for (AttributeDescriptor type : schema
.getAttributeDescriptors()) {
String name = type.getLocalName();
if (!name.equals(geometryPropertyName))
out.println(name + ":"
+ f.getProperty(name).getValue().toString()
+ "<br/>");
}
out.println("<br/>");
}
}
}
out.flush();
out.close();
}
最后的效果:
当时还准备做一个WFS的实例,做了一半,有其他的事情就放下了,以后有时间再做。
评论
发表评论
-
TestNG简单的学习(十三)TestNG中Junit的实现
2013-12-04 09:00 3291TestNG和junit的整合 ... -
TestNG简单的学习(十二)TestNG运行
2013-12-03 09:08 51456文档来自官方地址: ... -
TestNG简单的学习(十一)TestNG学习总结
2013-12-03 09:08 14000最近一直在学习关于TestNG方面的知识,根 ... -
TestNG简单的学习(十)TestNG @Listeners 的使用
2013-12-03 09:07 8630TestNG官方网站: http://testng.or ... -
TestNG简单的学习(九)TestNG Method Interceptors 的使用
2013-12-03 09:07 2662TestNG官方网站: http://testng ... -
TestNG简单的学习(八)TestNG Annotation Transformers 的使用
2013-12-03 09:07 2762TestNG官方网站: http://testng.or ... -
TestNG简单的学习(七)TestNG编程方式运行
2013-12-02 09:22 2401TestNG官方网站: http://testng.or ... -
TestNG简单的学习(六)测试工厂注释的使用
2013-12-02 09:22 2719TestNG官方网站: http://testng.or ... -
TestNG简单的学习(五)参数化测试数据的定制
2013-12-02 09:22 2647TestNG官方网站: http://testng.or ... -
TestNG简单的学习(四)测试方法通过名称名称依赖实现
2013-12-02 09:21 2036TestNG官方网站: http://testng.or ... -
TestNG简单的学习(三)测试方法通过测试分组依赖实现
2013-12-02 09:21 2771TestNG官方网站: http://testng.or ... -
TestNG简单的学习(二)参数化测试并发且多方法测试方法判定
2013-11-29 15:35 3638TestNG官方网站: http://testng.or ... -
TestNG简单的学习(一)类和方法级别@Test的区别
2013-11-29 15:31 9383TestNG官方文档的地址: http://testng ... -
Feed4Junit的简单使用(七)Feed4TestNg
2013-11-29 13:35 6084在Feed4Junit主要针对junit实现的 ... -
Feed4Junit的简单使用(六)数据来特定格式文件
2013-11-29 12:29 2691Feed4Junit官方地址: http://da ... -
Feed4Junit的简单使用(五)数据来自动态约束数据
2013-11-29 12:29 2567Feed4Junit官方地址: http://datab ... -
Feed4Junit的简单使用(四)数据来自定义数据源
2013-11-28 14:09 3045Feed4Junit官方地址: http://databe ... -
Feed4Junit的简单使用(三)数据源来自数据库
2013-11-28 13:58 3101Feed4Junit官方地址: http://databe ... -
Feed4Junit的简单使用(二)数据源来自文件
2013-11-28 13:50 4522Feed4Junit官方地址: http://datab ... -
Feed4Junit的简单使用(一)
2013-11-28 13:47 2161Feed4Junit官方地址: http://databe ...
相关推荐
geotools对wms的实现支持,给出了geotools如何开发一个简单的wms实现,对理解geoserver源码有很大的帮助作用。
org.geotools org.geotools.arcsde org.geotools.arcsde.data org.geotools.arcsde.data.versioning org.geotools.arcsde.data.view org.geotools.arcsde.filter org.geotools.arcsde.gce org.geotools....
springboot项目-基于geotools的倾斜摄影大地坐标转自定义投影坐标源码.zipspringboot项目-基于geotools的倾斜摄影大地坐标转自定义投影坐标源码.zipspringboot项目-基于geotools的倾斜摄影大地坐标转自定义投影坐标...
基于GeoTools的公交监控系统设计与实现,李萍,周东清,根据国内外公共交通管理的实际情况及GPS技术的发展现状,结合GPS在智能公交系统中的应用背景,本文设计了一种基于GPS,GPRS和GIS技术��
org.geotools org.geotools.arcsde org.geotools.arcsde.data org.geotools.arcsde.data.view org.geotools.arcsde.filter org.geotools.arcsde.jndi org.geotools.arcsde.logging org.geotools.arcsde....
为了在android下开发GIS.本文将GEOtools Googlemap ucmap 三种地图引擎各种优缺点作了比较,并实际代码测试过三种引擎,方便大家引擎选择.
geotools学习(一)IntelliJ IDEA搭建快速入门示例.pdf
import org.geotools.data.FileDataStore; import org.geotools.data.FileDataStoreFinder; import org.geotools.data.simple.SimpleFeatureSource; import org.geotools.map.FeatureLayer; import org.geotools.map...
geotools汉语版资料
1.java实现shapefile几何数据批量读取,并封装为通用的工具类,实现shp数据的...3.1.本工具类基于geotools开发,开发使用geotools版本为19.2,几何JTS拓扑套件版本为jts-1.13 4.使用前请自行配置好geotools及jts的jar包
GeoTools是一个开源(LGPL)Java代码库,它为操作地理空间数据提供符合标准的方法,例如实现地理信息系统(GIS)。GeoTools库在开发时实现了开放地理空间联盟(OGC)规范。本文档为GeoTools 21.2 api文档
geotools中文资料,我学习时候,整理的资料, 另外我的新浪博客,有关于gis 的flex -js-android-ios资料 欢迎下载,大家一起学习,我建了个gis群 291301205 新浪 http://blog.sina.com.cn/skywalkershaka 因为我下...
geotools整理的学习资料,详细的详述了geotools各个api,包括geoapi和jts。初学者的入门指南
geotools依赖包
英文版本,不是中文版本的 GeoTools开发的工具
geotools中文资料,我学习时候,整理的资料,
geotools 判断几何要素的交点 当时想到用的GDAL 但是 交点函数返回的对象总是null , 改用 GeoTools 这个库,需要用到jar 到官网上下载,主要是jts-core-1.16.0.jar
等值线等值面功能实现.使用技术包括Java+Geotools+WContour+Openlayers.包括整套开发示例代码及其相关jar包,下载即可使用。
支持对shp文件的读写操作,以便远程通过java代码实现对ArcGIS服务或图层进行操作,同时简单易用