2016年2月27日 星期六

Android Service,IntentServcie

The bad of async task
it is easy to use
it only suitable for the task that is within 2 s
But!!!!


It is not tie to the activity life cycle,that mean aithough the the Activity is onDestory, the async task keep run,until it finish its job. But no activity and onPostExecute will recive the result


Morever,forexample we set the asynctask will run when onResume, so eery time we rotate the device , will create a asynctask object.use many resource,However, only the final async task object is useful.


if we close the application,the asynctask still run.Nut its proority will low,if system need resource,the resource its hold will release.


finally,only the activity is visible that only can execute the asynctask

if we ned to run some task even the app is not visible,we need service!!!!
the main component of a android application
Service
是可以在背景中長時間執行操作的應用程式元件,且不提供使用者介面。 也是另一個可以啟動服務的應用程式元件,就算使用者切換至其他應用程式,也會繼續在背景中執行。 此外,元件也可以繫結至服務,以便與其互動,甚至執行處理程序間通訊 (IPC)。 例如,服務可能處理網路交易、播放音樂、執行檔案輸入/輸出或與內容供應程式互動,這些都可以從背景執行。


startService()
stopService()



priority of running service is higher than the background service


service example:
platingt music
voice 導航


How to start forground Service?

Foreground Service
當app有背景service在運行時, 可以利用foreground service的特性來提醒使用者目前有正在運行的背景服務,
這種方式也可以讓系統在記憶體不足時不會先被系統給砍掉
(先被砍掉的通常是那些看不到的背景服務)
要讓service變成Foreground Service的方法很簡單
只要在service動時, 發一個notification, 並且呼叫startForeground() method就可以了
Notification notification = new Notification(R.drawable.icon, getText(R.string.ticker_text),
        System.currentTimeMillis());
Intent notificationIntent = new Intent(this, ExampleActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);
notification.setLatestEventInfo(this, getText(R.string.notification_title),
        getText(R.string.notification_message), pendingIntent);
startForeground(ONGOING_NOTIFICATION_ID, notification);
如果要停止的話也只要呼叫stopForeground() method
P.S.
如果可以獲得系統權限的話, 也可以在AndroidManifest.xml的application增加
android:persistent="true"
如此一來也可以盡可能避免service被系統砍掉囉


some service this is run buy the user,the user must know it is running
forground dervice
前景服務是被認為使用者已主動意識到、就算在記憶體不足時系統也不會終結的服務。


in OnStartCommand-startForground




The Main thread is for th UI,so we should not run the service in the Main Thread.


we need to create a backgrounf thread and run it.

If want more easy,we can use IntentServcie




the concept of IntentService.


when we call startService, the Intent will go into the Intent Queue, the IntentService will handle the intent by first come first serve, until the Intent Queue is empty,the service self terminate. We a new Intent appear in the Queue, the service run again.
The IntentService already in the background thread.


Priority in the activity and service

three bucket,Low High,Critical

Critical should be never kill

High,if the system very need resource,kill

Low,kill the older first


3 law of android recource Management




example:

Using Service
Before we a using the asynctask in the application,new use Service to replac it.


1.create a folder call service

2.Create a new "service" package and a new IntentService called SunshineService.
3.Declare the SunshineService in the AndroidManifest.xml.<service android:name=".service.SunshineService"/>


<service
      android:name=".service.MyIntentService"
      android:exported="false"></service>
</application>


4.Move the code that fetches weather data from FetchWeatherTask'sdoInBackground into your new SunshineService's onHandleIntentmethod.
###the IntentService is the content;


5.Initiate the service using an explicit intent within the updateWeather method in ForecastFragment.
Intent intent=new Intent(getActivity(), MyIntentService.class);
intent.putExtra(Utility.STRAT_SERVICE_INTENT_KEY,location);
getActivity().startService(intent);

the soultion of intent service:


package com.example.android.sunshine.app.service;


import android.app.IntentService;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.net.Uri;
import android.text.format.Time;
import android.util.Log;


import com.example.android.sunshine.app.BuildConfig;
import com.example.android.sunshine.app.FetchWeatherTask;
import com.example.android.sunshine.app.Utility;
import com.example.android.sunshine.app.data.WeatherContract;


import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;


import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.Vector;

/**
* An {@link IntentService} subclass for handling asynchronous task requests in
* a service on a separate handler thread.
* <p/>
* TODO: Customize class - update intent actions and extra parameters.
*/
public class MyIntentService extends IntentService {
  private Context mContext;
  private final String LOG_TAG = FetchWeatherTask.class.getSimpleName();

  public MyIntentService() {
      super("MyIntentService");
      mContext=this;


  }


  @Override
  protected void onHandleIntent(Intent intent) {
      // If there's no zip code, there's nothing to look up.  Verify size of params.
      String[] params=new String[1];
      params[0]=intent.getStringExtra(Utility.STRAT_SERVICE_INTENT_KEY);
      if (params[0]==null) {
          return ;
      }
      String locationQuery = params[0];


      // These two need to be declared outside the try/catch
      // so that they can be closed in the finally block.
      HttpURLConnection urlConnection = null;
      BufferedReader reader = null;


      // Will contain the raw JSON response as a string.
      String forecastJsonStr = null;


      String format = "json";
      String units = "metric";
      int numDays = 14;


      try {
          // Construct the URL for the OpenWeatherMap query
          // Possible parameters are avaiable at OWM's forecast API page, at
          // http://openweathermap.org/API#forecast
          final String FORECAST_BASE_URL =
                  "http://api.openweathermap.org/data/2.5/forecast/daily?";
          final String QUERY_PARAM = "q";
          final String FORMAT_PARAM = "mode";
          final String UNITS_PARAM = "units";
          final String DAYS_PARAM = "cnt";
          final String APPID_PARAM = "APPID";


          Uri builtUri = Uri.parse(FORECAST_BASE_URL).buildUpon()
                  .appendQueryParameter(QUERY_PARAM, params[0])
                  .appendQueryParameter(FORMAT_PARAM, format)
                  .appendQueryParameter(UNITS_PARAM, units)
                  .appendQueryParameter(DAYS_PARAM, Integer.toString(numDays))
                  .appendQueryParameter(APPID_PARAM, BuildConfig.OPEN_WEATHER_MAP_API_KEY)
                  .build();


          URL url = new URL(builtUri.toString());


          // Create the request to OpenWeatherMap, and open the connection
          urlConnection = (HttpURLConnection) url.openConnection();
          urlConnection.setRequestMethod("GET");
          urlConnection.connect();


          // Read the input stream into a String
          InputStream inputStream = urlConnection.getInputStream();
          StringBuffer buffer = new StringBuffer();
          if (inputStream == null) {
              // Nothing to do.
              return ;
          }
          reader = new BufferedReader(new InputStreamReader(inputStream));


          String line;
          while ((line = reader.readLine()) != null) {
              // Since it's JSON, adding a newline isn't necessary (it won't affect parsing)
              // But it does make debugging a *lot* easier if you print out the completed
              // buffer for debugging.
              buffer.append(line + "\n");
          }


          if (buffer.length() == 0) {
              // Stream was empty.  No point in parsing.
              return ;
          }
          forecastJsonStr = buffer.toString();
          getWeatherDataFromJson(forecastJsonStr, locationQuery);
      } catch (IOException e) {
          Log.e(LOG_TAG, "Error ", e);
          // If the code didn't successfully get the weather data, there's no point in attempting
          // to parse it.
      } catch (JSONException e) {
          Log.e(LOG_TAG, e.getMessage(), e);
          e.printStackTrace();
      } finally {
          if (urlConnection != null) {
              urlConnection.disconnect();
          }
          if (reader != null) {
              try {
                  reader.close();
              } catch (final IOException e) {
                  Log.e(LOG_TAG, "Error closing stream", e);
              }
          }
      }
      return ;


  }


  long addLocation(String locationSetting, String cityName, double lat, double lon) {
      long locationId;


      // First, check if the location with this city name exists in the db
      Cursor locationCursor = mContext.getContentResolver().query(
              WeatherContract.LocationEntry.CONTENT_URI,
              new String[]{WeatherContract.LocationEntry._ID},
              WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING + " = ?",
              new String[]{locationSetting},
              null);


      if (locationCursor.moveToFirst()) {
          int locationIdIndex = locationCursor.getColumnIndex(WeatherContract.LocationEntry._ID);
          locationId = locationCursor.getLong(locationIdIndex);
      } else {
          // Now that the content provider is set up, inserting rows of data is pretty simple.
          // First create a ContentValues object to hold the data you want to insert.
          ContentValues locationValues = new ContentValues();


          // Then add the data, along with the corresponding name of the data type,
          // so the content provider knows what kind of value is being inserted.
          locationValues.put(WeatherContract.LocationEntry.COLUMN_CITY_NAME, cityName);
          locationValues.put(WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING, locationSetting);
          locationValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LAT, lat);
          locationValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LONG, lon);


          // Finally, insert location data into the database.
          Uri insertedUri = mContext.getContentResolver().insert(
                  WeatherContract.LocationEntry.CONTENT_URI,
                  locationValues
          );


          // The resulting URI contains the ID for the row.  Extract the locationId from the Uri.
          locationId = ContentUris.parseId(insertedUri);
      }


      locationCursor.close();
      // Wait, that worked?  Yes!
      return locationId;
  }


  /**
   * Take the String representing the complete forecast in JSON Format and
   * pull out the data we need to construct the Strings needed for the wireframes.
   *
   * Fortunately parsing is easy:  constructor takes the JSON string and converts it
   * into an Object hierarchy for us.
   */
  private void getWeatherDataFromJson(String forecastJsonStr,
                                      String locationSetting)
          throws JSONException {


      // Now we have a String representing the complete forecast in JSON Format.
      // Fortunately parsing is easy:  constructor takes the JSON string and converts it
      // into an Object hierarchy for us.


      // These are the names of the JSON objects that need to be extracted.


      // Location information
      final String OWM_CITY = "city";
      final String OWM_CITY_NAME = "name";
      final String OWM_COORD = "coord";


      // Location coordinate
      final String OWM_LATITUDE = "lat";
      final String OWM_LONGITUDE = "lon";


      // Weather information.  Each day's forecast info is an element of the "list" array.
      final String OWM_LIST = "list";


      final String OWM_PRESSURE = "pressure";
      final String OWM_HUMIDITY = "humidity";
      final String OWM_WINDSPEED = "speed";
      final String OWM_WIND_DIRECTION = "deg";


      // All temperatures are children of the "temp" object.
      final String OWM_TEMPERATURE = "temp";
      final String OWM_MAX = "max";
      final String OWM_MIN = "min";


      final String OWM_WEATHER = "weather";
      final String OWM_DESCRIPTION = "main";
      final String OWM_WEATHER_ID = "id";


      try {
          JSONObject forecastJson = new JSONObject(forecastJsonStr);
          JSONArray weatherArray = forecastJson.getJSONArray(OWM_LIST);


          JSONObject cityJson = forecastJson.getJSONObject(OWM_CITY);
          String cityName = cityJson.getString(OWM_CITY_NAME);


          JSONObject cityCoord = cityJson.getJSONObject(OWM_COORD);
          double cityLatitude = cityCoord.getDouble(OWM_LATITUDE);
          double cityLongitude = cityCoord.getDouble(OWM_LONGITUDE);


          long locationId = addLocation(locationSetting, cityName, cityLatitude, cityLongitude);


          // Insert the new weather information into the database
          Vector<ContentValues> cVVector = new Vector<ContentValues>(weatherArray.length());


          // OWM returns daily forecasts based upon the local time of the city that is being
          // asked for, which means that we need to know the GMT offset to translate this data
          // properly.


          // Since this data is also sent in-order and the first day is always the
          // current day, we're going to take advantage of that to get a nice
          // normalized UTC date for all of our weather.


          Time dayTime = new Time();
          dayTime.setToNow();


          // we start at the day returned by local time. Otherwise this is a mess.
          int julianStartDay = Time.getJulianDay(System.currentTimeMillis(), dayTime.gmtoff);


          // now we work exclusively in UTC
          dayTime = new Time();


          for(int i = 0; i < weatherArray.length(); i++) {
              // These are the values that will be collected.
              long dateTime;
              double pressure;
              int humidity;
              double windSpeed;
              double windDirection;


              double high;
              double low;


              String description;
              int weatherId;


              // Get the JSON object representing the day
              JSONObject dayForecast = weatherArray.getJSONObject(i);


              // Cheating to convert this to UTC time, which is what we want anyhow
              dateTime = dayTime.setJulianDay(julianStartDay+i);


              pressure = dayForecast.getDouble(OWM_PRESSURE);
              humidity = dayForecast.getInt(OWM_HUMIDITY);
              windSpeed = dayForecast.getDouble(OWM_WINDSPEED);
              windDirection = dayForecast.getDouble(OWM_WIND_DIRECTION);


              // Description is in a child array called "weather", which is 1 element long.
              // That element also contains a weather code.
              JSONObject weatherObject =
                      dayForecast.getJSONArray(OWM_WEATHER).getJSONObject(0);
              description = weatherObject.getString(OWM_DESCRIPTION);
              weatherId = weatherObject.getInt(OWM_WEATHER_ID);


              // Temperatures are in a child object called "temp".  Try not to name variables
              // "temp" when working with temperature.  It confuses everybody.
              JSONObject temperatureObject = dayForecast.getJSONObject(OWM_TEMPERATURE);
              high = temperatureObject.getDouble(OWM_MAX);
              low = temperatureObject.getDouble(OWM_MIN);


              ContentValues weatherValues = new ContentValues();


              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_LOC_KEY, locationId);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_DATE, dateTime);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_HUMIDITY, humidity);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_PRESSURE, pressure);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_WIND_SPEED, windSpeed);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_DEGREES, windDirection);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_MAX_TEMP, high);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_MIN_TEMP, low);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_SHORT_DESC, description);
              weatherValues.put(WeatherContract.WeatherEntry.COLUMN_WEATHER_ID, weatherId);


              cVVector.add(weatherValues);
          }


          int inserted = 0;
          // add to database
          if ( cVVector.size() > 0 ) {
              ContentValues[] cvArray = new ContentValues[cVVector.size()];
              cVVector.toArray(cvArray);
              inserted = mContext.getContentResolver().bulkInsert(WeatherContract.WeatherEntry.CONTENT_URI, cvArray);
          }


          Log.d(LOG_TAG, "FetchWeatherTask Complete. " + inserted + " Inserted");


      } catch (JSONException e) {
          Log.e(LOG_TAG, e.getMessage(), e);
          e.printStackTrace();
      }
  }
}



the full project answer: