Winform實現Shp-柵格圖形文件的讀取與顯示(外加shp轉WKB格式存入oracle)附源碼

来源:http://www.cnblogs.com/ATtuing/archive/2016/03/26/5183096.html
-Advertisement-
Play Games

前言:上學期GIS空間資料庫課程設計時,老師讓實現Shp-柵格圖形文件的讀取與顯示,外加shp轉WKB格式存入oracle,不使用第三方類庫,花了一天時間在網上找了一些資料,實現了一個簡單的柵格圖形文件的顯示。 有時候我們知道了一個文件的存儲格式,便可以寫個程式輕輕鬆松讀取出來想要的數據,讀取的時候 ...


前言:上學期GIS空間資料庫課程設計時,老師讓實現Shp-柵格圖形文件的讀取與顯示,外加shp轉WKB格式存入oracle,不使用第三方類庫,花了一天時間在網上找了一些資料,實現了一個簡單的柵格圖形文件的顯示。

有時候我們知道了一個文件的存儲格式,便可以寫個程式輕輕鬆松讀取出來想要的數據,讀取的時候自然要用到system.IO來操作IO流。這次我演示的地圖矢量數據(實際上就是讀取一些點,線,面坐標信息)的讀取與可視化顯示。下麵開始進入正題。

0.成果演示

基本顯示

坐標讀取

1.什麼是shp文件

shape文件由ESRI開發,一個ESRI(Environmental Systems Research Institute)的shape文件包括一個主文件,一個索引文件,和一個dBASE表。其中主文件的尾碼就是.shp

2.shp文件的格式

百度文庫詳細解釋 

(共用程式中根目錄也有此文檔-shp.txt)

3.共用程式解析

(1)首先讀取的時候用什麼IO類(中國矢量shp數據在共用文件夾中有,有興趣的可以打開看一下)

BinaryReader用特定的編碼將基元數據類型讀作二進位值。(註意:文件是已流的形式進行讀取的,如果某一些位元組在你程式中沒有使用價值時,你也要完
全讀取過去,才能保證這個文件流繼續下去)
   private void ReadFile(string filePath)
        {
            using (var reader = new StreamReader(filePath))
            {
                using (var br = new BinaryReader(reader.BaseStream))
                {
                    //讀取頭文件
                    readerHead.ReadFileHead(br);
                   
                }
            }
        }
 /// <summary>
    ///     讀取shp文件頭類
    /// </summary>
    public class ReaderHead
    {
        //    文件長度    文件的實際長度    Integer    big
        public int FileLength { get; set; }
        //    版本號    1000    Integer    Little
        public int FileBanben { get; set; }
        //    幾何類型    表示這個Shapefile文件所記錄的空間數據的幾何類型    Integer    Little
        public int ShapeType { get; set; }
        //    Xmin    空間數據所占空間範圍的X方向最小值    Double    Little
        public double Xmin { get; set; }
        //      Ymax    空間數據所占空間範圍的Y方向最大值    Double    Little
        public double Ymax { get; set; }
        //      Xmax    空間數據所占空間範圍的X方向最大值    Double    Little
        public double Xmax { get; set; }
        //    Ymin    空間數據所占空間範圍的Y方向最小值    Double    Little
        public double Ymin { get; set; }
        //    Zmin    空間數據所占空間範圍的X方向最小值    Double    Little
        public double Zmin { get; set; }
        //      Zmax    空間數據所占空間範圍的Y方向最大值    Double    Little
        public double Zmax { get; set; }
        //      Mmax    空間數據所占空間範圍的X方向最大值    Double    Little
        public double Mmax { get; set; }
        //    Mmin    空間數據所占空間範圍的Y方向最小值    Double    Little
        public double Mmin { get; set; } 
        /// <summary>
        ///     讀取文件頭信息
        /// </summary>
        /// <param name="fileStream">文件流</param>
        /// <returns></returns>
        public void ReadFileHead(BinaryReader fileStream)
        {
            //讀取文件過程
            //去除未使用的24個位元組//
            //起始位置    名稱    數值    類型    位序
            //0    File Code    9994    Integer    big
            //4    Unused    0    Integer    big
            //8    Unused    0    Integer    big
            //12    Unused    0    Integer    big
            //16    Unused    0    Integer    big
            //20    Unused    0    Integer    big

            fileStream.ReadBytes(24);
            //24    文件長度    文件的實際長度    Integer    big
            int FileLength = fileStream.ReadInt32();
            //  //28    版本號    1000    Integer    Little
            //<0代表數據長度未知
            int FileBanben = fileStream.ReadInt32();
            //32    幾何類型    表示這個Shapefile文件所記錄的空間數據的幾何類型    Integer    Little
            ShapeType = fileStream.ReadInt32();
            //36    Xmin    空間數據所占空間範圍的X方向最小值    Double    Little
            Xmin = fileStream.ReadDouble();
            //  44    Ymax    空間數據所占空間範圍的Y方向最大值    Double    Little
            Ymax = -1*fileStream.ReadDouble();
            //  52    Xmax    空間數據所占空間範圍的X方向最大值    Double    Little
            Xmax = fileStream.ReadDouble();
            //60    Ymin    空間數據所占空間範圍的Y方向最小值    Double    Little
            Ymin = -1*fileStream.ReadDouble();
          //    68*    Zmin    空間數據所占空間範圍的Z方向最小值    Double    Little
            Zmin = fileStream.ReadDouble();
            //76*    Zmax    空間數據所占空間範圍的Z方向最大值    Double    Little
            Zmax = fileStream.ReadDouble();
            //84*    Mmin    最小Measure值    Double    Little
            Mmin = fileStream.ReadDouble();
            //92*    Mmax    最大Measure值    Double    Little
            Mmax = fileStream.ReadDouble();
        }
    } 

(2)根據讀取的文件類型(點?線?面?)進行坐標信息的讀取(繼續文件流讀取),

  switch (readerHead.ShapeType)
                    {
                        case (int) EnumShapeType.Point:
                            //清除上次打開的數據
                            points.Clear();
                            while (br.PeekChar() != -1)
                            {
                                Point point = Point.ReaderPoint(br);
                                points.Add(point);
                            }
                            //將讀取的坐標信息存入本地Txt文件當中
                            WriteTxtFile.WriteTxtPoint(points, shpName);
                            break;
                        case (int) EnumShapeType.PolyLine:
                            //清除上次打開的數據
                            polylines.Clear();
                            while (br.PeekChar() != -1)
                            {
                                Polyline polyline = Polyline.ReaderPolyline(br);
                                polylines.Add(polyline);
                            }
                            //將讀取的坐標信息存入本地Txt文件當中
                            WriteTxtFile.WriteTxtPolyline(polylines, shpName);
                            break;
                        case (int) EnumShapeType.Polygon:
                            //清除上次打開的數據
                            polygons.Clear();
                            while (br.PeekChar() != -1)
                            {
                                Polygon polygon = Polygon.ReaderPolygon(br);
                                polygons.Add(polygon);
                            }
                            //將讀取的坐標信息存入本地Txt文件當中
                            WriteTxtFile.WriteTxtPolygon(polygons, shpName);
                            break;
                    }


 

以讀取線為例子(簡單來說就是把很多線分為一條一條線,再分為一個一個點存儲起來。)

 

public class Polyline//線類
    {

        public double[] Box { get;set;}
       //表示構成當前線目標的子線段的個數    
        public int NumParts{ get;set;}
       //表示構成當前線目標所包含的坐標點個數
        public int NumPoints{ get;set;}
         //文件類型
        public int ShpType { get;set;}
        public List<int> Parts{ get;set;} //在部分中第一個點的索引

        public List<Point> Points{ get;set;} //所有部分的點

        public Polyline()
        {
            Box = new double[4];
        }

        public static Polyline ReaderPolyline(BinaryReader br)
        {
            var polyline = new Polyline();

            polyline.Parts = new  List<int>();

            polyline.Points = new List<Point>();

            //實體信息的內容  記錄頭的內容包括記錄號(Record Number)和坐標記錄長度(Content Length) 兩個記錄項。
            uint RecordNum = br.ReadUInt32();
            int DataLength = br.ReadInt32();
            //讀取第i個記錄
            //幾何類型
          polyline.ShpType=  br.ReadInt32();
            //坐標範圍(Box)    表示當前線目標的坐標範圍    double型    32     4    Little
          for (int i = 0; i < 4; i++)
          {
              polyline.Box[i] = br.ReadDouble();
          }
            //子線段個數(NumParts)    表示構成當前線目標的子線段的個數    int型    4    1    Little
            polyline.NumParts = br.ReadInt32();
            //坐標點數(NumPoints)    表示構成當前線目標所包含的坐標點個數    int型    4    1    Little
            polyline.NumPoints = br.ReadInt32();
            // Parts數組    記錄了每個子線段的坐標在Points數組中的起始位置    int型    4×NumParts    NumParts    Little
            for (int i = 0; i < polyline.NumParts; i++)
            {
                var parts = new int();

                parts = br.ReadInt32();

                polyline.Parts.Add(parts);
            }
            // Points數組    記錄了所有的坐標信息    Point型    根據點個數來確定    NumPoints    Little
            for (int j = 0; j < polyline.NumPoints; j++)
            {
                var pointtemp = new Point();

                pointtemp.X = br.ReadDouble();

                pointtemp.Y = -1*br.ReadDouble();

                polyline.Points.Add(pointtemp);
            }
            return polyline;
        }
    }

 

(3)文件的可視化顯示

要在窗體中繪製顯示,使用using System.Drawing既GDI+繪圖。繪圖時的關鍵要找到可視範圍的最大矩形,把所有數據都繪製到窗體中,我們在文件頭信息

中可以看到這些信息,     Xmin Ymax Xmax Ymin ,這樣我可計算出可視範圍。

 

     double width = readerHead.Xmax - readerHead.Xmin;
            double height = readerHead.Ymax - readerHead.Ymin;
            showShp.ZooX = (float)(panel1.Width * 0.9 / width); //x軸放大倍數
            showShp.ZooY = (float)(panel1.Height * 0.9 / height); //y軸放大倍數

 

在panel1_Paint中繪製圖形

  private void panel1_Paint(object sender, PaintEventArgs e)
        {
            //顯示當前文件
            e.Graphics.DrawString("當前文件:" + shpName, Font, Brushes.Blue, new PointF(350, 0));


            switch (readerHead.ShapeType)
            {
                case (int) EnumShapeType.Point: //點類型

                    foreach (Point p in points)
                    {
                        //轉化為float類型
                        PointF pp = ShowShp.ToPointF(p, readerHead.Xmin, readerHead.Ymin, showShp.ZooX, showShp.ZooY);
                        e.Graphics.DrawEllipse(pen, pp.X, pp.Y, 1.5f, 1.5f);
                    }
                    break;
                case (int) EnumShapeType.PolyLine: //線類型
                    foreach (Polyline p in polylines)
                    {
                        //用於產生線
                        PointF[] pointFsx = ShowShp.ToPointFs(p, readerHead.Xmin, readerHead.Ymin, showShp.ZooX,
                            showShp.ZooY);
                        e.Graphics.DrawLines(pen, pointFsx);
                    }
                    break;
                     
                case (int) EnumShapeType.Polygon: //面類型
                    //用於產生線
                    foreach (Polyline p in polygons)
                    {
                        PointF[] pointFsm = ShowShp.ToPointFs(p, readerHead.Xmin, readerHead.Ymin, showShp.ZooX,
                            showShp.ZooY);
                        e.Graphics.DrawLines(pen, pointFsm);
                        e.Graphics.FillPolygon(Brushes.Pink, pointFsm);
                    }
                    break;
            }
        }

(4)讀取坐標信息

這個比較簡單,在文件讀取的時候將點數據存儲進去,這裡不多說,有興趣的可以看源碼,我們來看一下坐標結構,

獨立的點坐標:

517336.983662845,  2860942.68850762
517445.323959293,  2860897.87513047
線坐標:
線1:
    -431977923441.66,197606922425.44,0
    -3676407859.07796,46874200203.2439,0
線2:
    -295950832655.775,-179224883130.05,0
    112130439701.878,-50550608062.3219,0
線3:
    -338229523035.172,-919101964.769489,0
    294112628726.236,13786529471.5423,0
面坐標(實際上是首尾相連的面):

多邊形2:
    517494.301149368,2861056.80072212,0
    517492.347002029,2861056.61300087,0
    517491.879999161,2861058.99200249,0
    517493.744001389,2861059.31100273,0
    517494.301149368,2861056.80072212,0
多邊形3:
    517496.790002823,2861099.2650013,0
    517495.791894913,2861099.20348549,0
    517495.546682358,2861103.18202782,0
    517496.544790268,2861103.24354362,0
    517496.790002823,2861099.2650013,0
多邊形4:
    517512.920000076,2861099.54700279,0
    517499.350999832,2861097.9810009,0
    517499.047838211,2861100.16797829,0
    517512.573999405,2861102.04300117,0
    517512.920000076,2861099.54700279,0

 

 (5)源碼解析

結構應該還是挺明顯的,就不多解釋了,shp轉WKB(OGC well-known binary)格式存入oracleoracle連接配置文件存放在App.config下麵,關於shp轉WKB格式存入oracle中就不多解釋是了,比較專業,如果有學GIS的同學可以參考一下。

有不懂得可以給我留言,或者有什麼錯誤請指出,謝謝!!!

源碼地址:http://pan.baidu.com/s/1boInPIR(含中國矢量地圖數據)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 01.什麼是三層架構 01.表示層(User Interface layer):負責接收用戶的輸入,將輸出呈現給用戶,以及訪問安全性驗證,並對輸入的數據的正確性,有效性及呈現樣式負責,但對輸出的數據的正確性不負責。 02.業務邏輯層(Business Logic Layer):負責系統領域業務的處理 ...
  • 一、預編譯的優點 1. 由於頁和代碼文件在第一次被請求時無需編譯,因此可以縮短對用戶的響應時間。這對於更新頻繁的大型網站尤為有用 2. 可以在用戶看到網站之前識別編譯時的 Bug 3. 可以創建站點的已編譯版本,並將該版本部署到成品伺服器,而無需使用源代碼 二、就地預編譯與針對部署的預編譯 1. 就 ...
  • 問題背景:如果使用plsql等工具可以連接Oralce說明與Oralce安裝無關。 VS連接資料庫時提示“ORA-12154: TNS: 無法解析指定的連接標識符”的解決方法 解決步驟: 1、去網上下載“instantclient-basic-win32-11.2.0.1.0.zip”,名稱為ins ...
  • 其實很早之前我就已經瞭解了在winform下實現插件編程,原理很簡單,主要實現思路就是:先定一個插件介面作為插件樣式及功能的約定,然後具體的插件就去實現這個插件介面,最後宿主(應用程式本身)就利用反射動態獲取實現了插件介面的類型作為合法的插件,從而完成動態載入及宿主與插件之間的互動。因為之前一段時間 ...
  • Windows是一個強大的操作系統,也會向開發者提供海量的系統API來幫助開發者來完成Windows系統軟體的開發工作。 整理的部分Windows API,C#可以直接調用。 1.獲取.exe應用程式的圖標 2.獲取硬碟信息 3.開機啟動程式 4.系統熱鍵操作 5.系統進程操作 6.改變視窗 ...
  • 1、什麼是類? 面向對象的語言,最基本的就是類。在C#中,類是這樣來定義的:類代表一組具有公共屬性和行為的對象。 舉個例子,現實生活中,人就是一個“類”,但這隻是一個統稱,指所有的人。我們要找個人一起去玩,此時就需要找到一個具體的人。比如我和小紅,我們兩個就是“人”這個類的兩個實例。 2、C#中如何 ...
  • 總結C#獲取當前路徑的7種方法 C#獲取當前路徑的方法如下: 1. System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName -獲取模塊的完整路徑。 2. System.Environment.CurrentDirector ...
  • Html.ActionLink 在 LinkExtensions 類中,ActionLink方法參數說明: 簡單來說捏 就是這樣滴 參數 類型 說明 linkText string 生成的鏈接所顯示的文字 actionName string 對應控制器的方法 controllerName strin ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...