用圖層服務作為GP服務引數。GP服務之引數 GPFeatureRecordSetLayer

2020-10-18 11:00:59

之前我做過一個利用mapserver提供的查詢介面,獲取json格式的要素,然後將json下載再轉換成shp,來進行各種處理的gp服務。今天我偶然發現arcgis官方也可能提供了這種模式。之所以說是可能是因為,我只是根據官方的一段介紹得到了這樣的推斷。

起因是下午快下班的時候,我偶然看到了這篇部落格。

https://my.oschina.net/u/4369360/blog/3388395

我突然發現這就是我夢寐以求的以圖層服務作為gp服務的例子。截圖來自部落格

隨後我注意到作者使用了這個引數型別 GPFeatureRecordSetLayer截圖來自部落格

GPFeatureRecordSetLayer 這個型別,我竟然完全不熟悉,我可是研究了那麼久的gp服務。好吧,我攤牌了。我一直也沒仔細看過gp服務的引數型別。我潛意識裡一直把他們都當作字串。

這裡可以說一下GP工具和GP服務,給我們的感覺可能都是相似的,只是一個執行在本地一個執行在伺服器上。但是其引數型別並不是一致的,而是對應的關係。如下圖中的後兩列。圖中REST型別就是GP服務中的引數型別。不過這幅圖還少了一列就是arcgis js 對應的型別,需要的朋友可以找一下。

在arcgis 10.*後的版本,要想釋出GP服務,使用者必須先在本機成功執行一次。而這個執行成功的gp工具中的引數型別也隨之確定。實際上我認為GP服務最需要考慮的一個因素就是輸入。如下圖,由於GP服務由前端呼叫,因此只能接受很少的引數型別,遠少於我們平時在ArcMap中操作的型別。而一旦GP服務在伺服器上執行,由於伺服器包含完整的ArcGIS環境,因此任何ArcGIS支援的引數型別都可以使用。同時由於輸出給前端,因此輸出型別一樣受限。

 

 

這裡首先要說的就是我們平常用得最多的shp格式。如果你在GP工具使用這個格式,那麼當GP工具釋出後,對應的GP服務中引數就是常數(constant)。也就是這個GP服務等於沒用,還不如釋出一個結果在到伺服器上。

這裡要指出的是constant不是資料型別,而是GP服務的輸入模式 。GP服務有三種輸入模式,如下圖所示。根據GP工具的引數型別,這個輸入模式有所不同。

鋪墊了一些知識後,我再說說題目。GPFeatureRecordSetLayer對應的GP工具型別是FeatureSet。這個型別,我之前是知道的,但是我只知道如果想讓自己的GP工具(服務)具有互動性。如使用者可以選擇地圖上一個位置,計算集水區。那這個位置,就需要設定成FeatureSet。沒錯,這個例子也是來自於ArcGIS官方的GP服務例子 watershed。同時這個引數型別也可以接受shp,layer這樣的常用引數,還是蠻好用的。

但是我在使用FeatureSet的時候,在輸入模式時使用了預設的ChoiceList,如圖。從來沒選過 User defined value。選了ChoiceList後,情況比constant好一點。可以在參數列選擇值,但是不能選擇超過這個範圍的值。如果選擇了這種輸入模式,那麼FeatureSet對應的GP服務引數型別是GPString。

我之前一直以為只有我在前臺構建一個互動式要素的時候,才需要將FearueSet的輸入模式設定為User defiend value。也就是像下圖官方介紹的。我們可以看到,圖中的例子是一個包含2個Point的json。這很符合我們在圖上點了一個點後,對於這個點的描述。

https://desktop.arcgis.com/zh-cn/arcmap/10.3/analyze/sharing-workflows/task-parameter-properties.htm

關鍵來了,緊隨其後,官方做了這樣一段補充。

對於 GPRecordSet 和 GPFeatureRecordSetLayer 引數,如果有大量記錄和要素,則可以將 JSON 結構儲存在檔案中,然後提供 URL 作為輸入。例如,{ "url" :"http://myserver/myfeatures/afile.txt"}。文字檔案必須包含 JSON 格式的要素集或記錄集。

也就是說我們可以把大量的JSON格式的要素儲存在.json或.txt中,傳入這個檔案的url作為GPFeatureRecordSetLayer 型別引數的值。

還能有這樣的操作,我是完全沒有想過的。所以這裡有個問題就是,我之前都只是從GP工具的引數型別出發,從釋出的角度來思考問題;但是沒有從呼叫的時候來思考問題。

那麼我夢寐以求的將圖層服務的URL作為引數怎麼做呢。問題就變成了圖層服務如何提供一個JSON?當然這個問題我之前已經做過了,就是對圖層服務進行查詢。因此就有了這篇文章裡說得,將在mapserver的查詢url直接傳入引數。

https://blog.csdn.net/peckerzeng/article/details/83029032

下面我來大概說一下如何利用GPFeatureRecordSetLayer將釋出的地圖服務作為引數:

首先我們的GP工具引數型別要選擇FeatureSet,將GP工部發布成GP服務時,輸入模式選擇User defiend value。

其次我們對釋出的mapserver進行查詢,獲取json。如http://*********/arcgis/rest/services/gptest/byblk/MapServer/0/query?where=OBJECTID%3C%3E-1&f=json是一個查詢url,表示查出這個地圖服務的子圖層0的所有要素,且返回格式為JSON。這裡條件依舊是OBJECTID!=-1,這樣可以查詢出所有的要素。(我一直沒找到獲取mapserver所有引數的方法。這裡如果條件置空,則是什麼都不返回。所有要是有朋友查詢方法可以留言)

提示:構建形如{ "url" :"http://myserver/myfeatures/afile.txt"} 的引數。這裡中括號是不能少的,因為GPFeatureRecordSetLayer一直都是接受JSON型別的引數。

剩下的就是輸入了

至此使用講解結束。我們稍微探究一下,它的工作原理。回到我們開篇講的,由於輸入引數是json,ArcGIS本身是不能對json進行處理的,都要進行轉化工作。那這個轉化工作在哪裡進行呢?

補充一點知識,如果arcgisserver 執行了GP服務,會在伺服器上生成GP服務的jobid對應的資料夾。一般目錄如下圖。

但是非常沮喪,我開啟伺服器上對應的job目錄,並沒有找到下載的json檔案。那這個json檔案去哪了呢,我在server下的cache目錄裡也沒看到。我開啟scratch.gdb之後,也是隻有結果檔案。也沒有轉化後的輸入檔案。非常神奇。

當然現在很晚了,我沒辦法測試這種模式下服務的效能。現在只是用了一個很簡單的資料,做了一個很簡單的處理。要是和我之前做的那個一樣慢,我可真是太高興了。

補充:我寫完整篇博文後,感覺文章有點頭重腳輕,結尾也是戛然而止。但是能力有限,到此為止。哈哈