17

On onLocationChanged event I want to save my GPS route (Latitude , Longitude). Later I want to load this data and draw the route.

Who is the best way to do this, by using some array type (and save or load by using database) or XML files or something else?

Thanks.

Jovan
  • 4,684
  • 13
  • 64
  • 98

4 Answers4

72

The best way of storing double values in SharedPreferences without losing precision is:

  • Transform to bit representation to store it as long:

    prefsEditor.putLong("Latitude", Double.doubleToLongBits(location.getLatitude()));
    
  • To retrieve, transfrom from bit representation to double:

    double latitude = Double.longBitsToDouble(prefs.getLong("Latitude", 0);
    

However, I think that if you want to store a big amount of points is better using a SQLite database than saving each coordinate in a Key-Value pair in SharedPreferences or serialize the array in XML and write it to a file or to SharedPreferences. The database offers you the advantage of loading in memory only the points within the area you are displaying in the map, so you can save memory.

Juan T
  • 1,018
  • 11
  • 12
  • 1
    This is the correct answer, the accepted answer loses precision and is therefore not storing the right values. – SimpsOff Jun 28 '13 at 16:04
11

Correct me if I'm wrong, but it might be better to convert the double values to strings and then whenever you call them convert them back to doubles. Sounds like a pain, but oracle's website (http://download.oracle.com/javase/tutorial/java/nutsandbolts/datatypes.html) says:

"use a float (instead of double) if you need to save memory in large arrays of floating point numbers. This data type should never be used for precise values, such as currency"

prefEditor.putString("Latitude", Double.valueOf(loc.getLatitude()).toString());

Then, to call the value

String latitudeString = pref.getString("Latitude", "0");
double latitude = Double.parseDouble(latitudeString);

It definitely seems like more of a nuisance, but seems also more likely to return the exact value that you saved, also, instead of something slightly off. And I suppose that would depend on how completely accurate you need the values to be.

Though, as far as using an array, I'm clueless because I've never needed to use them and have no experience with them.

Reed
  • 14,703
  • 8
  • 66
  • 110
  • it's obvious why, from float to double you lose 'precision' -> accuracy – Filippo Mazza Sep 07 '12 at 13:46
  • 1
    This indeed makes a lot of sense, for example if you get address string from certain coordinates, then save the coordinates as float, and get address again - you get completly different string. Users will wonder why. – User Nov 21 '12 at 11:12
  • 1
    default value must be string, please correct pref.getString("Latitude", 0); to pref.getString("Latitude", "0"); (can't do it myself, edits must be 6 characters min) – Tilman Hausherr Jul 09 '13 at 11:25
  • @TilmanHausherr Thanks. I miss these things when I don't have Eclipse griping at me. – Reed Jul 09 '13 at 17:42
0

Think about the precision (not accuracy) you need to represent the data. Would a couple inches be close enough? Most handheld GPS receivers are lucky to be good to a couple meters. One degree-second of longitude at the equator moves you about 100 feet. If you use a signed 32-bit integer to represent your latitudes and longitudes, LatE6 & LonE6, you could store the largest longitude there is as +/-180,000,000 and the least significant digit is:

(1 / 1E6) * (3600 sec/deg) * (100 feet/sec) = 0.36 feet or 4 inches.

I've got to imagine my Garmin uses something like this, not a float with only 24 bits of precision and certainly not a 64 bit double. Conversion between integer and the double shouldn't cost much. You could even pack a lat/lon pair into a long.

0

Use SharedPreferences and Editor.

Check out this open source code, it will probably help you a lot: OsmandSettings.java

I'll explain the important parts of the code:

import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;

// These settings are stored in SharedPreferences, replace com.osmand.settings 
// with your own package name, or whatever String you want.
public static final String SHARED_PREFERENCES_NAME = "com.osmand.settings";
public static final String LATITUDE = "latitude";
public static final String LONGITUDE = "longitude";

To write to SharedPreferences:

public void onLocationChanged(Location location){
    SharedPreferences prefs = Context.getSharedPreferences(SHARED_PREFERENCES_NAME, Context.MODE_WORLD_READABLE);
    Editor editor = prefs.edit();
    //Save it as a float since SharedPreferences can't deal with doubles
    edit.putFloat(LATITUDE, (float) Location.getLatitude());
    edit.putFloat(LONGITUDE, (float) Location.getLongitude());
    edit.commit();
}

To read from SharedPreferences:

public void onLocationChanged(Location location){
    SharedPreferences prefs = Context.getSharedPreferences(SHARED_PREFERENCES_NAME, Context.MODE_WORLD_READABLE);
    double lat = (double)prefs.getFloat(LATITUDE, 0);
    double lon = (double)prefs.getFloat(LONGITUDE, 0);
}
magicman
  • 1,199
  • 6
  • 8
  • 1
    This will only store the last location. Just store them in an ArrayList or something – Falmarri Aug 31 '10 at 01:01
  • 2
    Per oracle `This data type [float] should never be used for precise values, such as currency`. Thus, if you want to retrieve the same value (which I assume you do), you shouldn't save/get as floats, but as String or Long. See [my answer](http://stackoverflow.com/a/6527594/802469) – Reed Nov 21 '12 at 18:43
  • geo coordinates dont have to be precise, the must be accurate. storing as float, double, int or long all works. – AlexWien Jun 02 '13 at 20:54