これは MIERUNE AdventCalendar 2022 20日目の記事です。
昨日は @northprint さんによる Cesium for Unity クイックスタート でした。
はじめに
QGISでは、SQLのような式を使ってデータ処理できる機能が用意されています。
式を使うことで、レイヤのスタイルやラベルといったシンボロジに関わる部分を制御したり、属性テーブルの値やジオメトリを処理したりするといったさまざまな処理が可能です。
意外に便利なのがこの式ですが、日本語の解説記事はあまり多くないように思うため、この記事では式の使い方についてまとめてみたいと思います。
式を使用できる場所
以下では代表的な場所をいくつか挙げてみます。なお、この他にもラベルだったりシンボロジなどさまざまな場所で使用可能です。
フィールド計算機
まず一番利用することが多い、フィールド計算機です。
フィールド計算機では、フィールドのデータ型を文字列から数値に変換したり、複数のフィールドの文字列を結合したりといったスプレッドシートの関数と同じようなことが可能です。それ以外にも、ラインの長さやポリゴンの面積を算出したり、ジオメトリの重心にポイントを生成といったようなGIS特有の関数も用意されています。
式によるジオメトリ
式によるジオメトリは、式を使って新しいジオメトリを作成することが可能です。
たとえば、ラインの始点や終点にポイントを作成するなど、プロセシングツールでやろうとすると以外に面倒な処理が簡単に実現可能だったりします。
●使用方法
プロセシングツールボックス>ベクタジオメトリ>式によるジオメトリ
ジオメトリジェネレータ
ジオメトリジェネレータは、式によるジオメトリと同様に式を使ってジオメトリを作成できます。
式によるジオメトリと異なるのは、ジオメトリジェネレータで作成されるジオメトリは、既存レイヤのシンボルとしてジオメトリが生成される点です。
なので、ジオメトリジェネレータでラインの始点にシンボルを表示させていた場合、ラインの始点の位置を修正すればシンボルの位置も動的に変更されることになります。
●使用方法
レイヤのシンボロジ>シンボルレイヤタイプ>ジオメトリジェネレータ
QGIS式による選択 or 抽出
フィールドに特定の値が入っている地物のみ選択や抽出したい場合に利用します。例えば、面積フィールドが5,000㎡よりも大きい(”area” > 5000)地物のみを抽出したいといった場合に有用です。
●使用方法
プロセシングツール>ベクタ選択>QGIS式による選択or抽出
式文字列ビルダー
上記の場所に配置されているε
のボタンをクリックすると「式文字列ビルダー」という式を記述するためのウィンドウが表示されます。
中央の枠内にはQGISに組み込まれている関数が分類されており、ここから目的の関数を探すことができます。
フィールド計算機
それでは、ここからはフィールド計算機の使い方を紹介します。
フィールドの設定
フィールド計算機を使用する場合、まずはレイヤのどこのフィールドに処理結果を書き出すのかを設定します。
新しいフィールドに結果を出力したい場合は、「新規フィールドを作成」に✓をいれます。
この場合、フィールド型(文字列、整数、少数など)を求める結果に応じて適切に入力する必要があります。出力結果は文字列なのに、フィールド型が数値となっているなど型が一致していない場合は、結果がNULLとなるため注意が必要です。
既存のフィールドに結果を出力したい場合は、「既存のフィールドを更新する」に✓をいれて、出力先のフィールドを選択してください。
なお、地物を選択した状態でフィールド計算機を開くと、左上の「選択されているn個の地物のみ更新する」がデフォルトで✓が入る仕様になっています。
これに✓をいれた状態でフィールド計算機を実行すると選択地物のみしか処理結果が反映されません。そのため、想定外の結果となるのを避けるためにも地物の選択を全解除したうえで実行するのが安全でしょう。
基本的事項
フィールド計算機を使用するにあたっての基本的事項です。
--何も囲わないと数値
123
--単一引用符で囲むと文字列
'string'
--二重引用符で囲むとフィールド名
"field"
フィールドの値を使用した計算例です。
--平方メートルの数値を平方キロメートルに変換
"area" / 1000000
--人口フィールドと面積フィールドから人口密度を計算
"population" / "area"
式にはコメントの記述も可能です。次の2通りの方法があります。
-- コメント記載
/* コメントを記載 */
データ型の変換
特定のカラムの値を文字列から数値に変換したり、数値を文字列に変換するなどが可能です。
--文字列のフィールドを数値に変換
to_int(string) --整数型
to_real(string) --実数型
--数値のフィールドを文字列に変換
to_string(number)
--数値のフィールドを文字列に変換し3桁で0埋め
lpad(to_string(number),3,'0')
文字列操作
文字列を操作する関数です。
--文字列を大文字に変換
upper(string)
--文字列を小文字に変換
lower(string)
--文字列の左端のn文字分を返す
left(string, n)
--文字列の右端のn文字分を返す
right(string, n)
--文字列を指定した別の文字列で置換した文字列を返す
replace(string, before, after):
--文字列の文字数を返す
length(string)
--複数の文字列を1つに結合したものを返す
concat(string1, string2)
条件分岐による値の入力
--条件をテストし、それに応じて異なる結果を返す
if(チェックする条件,正の場合の結果,負の場合の結果)
--cdをテキストに置換する(国土数値情報の都市地域データでの例)
case
when "layer_no" = 1 then '市街化区域'
when "layer_no" = 2 then '市街化調整区域'
when "layer_no" = 3 then 'その他用途地域'
when "layer_no" = 4 then '用途未設定'
end
--価格に応じてランク付け(国土数値情報の地価公示データでの例)
case
when "L01_006" < 100000 then 'low'
when "L01_006" < 300000 then 'mid'
else 'high'
end
- 都市地域データ
- 地価公示データ
ジオメトリ関係
それぞれの地物に応じた結果が返されます。
--現在の地物のジオメトリを返す(単体では使わず、他の関数と組み合わせて使用する)
$geometry
--ポリゴンの面積を返す
$area
--ラインの長さを返す
$length
--ポリゴンの周長を返す
$perimeter
--ポイントのXY座標を返す
$x
$y
--ジオメトリのwkt表現を返す
geom_to_wkt($geometry)
次のように複数の関数を組み合わせることも可能です。
--WGS84から平面直角座標系に変換し面積を算出
area(transform($geometry,'EPSG:4326','EPSG:6675'))
--ラインやポリゴンの重心点のXY座標の取得
x(centroid($geometry))
y(centroid($geometry))
--平面直角座標系のポイントレイヤをWGS84に変換して、緯度経度をカンマで区切りで取得(||は文字列の結合)
y(transform($geometry,'EPSG:6675','EPSG:4326'))|| ','||x(transform($geometry,'EPSG:6675','EPSG:4326'))
--二点のポイントから角度を計算(ここではラインの始点と終点を使用)
degrees(azimuth(start_point($geometry),end_point($geometry)))
空間結合
式を使って空間結合を行うことも可能です。
なお、大量の地物を処理しようとすると結構時間がかかるので、その場合はプロセシングツールの空間結合を使いましょう。
ここで使用するoverlay関数
は一対多の結合となる場合が想定されるためデータが配列で返ってくる仕様となっています。
配列は直接フィールドに出力できないため、一対一の結合の場合は[0]を式の後ろにつけて配列の先頭の要素を指定をします。一対多の場合はarray_to_string
で配列を文字列に変換させてから書き出しましょう。
--空港から最も近い駅名を取得
overlay_nearest('駅',"N02_005")[0]
--空港を含む行政区域から都道府県を取得
overlay_within('行政区域',"N03_001")[0]
--都道府県の行政区域に含まれる空港を抽出(一対多の結合)
array_to_string(overlay_contains('空港',"C28_005"))
- 行政区域データ(都道府県名でディゾルブしてから使用)
- 駅データ(鉄道データ内)
- 空港データ(ポイントに変換して使用)
集計関数
プロセシングツールの「空間結合(集計付き)」と概ね同様の機能です。
--市区町村の行政区域に含まれる地価公示のポイント数を集計
aggregate(
layer:='地価公示',
aggregate:='count',
expression:="L01_001",
filter:=contains(geometry(@parent), $geometry)
)
--市区町村の行政区域に含まれる地価公示のうち一番高い価格を取得
aggregate(
layer:='地価公示',
aggregate:='max',
expression:="L01_006",
filter:=contains(geometry(@parent), $geometry)
)
- 行政区域データ
- 地価公示データ
その他
最後は、施設を示すポイントデータと道路を示すラインデータから、その施設が接面する道路の方位を取得してみます。
with_variable
は第3引数の式で使われる変数を設定することができる関数です。第1引数で設定した変数名の前に@
をつけて使用します。
--施設(①)と施設から最も近い道路への最短となるラインの終点(②)を基に角度を計算
with_variable(
'angle',
degrees( --ラジアンから度に変換
azimuth( -- ①から見た②の方位をラジアンで返す
$geometry, --①
end_point( --②
shortest_line(
$geometry,
overlay_nearest('line',$geometry)[0])
)
)
),
--case文で角度に応じて方位を入力する
case
when @angle < 22.5 then '北'
when @angle < 67.5 then '北東'
when @angle < 110.5 then '東'
when @angle < 157.5 then '南東'
when @angle < 202.5 then '南'
when @angle < 247.5 then '南西'
when @angle < 292.5 then '西'
when @angle < 337.5 then '北西'
else '北'
end
)
※施設が角地にある場合などは、適切に取得できない可能性があります。
ラベル
ラベルでの式の利用例です。値の欄に直接式を入力することで、ラベル表示用に新規フィールドを作成することなく目的のラベルを表示することができます。
--人口フィールドを文字列化して単位をつけて表示
to_string("JINKO") || '人'
- 境界データ | 13112 世田谷区(e-stat)
https://www.e-stat.go.jp/gis/statmap-search?page=1&type=2&aggregateUnitForBoundary=A&toukeiCode=00200521&toukeiYear=2020&serveyId=A002005212020&prefCode=13&coordsys=1&format=shape&datum=2000
ジオメトリ作成関係
以下の式の結果はすべてジオメトリを返すので、「式によるジオメトリ」と「ジオメトリジェネレーター」どちらでも動作します。下記はジオメトリジェネレーターで表示している例です。
--100mのバッファを作成
buffer($geometry,100)
--ラインの始点にシンボル作成(青色)
start_point($geometry)
--ラインの終点にシンボル作成(赤色)
end_point($geometry)
--ラインの重心点にシンボル作成(青色)
centroid($geometry)
--ラインの中点にシンボル作成(赤色)
line_interpolate_point($geometry,$length/2)
--ラインの指定した頂点にポイント作成
make_point($x_at(1),$y_at(1))
--ターゲットレイヤの最も近い地物に対して最短となるラインを返す(黒の破線)
shortest_line($geometry, overlay_nearest('line', $geometry)[0])
--元のラインから50メートル間隔でラインを作成(黒の破線)
collect_geometries (
array_foreach (
generate_series(50,200,50), --50~200まで50間隔で配列を作る
offset_curve($geometry, @element)
)
)
--ポイントから16方位を描く(generate_seriesの引数を”0,315,45”にすれば8方位)
collect_geometries(
array_foreach(
generate_series(0,337.5,22.5),
make_line(
$geometry,
project($geometry, 100, radians(@element))
)
)
)
最後に、こちらの記事で紹介されていた、Joy Divisionの「Unknown Pleasures」のジャケットのような地図を作る、というのを試してみます。
このような図は「Joy Plot」と呼ばれているようで、30DayMapChallengeでも作成している方がいたり、PythonやRでも作成できたりするようです。
羊蹄山付近で試してみると以下のような結果になりました。
おわりに
以上、式を使ってできることをいろいろと紹介させていただきました。式では思った以上に多くのことに使えるので覚えておくといずれ役に立つ場面があるかと思います。
この記事を書いている時に知りましたが、QGISの公式ドキュメントに式や関数について詳しくまとまっていました。こちらも合わせて見ていただくと、より理解が深まると思います。
-
QGISのドキュメント | 式について
https://docs.qgis.org/3.22/ja/docs/user_manual/expressions/expression.html -
QGISのドキュメント | 関数一覧
https://docs.qgis.org/3.22/ja/docs/user_manual/expressions/functions_list.html#functions-list
また、こちらの記事もおすすめです。
-
QGISのフィールド演算機でできること
https://medium.com/@mitsuhamiyake/qgisのフィールド演算機でできること-530e7c86ad12 -
QGISのフィールド演算機を使いこなす
https://medium.com/@mitsuhamiyake/qgisのフィールド演算機を使いこなす-66459176c822 -
QGISのフィールド演算機でデータ型変換と文字列操作
https://medium.com/@mitsuhamiyake/qgisのフィールド演算機を使いこなす-2-31eb889a4502 -
QGISのフィールド演算機でジオメトリの属性を調べる、ジオメトリジェネレータでシンボルを表現する
https://medium.com/@mitsuhamiyake/qgisのフィールド演算機でジオメトリの属性を調べる-ジオメトリジェネレータでシンボルを表現する-c076dbd31450
明日は@asahina820さんです!お楽しみにー