這篇文章主要介紹了Java編程如何實(shí)現(xiàn)軌跡壓縮算法開放窗口,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
成都創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于做網(wǎng)站、網(wǎng)站制作、秦淮網(wǎng)絡(luò)推廣、小程序定制開發(fā)、秦淮網(wǎng)絡(luò)營銷、秦淮企業(yè)策劃、秦淮品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運(yùn)營等,從售前售中售后,我們都將竭誠為您服務(wù),您的肯定,是我們最大的嘉獎;成都創(chuàng)新互聯(lián)為所有大學(xué)生創(chuàng)業(yè)者提供秦淮建站搭建服務(wù),24小時(shí)服務(wù)熱線:028-86922220,官方網(wǎng)址:vcdvsql.cn
軌跡壓縮算法
場景描述
給定一個(gè)GPS數(shù)據(jù)記錄文件,每條記錄包含經(jīng)度和維度兩個(gè)坐標(biāo)字段,根據(jù)距離閾值壓縮記錄,將過濾后的所有記錄的經(jīng)緯度坐標(biāo)構(gòu)成一條軌跡
算法描述
這種算法的用處還是相當(dāng)廣泛的。
軌跡壓縮算法分為兩大類,分別是無損壓縮和有損壓縮,無損壓縮算法主要包括哈夫曼編碼,有損壓縮算法又分為批處理方式和在線數(shù)據(jù)壓縮方式,其中批處理方式又包括DP(Douglas-Peucker)算法、TD-TR(Top-Down Time-Ratio)算法和Bellman算法,在線數(shù)據(jù)壓縮方式又包括滑動窗口、開放窗口、基于安全區(qū)域的方法等。
大家也可參考這篇文章:《Java編程實(shí)現(xiàn)軌跡壓縮之Douglas-Peucker算法詳細(xì)代碼》
代碼實(shí)現(xiàn)
import java.awt.Color; import java.awt.Graphics; import java.awt.Point; import java.awt.Toolkit; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.InputStreamReader; import java.io.RandomAccessFile; import java.text.DecimalFormat; import java.util.ArrayList; import java.util.Iterator; import javax.swing.JFrame; import javax.swing.JPanel; public class TrajectoryCom { public static void main(String[] args) throws Exception{ //閾值定義 double maxDistanceError = 30; /* * 文件讀取 * */ //存放從文件讀取的位置點(diǎn)的信息列表 ArrayList<enpoint> ENPList = new ArrayList<enpoint>(); //源數(shù)據(jù)文件的地址 建立文件對象 //這里是需要更改的地方 改你源文件的存放地址 記住如果地址中含"\",記得再加一個(gè)"\",原因"\"是轉(zhuǎn)義符號 //這里可以寫成C:/Users/Administrator/Desktop/11.6/2007-10-14-GPS.log File sourceFile = new File("./2007-10-14-GPS.log"); //調(diào)用文件讀取函數(shù) 讀取文件數(shù)據(jù) ENPList = getENPointFromFile(sourceFile); //這里是測試 有沒有讀到里面 看看列表里的數(shù)據(jù)個(gè)數(shù) 交作業(yè)的時(shí)候記得注釋掉 System.out.println(ENPList.size()); /* * 數(shù)據(jù)處理 * 方法:開放窗口軌跡壓縮法 * */ //存放目標(biāo)點(diǎn)的集合 ArrayList<enpoint> rePointList = new ArrayList<enpoint>(); rePointList = openWindowTra(ENPList,maxDistanceError); System.out.println(rePointList.size()); /* * 寫入目標(biāo)文件 * */ File targetFile = new File("./2007-10-14-GPSResult.log"); writeTestPointToFile(targetFile,rePointList); /* * 壓縮率計(jì)算 */ double cpL = (double)rePointList.size() / (double)ENPList.size() * 100; DecimalFormat df = new DecimalFormat("0.000000"); System.out.println("壓縮率:"+ df.format(cpL) + "%"); /* * 計(jì)算平均距離誤差 * */ double aveDisErr = getMeanDistError(ENPList,rePointList); System.out.println(aveDisErr); /* * 畫線形成對比圖 * */ //generateImage(ENPList,rePointList); } /* * 從提供的文件信息里提取位置點(diǎn) * 并將每個(gè)點(diǎn)的坐標(biāo)數(shù)值調(diào)用轉(zhuǎn)換函數(shù)存到列表里 * 函數(shù)返回一個(gè) 存放所有位置點(diǎn) 的集合 */ public static ArrayList<enpoint> getENPointFromFile(File fGPS)throws Exception{ ArrayList<enpoint> pGPSArray = new ArrayList<enpoint>(); if(fGPS.exists()&&fGPS.isFile()){ InputStreamReader read = new InputStreamReader(new FileInputStream(fGPS)); //輸入流初始化 BufferedReader bReader = new BufferedReader(read); //緩存讀取初始化 String str; String[] strGPS; int i = 0; while((str = bReader.readLine())!=null){ //每次讀一行 strGPS = str.split(" "); ENPoint p = new ENPoint(); p.id = i; i++; p.pe = (dfTodu(strGPS[3])); p.pn = (dfTodu(strGPS[5])); pGPSArray.add(p); } bReader.close(); } return pGPSArray; } /** * 函數(shù)功能:將原始經(jīng)緯度坐標(biāo)數(shù)據(jù)轉(zhuǎn)換成度 * 獲取的經(jīng)緯度數(shù)據(jù)為一個(gè)字符串 */ public static double dfTodu(String str){ int indexD = str.indexOf('.'); //獲取 . 字符所在的位置 String strM = str.substring(0,indexD-2); //整數(shù)部分 String strN = str.substring(indexD-2); //小數(shù)部分 double d = double.parsedouble(strM)+double.parsedouble(strN)/60; return d; } /* * 開放窗口方法實(shí)現(xiàn) * 返回一個(gè)壓縮后的位置列表 * 列表每條數(shù)據(jù)存放ID、點(diǎn)的坐標(biāo) * * 算法描述: * 初始點(diǎn)和浮動點(diǎn)計(jì)算出投影點(diǎn),判斷投影點(diǎn)和軌跡點(diǎn)的距離與閾值 若存在距離大于閾值 * 則初始點(diǎn)放入targetList,浮動點(diǎn)向前檢索一點(diǎn)作為新的初始點(diǎn),新的初始點(diǎn)向后檢索第二個(gè)作為新的浮動點(diǎn) 這里存在判斷 即新的初始點(diǎn)位置+1是不是等于列表長度 這里決定了浮動點(diǎn)的選取 * 如此處理至終點(diǎn) * */ public static ArrayList<enpoint> openWindowTra(ArrayList<enpoint> sourceList,double maxDis){ ArrayList<enpoint> targetList = new ArrayList<enpoint>(); //定義初始點(diǎn)位置 最開始初始點(diǎn)位置為0 int startPoint = 0; //定義浮動點(diǎn)位置 最開始初始點(diǎn)位置2 int floatPoint = 2; //定義當(dāng)前軌跡點(diǎn)位置 最開始初始點(diǎn)位置為1 int nowPoint = 1; int len = sourceList.size(); //存放所有窗口內(nèi)的點(diǎn)的信息集合 ArrayList<enpoint> listPoint = new ArrayList<enpoint>(); listPoint.add(sourceList.get(nowPoint)); //浮動點(diǎn)位置決定循環(huán) while(true){ //標(biāo)志 用來控制判斷是否進(jìn)行窗口內(nèi)軌跡點(diǎn)更新 Boolean flag = false; //計(jì)算并判斷窗口內(nèi)所有點(diǎn)和投影點(diǎn)的距離是否大于閾值 for (ENPoint point:listPoint){ double disOfTwo = getDistance(sourceList.get(startPoint),sourceList.get(floatPoint),point); if(disOfTwo >= 30){ flag = true; break; } } if(flag){ //窗口內(nèi)點(diǎn)距離都大于閾值 //初始點(diǎn)加到目標(biāo)列表 targetList.add(sourceList.get(startPoint)); //初始點(diǎn)變化 startPoint = floatPoint - 1; //浮動點(diǎn)變化 floatPoint += 1; if(floatPoint >= len){ targetList.add(sourceList.get(floatPoint-1)); break; } //窗口內(nèi)點(diǎn)變化 listPoint.clear(); //System.out.println(listPoint.size()); listPoint.add(sourceList.get(startPoint+1)); } else{ //距離小于閾值的情況 //初始點(diǎn)不變 //當(dāng)前窗口集合加入當(dāng)前浮動點(diǎn) listPoint.add(sourceList.get(floatPoint)); //浮動點(diǎn)后移一位 floatPoint += 1; //如果浮動點(diǎn)是終點(diǎn) 且當(dāng)前窗口點(diǎn)距離都小于閾值 就直接忽略窗口點(diǎn) 直接將終點(diǎn)加入目標(biāo)點(diǎn)集合 if(floatPoint >= len){ targetList.add(sourceList.get(startPoint)); targetList.add(sourceList.get(floatPoint-1)); break; } } flag = false; } return targetList; } /*計(jì)算投影點(diǎn)到軌跡點(diǎn)的距離 * 入口是初始點(diǎn)A、浮動點(diǎn)B、當(dāng)前軌跡點(diǎn)C * 三角形面積公式 */ public static double getDistance(ENPoint A,ENPoint B,ENPoint C){ double distance = 0; double a = Math.abs(geoDist(A,B)); double b = Math.abs(geoDist(B,C)); double c = Math.abs(geoDist(A,C)); double p = (a + b + c)/2.0; double s = Math.sqrt(p * (p-a) * (p-b) * (p-c)); distance = s * 2.0 / a; return distance; } /* * ArrayList 拷貝函數(shù) * */ /*提供的函數(shù) * 其中計(jì)算距離的函數(shù) 經(jīng)過改造得到下面的距離計(jì)算方法 * 具體是怎么計(jì)算距離的 我也沒研究了 * */ public static double geoDist(ENPoint pA,ENPoint pB){ double radLat1 = Rad(pA.pn); double radLat2 = Rad(pB.pn); double delta_lon = Rad(pB.pe - pA.pe); double top_1 = Math.cos(radLat2) * Math.sin(delta_lon); double top_2 = Math.cos(radLat1) * Math.sin(radLat2) - Math.sin(radLat1) * Math.cos(radLat2) * Math.cos(delta_lon); double top = Math.sqrt(top_1 * top_1 + top_2 * top_2); double bottom = Math.sin(radLat1) * Math.sin(radLat2) + Math.cos(radLat1) * Math.cos(radLat2) * Math.cos(delta_lon); double delta_sigma = Math.atan2(top, bottom); double distance = delta_sigma * 6378137.0; return distance; } public static double Rad(double d){ return d * Math.PI / 180.0; } /* * 將壓縮后的位置點(diǎn)信息寫入到文件中 * */ public static void writeTestPointToFile(File outGPSFile,ArrayList<enpoint> pGPSPointFilter)throws Exception{ Iterator<enpoint> iFilter = pGPSPointFilter.iterator(); RandomAccessFile rFilter = new RandomAccessFile(outGPSFile,"rw"); while(iFilter.hasNext()){ ENPoint p = iFilter.next(); String sFilter = p.getResultString(); byte[] bFilter = sFilter.getBytes(); rFilter.write(bFilter); } rFilter.close(); } /** * 函數(shù)功能:求平均距離誤差 * 返回平均距離 */ public static double getMeanDistError(ArrayList<enpoint> pGPSArray,ArrayList<enpoint> pGPSArrayRe){ double sumDist = 0.0; for (int i=1;i<pgpsarrayre.size();i++){ double="" end="pGPSArrayRe.get(i).id;" int="" j="start+1;j<end;j++){" meandist="sumDist/(pGPSArray.size());" pre="" return="" start="pGPSArrayRe.get(i-1).id;" sumdist=""><pre class="brush:java;">import java.text.DecimalFormat; public class ENPoint implements Comparable<enpoint>{ public int id; //點(diǎn)ID public double pe; //經(jīng)度 public double pn; //維度 public ENPoint(){ } //空構(gòu)造函數(shù) public String toString(){ return this.id+"#"+this.pn+","+this.pe; } public String getResultString(){ DecimalFormat df = new DecimalFormat("0.000000"); return this.id+"#"+df.format(this.pe)+","+df.format(this.pn)+" \n"; } @Override public int compareTo(ENPoint other) { if(this.id<other.id) else="" return="" this.id="">other.id) return 1; else return 0; } }
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Java編程如何實(shí)現(xiàn)軌跡壓縮算法開放窗口”這篇文章對大家有幫助,同時(shí)也希望大家多多支持創(chuàng)新互聯(lián),關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,更多相關(guān)知識等著你來學(xué)習(xí)!
網(wǎng)頁名稱:Java編程如何實(shí)現(xiàn)軌跡壓縮算法開放窗口
轉(zhuǎn)載來源:http://vcdvsql.cn/article48/pepjhp.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供商城網(wǎng)站、App設(shè)計(jì)、品牌網(wǎng)站制作、ChatGPT、移動網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)公司
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)