Android - GPS, Network 위치 정보 얻기 (LocationManager)

JS · 24 Dec 2020

LocationManager를 통해 GPS, Network의 위치 정보를 얻을 수 있습니다.

위치 정보를 얻는 방법은 LocationManager.getLastKnownLocation()으로 가장 마지막에 기록된 위치 정보를 가져오거나 LocationManager.requestLocationUpdates()으로 Listener를 등록하여 위치가 변경될 때마다 이벤트를 받을 수 있습니다.

Location 퍼미션

위치 정보를 얻으려면 다음 퍼미션 중 하나는 AndroidManifest에 정의해야 합니다.

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
  • ACCESS_COARSE_LOCATION : 도시 Block 단위의 정밀도의 위치 정보를 얻을 수 있습니다.
  • ACCESS_FINE_LOCATION : ACCESS_COARSE_LOCATION보다 더 정밀한 위치 정보를 얻을 수 있습니다.

Runtime Permission 요청에 대한 자세한 구현은 Android - Runtime permission 요청 방법을 참고해주세요.

마지막으로 확인된 위치 정보 얻기

LocationManager.getLastKnownLocation(provider) API로 가장 마지막으로 확인된 위치 정보를 얻을 수 있습니다. 오랫동안 Location 갱신이 안되는 등의 이유로, 매우 오래 전에 저장된 위치 정보가 리턴될 수 있습니다. 또는, 해당 Provider에 대한 저장된 위치 정보가 없을 때는 null이 리턴될 수 있습니다.

인자로 전달되는 Provider의 종류에는 일반적으로 GPS의 위치를 제공해주는 GPS_PROVIDER와 네트워크의 위치를 제공해주는 NETWORK_PROVIDER가 있습니다.

다음과 같이 위치 정보의 위도(latitude)와 경도(longitude)를 얻을 수 있습니다.

val location : Location? = locationManager
        .getLastKnownLocation(LocationManager.GPS_PROVIDER)
if (location != null) {
    val latitude = location.latitude
    val longitude = location.longitude
    Log.d("Test", "GPS Location changed, Latitude: $latitude" +
            ", Longitude: $longitude")
}

Location은 다음과 같은 정보들을 제공합니다.

  • getAccuracy() : 정확도
  • getLatitude() : 위도
  • getLongitude() : 경도
  • getTime() : 생성된 시간(UTC)
  • getElapsedRealtimeNanos() : 생성된 시간(Elapsed time)

다음과 같이 Location.getElapsedRealtimeNanos()으로 언제 생성된 Location 객체인지 알 수 있고, 현재에서 몇초 전에 생성된 것인지 계산할 수 있습니다. 이것으로 오래 된 데이터는 필터링할 수 있습니다.

val location : Location? = locationManager
        .getLastKnownLocation(LocationManager.GPS_PROVIDER)

val ageMs = TimeUnit.NANOSECONDS.toMillis(SystemClock.elapsedRealtimeNanos()
        - location.getElapsedRealtimeNanos())

위치 정보 변경에 대한 Listener

다음과 같이 위치 정보에 대한 리스너를 등록할 수 있으며, 리스너를 등록하면 Callback이 전달됩니다.

locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,
    10000,  // 10-second interval.
    10.0f,  // 10 meters
    gpsListener)


var gpsListener: LocationListener = object : LocationListener {
    override fun onLocationChanged(location: Location) {
        val latitude = location.latitude
        val longitude = location.longitude
        Log.d("Test", "GPS Location changed, Latitude: $latitude" +
                ", Longitude: $longitude")
    }

    override fun onStatusChanged(provider: String, status: Int, extras: Bundle) {
    }
    override fun onProviderEnabled(provider: String) {
    }
    override fun onProviderDisabled(provider: String) {
    }
}

requestLocationUpdates()의 인자에는 다음과 같은 내용이 전달됩니다.

  • Provider: GPS_PROVIDER 또는 NETWORK_PROVIDER 가 될 수 있습니다.
  • minInterval(ms) : 10초로 설정하면 업데이트가 있을 때 최소 10초 간격으로 이벤트를 전달합니다.
  • minDistance(meters) : 설정된 거리만큼의 변화가 있을 때 이벤트를 전달합니다.
  • Listener : Listener 객체를 전달합니다.

Listener 등록 해지

Listener를 사용하지 않을 때는 다음과 같이 해지할 수 있습니다.

locationManager.removeUpdates(listener);

사용 가능한 Provider인지 확인

다음과 같이 디바이스에서 GPS 또는 Network 프로바이더가 사용 가능한지 확인할 수 있습니다.

val isGPSEnabled =
    locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER)
val isNetworkEnabled =
    locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)

만약 GPS 설정이 꺼져있다면 GPS_PROVIDER가 비활성화되었다고 리턴됩니다. 이런 경우, 다음과 같이 Settings 화면을 실행하여 사용자에게 GSP를 활성화하도록 가이드할 수 있습니다.

private fun enableLocationSettings() {
    val settingsIntent = Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS)
    startActivity(settingsIntent)
}

location settings

참고

댓글을 보거나 쓰려면 이 버튼을 눌러주세요.
codechachaCopyright ©2019 codechacha