Mark Tyers
Mark Tyers

Reputation: 93

Android Intent.getStringExtra() returns null in Broadcastreceiver

I made a service which makes an intent after the location changed and puts the location data into it.

intent.putExtra("Latitude", String.valueOf(loc.getLatitude())); //Double
            intent.putExtra("Longitude", String.valueOf(loc.getLongitude())); //Double
            intent.putExtra("Provider", loc.getProvider()); //String
            intent.putExtra("Altitude", loc.getAltitude());//Double
            intent.putExtra("Accuracy", loc.getAccuracy()); //float
            intent.putExtra("Speed", loc.getSpeed()); //float

Now, I have a Broadcastreceiver which is being fired every 30 seconds for posting that information to a server. However, when I try to get the Intent:

        Intent myIntent = new Intent(context, GPSService.class); //GPSService is the service
        String lat = (String) myIntent.getExtras().get("Latitude");
        String longi = (String) myIntent.getExtras().get("Longitude");

I get a NullPointerException

Attempt to invoke virtual method 'java.lang.Object android.os.Bundle.get(java.lang.String)' on a null object reference

Why is that? I checked the other solutions: Some say to check whether I put the right Dateformat into the intent (Which in this case is String), which I did, some say to use getIntent().getExtras().getString("aKey"); I also tried that as you can see above, but I still get null from the String.

There has to be some problem while passing it to the broadcastreceiver, because I log the values after doing the intent.putExtra() stuff, and they are definitely not null.

EDIT:

GPSService:

    package com.dandddeveloper.alarmisto;

import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;

public class GPSService extends Service
{
    public static final String BROADCAST_ACTION = "Hello World";
    private static final int TWO_MINUTES = 1000 * 60 * 2;
    public LocationManager locationManager;
    public MyLocationListener listener;
    public Location previousBestLocation = null;

    Intent intent;
    int counter = 0;

    @Override
    public void onCreate()
    {
        super.onCreate();
        intent = new Intent(BROADCAST_ACTION);
    }

    @Override
    public void onStart(Intent intent, int startId)
    {
        locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
        listener = new MyLocationListener();
        //locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 4000, 0, listener);
        locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 4000, 0, listener);
    }

    @Override
    public IBinder onBind(Intent intent)
    {
        return null;
    }

    protected boolean isBetterLocation(Location location, Location currentBestLocation) {
        if (currentBestLocation == null) {
            // A new location is always better than no location
            return true;
        }

        // Check whether the new location fix is newer or older
        long timeDelta = location.getTime() - currentBestLocation.getTime();
        boolean isSignificantlyNewer = timeDelta > TWO_MINUTES;
        boolean isSignificantlyOlder = timeDelta < -TWO_MINUTES;
        boolean isNewer = timeDelta > 0;

        // If it's been more than two minutes since the current location, use the new location
        // because the user has likely moved
        if (isSignificantlyNewer) {
            return true;
            // If the new location is more than two minutes older, it must be worse
        } else if (isSignificantlyOlder) {
            return false;
        }

        // Check whether the new location fix is more or less accurate
        int accuracyDelta = (int) (location.getAccuracy() - currentBestLocation.getAccuracy());
        boolean isLessAccurate = accuracyDelta > 0;
        boolean isMoreAccurate = accuracyDelta < 0;
        boolean isSignificantlyLessAccurate = accuracyDelta > 200;

        // Check if the old and new location are from the same provider
        boolean isFromSameProvider = isSameProvider(location.getProvider(),
                currentBestLocation.getProvider());

        // Determine location quality using a combination of timeliness and accuracy
        if (isMoreAccurate) {
            return true;
        } else if (isNewer && !isLessAccurate) {
            return true;
        } else if (isNewer && !isSignificantlyLessAccurate && isFromSameProvider) {
            return true;
        }
        return false;
    }



    /** Checks whether two providers are the same */
    private boolean isSameProvider(String provider1, String provider2) {
        if (provider1 == null) {
            return provider2 == null;
        }
        return provider1.equals(provider2);
    }



    @Override
    public void onDestroy() {
        // handler.removeCallbacks(sendUpdatesToUI);
        super.onDestroy();
        Log.v("STOP_SERVICE", "DONE");
        locationManager.removeUpdates(listener);
    }

    public static Thread performOnBackgroundThread(final Runnable runnable) {
        final Thread t = new Thread() {
            @Override
            public void run() {
                try {
                    runnable.run();
                } finally {

                }
            }
        };
        t.start();
        return t;
    }




    public class MyLocationListener implements LocationListener
    {

        public void onLocationChanged(final Location loc)
        {
            Log.i("***********************", "Location changed");
            if(isBetterLocation(loc, previousBestLocation)) {
                loc.getLatitude();
                loc.getLongitude();
                intent.putExtra("Latitude", String.valueOf(loc.getLatitude())); //Double
                intent.putExtra("Longitude", String.valueOf(loc.getLongitude())); //Double
                intent.putExtra("Provider", loc.getProvider()); //String
                intent.putExtra("Altitude", loc.getAltitude());//Double
                intent.putExtra("Accuracy", loc.getAccuracy()); //float
                intent.putExtra("Speed", loc.getSpeed()); //float
                intent.putExtra("Bearing", loc.getBearing()); //float
                Log.v("LOC", String.valueOf(loc.getLatitude()));
                Log.v("LOC", String.valueOf(loc.getLongitude()));
                Log.v("LOC", loc.getProvider());
                Log.v("LOC", String.valueOf(loc.getAltitude()));
                Log.v("LOC", String.valueOf(loc.getAccuracy()));
                Log.v("LOC", String.valueOf(loc.getSpeed()));
                Log.v("LOC", String.valueOf(loc.getBearing()));

                sendBroadcast(intent);

            }
        }

        public void onProviderDisabled(String provider)
        {
            Toast.makeText( getApplicationContext(), "Gps Disabled", Toast.LENGTH_SHORT ).show();
        }


        public void onProviderEnabled(String provider)
        {
            Toast.makeText( getApplicationContext(), "Gps Enabled", Toast.LENGTH_SHORT).show();
        }


        public void onStatusChanged(String provider, int status, Bundle extras)
        {

        }

    }
}

Calling the Alarmreceiver in MainActivity:

Intent alarmIntent = new Intent(getApplicationContext(), AlarmReceiver.class);
    pendingIntent = PendingIntent.getBroadcast(MainActivity.this, 0, alarmIntent, 0);

manager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, timeInMillis, pendingIntent);

Upvotes: 3

Views: 2086

Answers (2)

Sourav Chandra
Sourav Chandra

Reputation: 843

Create a BroadcastReceiver object in your MainActivity class as follows:

private BroadcastReceiver mMessageReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        // Extract data included in the Intent

    }
};

Also instead to using periodic checks in the MainActivity, instead create a Timer object and use sendBroadcast(intent) to send timed boradcasts.

Upvotes: 3

Pritish
Pritish

Reputation: 1368

There must be an onreceieve method for receiver.So use its intent and use intent.getStringExtra("key"); and use it.

Upvotes: -1

Related Questions