編輯:Android開發實例
大家好今天我將為大家分享基於Service與ContentProvider的音樂播放實例,對於接觸Android有一些時日的人來說,Android的核心也就是Activity,Service,ContentProvider,BroadCastReceiver,以及串聯它們的Intent五大模塊,Activity我就不用多說了,而我將就這個例子來說一下Service,以及ContentProvider.
Service:
Android中的服務,它與Activity不同,它是不能與用戶交互的,運行在後台的程序,如果我們退出應用時,沒有結束進程,它仍然在後台運行,那我們什麼時候會用到Service呢?比如我們播放音樂的時候,有可能想邊聽音樂邊干些其他事情,當我們退出播放音樂的應用,如果不用Service,我們就聽不到歌了,所以這時候就得用到Service了,又比如當我們一個應用的數據是通過網絡獲取的,不同時間(一段時間)的數據是不同的這時候我們可以用Service在後台定時更新,而不用每打開應用的時候在去獲取。
CotentProvider:
Android中的內容提供者,它讓我們可以通過一個URL跨應用獲取數據(通常是SQLite數據庫),我覺得Android這個還是機制還是非常不錯的,特別是我們想獲取Sdcard裡一些數據時,比如我們想獲取所有Sdcard裡的音頻,視頻,圖片等,我們只要通過一個URL就可以輕松搞定,其實我們在開機或者插入Sdcard時,Android會做一些事情,就是它自動建庫,將我們卡裡所有音頻,視頻,圖片等信息存在相應的表中,我們可以用DDMS打開看一下如下圖(data/data目錄下),紅線是我手機當前卡建立的數據庫(不同卡會建立不同的數據庫)
然後我們可以將這個數據庫導出,用可以打開.db的工具打開浏覽數據庫的相關信息如下圖所示(我這裡打開了音頻的數據表,可以看到我手機裡所有音頻文件,當然還有數據表字段):
本來這個應用是我用來寫播放音樂Widget的代碼,但是布局有點多,我就簡單化了,做了一個比較 簡單的Demo,老規矩Step by Step.
第一步:新建一個Android工程命名為MusicDemo.
第二步:候改main.xml布局文件(我這裡增加了四個按鈕,上一首,播放,下一首,暫停)代碼如下:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- >
- <TextView
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="Welcome to Mr Wei's blog."
- />
- <LinearLayout
- android:orientation="horizontal"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- >
- <Button
- android:id="@+id/previous"
- android:layout_height="fill_parent"
- android:layout_width="wrap_content"
- android:layout_weight="1"
- android:text="上一首"
- />
- <Button
- android:id="@+id/play"
- android:layout_height="fill_parent"
- android:layout_width="wrap_content"
- android:layout_weight="1"
- android:text="播放"
- />
- <Button
- android:id="@+id/next"
- android:layout_height="fill_parent"
- android:layout_width="wrap_content"
- android:layout_weight="1"
- android:text="下一首"
- />
- <Button
- android:id="@+id/pause"
- android:layout_height="fill_parent"
- android:layout_width="wrap_content"
- android:layout_weight="1"
- android:text="暫停"
- />
- </LinearLayout>
- </LinearLayout>
第三步:新建一個MusicService.java類,播放音樂都是在這個類裡進行的哦,代碼如下:
- package com.tutor.music;
- import java.io.IOException;
- import android.app.Service;
- import android.content.Intent;
- import android.database.Cursor;
- import android.media.MediaPlayer;
- import android.net.Uri;
- import android.os.IBinder;
- import android.provider.MediaStore;
- import android.widget.Toast;
- public class MusicService extends Service {
- String[] mCursorCols = new String[] {
- "audio._id AS _id", // index must match IDCOLIDX below
- MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.ALBUM,
- MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.DATA,
- MediaStore.Audio.Media.MIME_TYPE, MediaStore.Audio.Media.ALBUM_ID,
- MediaStore.Audio.Media.ARTIST_ID, MediaStore.Audio.Media.DURATION
- };
- private MediaPlayer mMediaPlayer;
- private Cursor mCursor;
- private int mPlayPosition = 0;
- public static final String PLAY_ACTION = "com.tutor.music.PLAY_ACTION";
- public static final String PAUSE_ACTION = "com.tutor.music.PAUSE_ACTION";
- public static final String NEXT_ACTION = "com.tutor.music.NEXT_ACTION";
- public static final String PREVIOUS_ACTION = "com.tutor.music.PREVIOUS_ACTION";
- @Override
- public IBinder onBind(Intent arg0) {
- // TODO Auto-generated method stub
- return null;
- }
- @Override
- public void onCreate() {
- super.onCreate();
- mMediaPlayer = new MediaPlayer();
- //通過一個URI可以獲取所有音頻文件
- Uri MUSIC_URL = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
- //這裡我過濾了一下,因為我機裡有些音頻文件是游戲音頻,很短
- //播放不到一秒鐘,我這裡作了處理,默認大於10秒的可以看作是歌
- mCursor = getContentResolver().query(MUSIC_URL, mCursorCols, "duration > 10000", null, null);
- }
- @Override
- public void onStart(Intent intent, int startId) {
- super.onStart(intent, startId);
- String action = intent.getAction();
- if(action.equals(PLAY_ACTION)){
- play();
- }else if(action.equals(PAUSE_ACTION)){
- pause();
- }else if(action.equals(NEXT_ACTION)){
- next();
- }else if(action.equals(PREVIOUS_ACTION)){
- previous();
- }
- }
- //play the music
- public void play() {
- inite();
- }
- //暫停時,結束服務
- public void pause() {
- stopSelf();
- }
- //上一首
- public void previous() {
- if (mPlayPosition == 0) {
- mPlayPosition = mCursor.getCount() - 1;
- } else {
- mPlayPosition--;
- }
- inite();
- }
- public void next() {
- if (mPlayPosition == mCursor.getCount() - 1) {
- mPlayPosition = 0;
- } else {
- mPlayPosition++;
- }
- inite();
- }
- public void inite() {
- mMediaPlayer.reset();
- String dataSource = getDateByPosition(mCursor, mPlayPosition);
- String info = getInfoByPosition(mCursor, mPlayPosition);
- //用Toast顯示歌曲信息
- Toast.makeText(getApplicationContext(), info, Toast.LENGTH_SHORT).show();
- try {
- mMediaPlayer.setDataSource(dataSource);
- mMediaPlayer.prepare();
- mMediaPlayer.start();
- } catch (IllegalArgumentException e1) {
- e1.printStackTrace();
- } catch (IllegalStateException e1) {
- e1.printStackTrace();
- } catch (IOException e1) {
- e1.printStackTrace();
- }
- }
- //根據位置來獲取歌曲位置
- public String getDateByPosition(Cursor c,int position){
- c.moveToPosition(position);
- int dataColumn = c.getColumnIndex(MediaStore.Audio.Media.DATA);
- String data = c.getString(dataColumn);
- return data;
- }
- //獲取當前播放歌曲演唱者及歌名
- public String getInfoByPosition(Cursor c,int position){
- c.moveToPosition(position);
- int titleColumn = c.getColumnIndex(MediaStore.Audio.Media.TITLE);
- int artistColumn = c.getColumnIndex(MediaStore.Audio.Media.ARTIST);
- String info = c.getString(artistColumn)+" " + c.getString(titleColumn);
- return info;
- }
- //服務結束時要釋放MediaPlayer
- public void onDestroy() {
- super.onDestroy();
- mMediaPlayer.release();
- }
- }
第四步:修改Musicdemo.java代碼如下(代碼比較簡潔易懂):
- package com.tutor.music;
- import android.app.Activity;
- import android.content.ComponentName;
- import android.content.Intent;
- import android.os.Bundle;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.widget.Button;
- public class MusicDemo extends Activity implements OnClickListener {
- private Button mPrevious,mPlay,mNext,mPause;
- private ComponentName component;
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- //oncreate裡代碼一如既往的少
- setupViews();
- }
- //初始化一些工作
- public void setupViews(){
- component = new ComponentName(this,
- MusicService.class);
- mPrevious = (Button)findViewById(R.id.previous);
- mPlay = (Button)findViewById(R.id.play);
- mNext = (Button)findViewById(R.id.next);
- mPause = (Button)findViewById(R.id.pause);
- mPrevious.setOnClickListener(this);
- mPlay.setOnClickListener(this);
- mNext.setOnClickListener(this);
- mPause.setOnClickListener(this);
- }
- //按鈕點擊事件響應
- public void onClick(View v) {
- if(v == mPrevious){
- Intent mIntent = new Intent(MusicService.PREVIOUS_ACTION);
- mIntent.setComponent(component);
- startService(mIntent);
- }else if(v == mPlay){
- Intent mIntent = new Intent(MusicService.PLAY_ACTION);
- mIntent.setComponent(component);
- startService(mIntent);
- }else if(v == mNext){
- Intent mIntent = new Intent(MusicService.NEXT_ACTION);
- mIntent.setComponent(component);
- startService(mIntent);
- }else{
- Intent mIntent = new Intent(MusicService.PAUSE_ACTION);
- mIntent.setComponent(component);
- startService(mIntent);
- }
- }
- }
第五步:修改AndroidManifest.xml,這裡只是把我們的MusicService申明進去,不然會報錯(第14行代碼),代碼如下:
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.tutor.music"
- android:versionCode="1"
- android:versionName="1.0">
- <application android:icon="@drawable/icon" android:label="@string/app_name">
- <activity android:name=".MusicDemo"
- android:label="@string/app_name">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <service android:name=".MusicService" android:exported="true" />
- </application>
- <uses-sdk android:minSdkVersion="7" />
- </manifest>
第六步:運行上述Android工程,效果如下圖所示:
效果1:首界面:
效果2:點擊播發按鈕開始播放音樂:
效果3:我們可以在設置(Settings)->應用(Applications)->正在運行的服務(Running Services)查看我們啟動了一個新的Service:
可以顯示在的Android任務,通過加載進度條的進展。進度條有兩種形狀。加載欄和加載微調(spinner)。在本章中,我們將討論微調(spinner)。Spinner 用
從事Android開發,免不了會在應用裡嵌入一些廣告SDK,在嵌入了眾多SDK後,發現幾乎每個要求在AndroidManifest.xml申明Activity
BroadcastReceiver 是系統全局廣播監聽類, 其主要方法是onReceive(),自定義的廣播類繼承於它並實現自己的onReceive()處理邏輯B
就目前的互聯網發展來看,已經有越來越多互聯網企業都在Android平台上部署其客戶端,並且為了提升用戶體驗,這些客戶端都做得布局合理而且美觀。本文所要介紹的And