Android實現線上播放音樂
Android實現線上播放音樂
2014年3月10日
hello,小夥伴們,3月份珊珊來遲的第一篇部落格,最近小巫在找工作,加上又生病了,就沒有太多精力去寫部落格了。今天拖著病發表一篇之前已經實現的線上播放音樂效果,線上播放音樂並不難,也就是傳入的Url是一個網路地址。這裡我要實現一個可以進行網路緩衝的線上音樂播放。
效果實現:
本篇部落格是在多執行緒下載那篇部落格增加線上播放音樂實現,下面是提供專案下載地址:
下面介紹具體實現過程:
定義一個具有緩衝效果的播放器:Player
我們看到的緩衝效果,是通過設定拖動條SeekBar的二級進度實現的,這就要設定MediaPlayer的緩衝更新的監聽了。
具體程式碼實現:
package com.wwj.download.util;
import java.io.IOException;
import java.util.Timer;
import java.util.TimerTask;
import android.media.AudioManager;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnBufferingUpdateListener;
import android.media.MediaPlayer.OnCompletionListener;
import android.media.MediaPlayer.OnPreparedListener;
import android.os.Handler;
import android.util.Log;
import android.widget.SeekBar;
public class Player implements OnBufferingUpdateListener, OnCompletionListener,
OnPreparedListener {
public MediaPlayer mediaPlayer; // 媒體播放器
private SeekBar seekBar; // 拖動條
private Timer mTimer = new Timer(); // 計時器
// 初始化播放器
public Player(SeekBar seekBar) {
super();
this.seekBar = seekBar;
try {
mediaPlayer = new MediaPlayer();
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);// 設定媒體流型別
mediaPlayer.setOnBufferingUpdateListener(this);
mediaPlayer.setOnPreparedListener(this);
} catch (Exception e) {
e.printStackTrace();
}
// 每一秒觸發一次
mTimer.schedule(timerTask, 0, 1000);
}
// 計時器
TimerTask timerTask = new TimerTask() {
@Override
public void run() {
if (mediaPlayer == null)
return;
if (mediaPlayer.isPlaying() && seekBar.isPressed() == false) {
handler.sendEmptyMessage(0); // 傳送訊息
}
}
};
Handler handler = new Handler() {
public void handleMessage(android.os.Message msg) {
int position = mediaPlayer.getCurrentPosition();
int duration = mediaPlayer.getDuration();
if (duration > 0) {
// 計算進度(獲取進度條最大刻度*當前音樂播放位置 / 當前音樂時長)
long pos = seekBar.getMax() * position / duration;
seekBar.setProgress((int) pos);
}
};
};
public void play() {
mediaPlayer.start();
}
/**
*
* @param url
* url地址
*/
public void playUrl(String url) {
try {
mediaPlayer.reset();
mediaPlayer.setDataSource(url); // 設定資料來源
mediaPlayer.prepare(); // prepare自動播放
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalStateException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
// 暫停
public void pause() {
mediaPlayer.pause();
}
// 停止
public void stop() {
if (mediaPlayer != null) {
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
}
// 播放準備
@Override
public void onPrepared(MediaPlayer mp) {
mp.start();
Log.e("mediaPlayer", "onPrepared");
}
// 播放完成
@Override
public void onCompletion(MediaPlayer mp) {
Log.e("mediaPlayer", "onCompletion");
}
/**
* 緩衝更新
*/
@Override
public void onBufferingUpdate(MediaPlayer mp, int percent) {
seekBar.setSecondaryProgress(percent);
int currentProgress = seekBar.getMax()
* mediaPlayer.getCurrentPosition() / mediaPlayer.getDuration();
Log.e(currentProgress + "% play", percent + " buffer");
}
}
package com.wwj.download;
import java.io.File;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.SeekBar;
import android.widget.SeekBar.OnSeekBarChangeListener;
import android.widget.TextView;
import android.widget.Toast;
import com.wwj.download.util.Player;
import com.wwj.net.download.DownloadProgressListener;
import com.wwj.net.download.FileDownloader;
public class MainActivity extends Activity {
private static final int PROCESSING = 1;
private static final int FAILURE = -1;
private EditText pathText; // url地址
private TextView resultView;
private Button downloadButton;
private Button stopButton;
private ProgressBar progressBar;
private Button playBtn;
private Player player; // 播放器
private SeekBar musicProgress; // 音樂進度
private Handler handler = new UIHandler();
private final class UIHandler extends Handler {
public void handleMessage(Message msg) {
switch (msg.what) {
case PROCESSING: // 更新進度
progressBar.setProgress(msg.getData().getInt("size"));
float num = (float) progressBar.getProgress()
/ (float) progressBar.getMax();
int result = (int) (num * 100); // 計算進度
resultView.setText(result + "%");
if (progressBar.getProgress() == progressBar.getMax()) { // 下載完成
Toast.makeText(getApplicationContext(), R.string.success,
Toast.LENGTH_LONG).show();
}
break;
case FAILURE: // 下載失敗
Toast.makeText(getApplicationContext(), R.string.error,
Toast.LENGTH_LONG).show();
break;
}
}
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
pathText = (EditText) findViewById(R.id.path);
resultView = (TextView) findViewById(R.id.resultView);
downloadButton = (Button) findViewById(R.id.downloadbutton);
stopButton = (Button) findViewById(R.id.stopbutton);
progressBar = (ProgressBar) findViewById(R.id.progressBar);
ButtonClickListener listener = new ButtonClickListener();
downloadButton.setOnClickListener(listener);
stopButton.setOnClickListener(listener);
playBtn = (Button) findViewById(R.id.btn_online_play);
playBtn.setOnClickListener(listener);
musicProgress = (SeekBar) findViewById(R.id.music_progress);
player = new Player(musicProgress);
musicProgress.setOnSeekBarChangeListener(new SeekBarChangeEvent());
}
private final class ButtonClickListener implements View.OnClickListener {
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.downloadbutton: // 開始下載
// http://abv.cn/music/光輝歲月.mp3,可以換成其他檔案下載的連結
String path = pathText.getText().toString();
String filename = path.substring(path.lastIndexOf('/') + 1);
try {
// URL編碼(這裡是為了將中文進行URL編碼)
filename = URLEncoder.encode(filename, "UTF-8");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
path = path.substring(0, path.lastIndexOf("/") + 1) + filename;
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
// File savDir =
// Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MOVIES);
// 儲存路徑
File savDir = Environment.getExternalStorageDirectory();
download(path, savDir);
} else {
Toast.makeText(getApplicationContext(),
R.string.sdcarderror, Toast.LENGTH_LONG).show();
}
downloadButton.setEnabled(false);
stopButton.setEnabled(true);
break;
case R.id.stopbutton: // 暫停下載
exit();
Toast.makeText(getApplicationContext(),
"Now thread is Stopping!!", Toast.LENGTH_LONG).show();
downloadButton.setEnabled(true);
stopButton.setEnabled(false);
break;
case R.id.btn_online_play:
new Thread(new Runnable() {
@Override
public void run() {
player.playUrl(pathText.getText().toString());
}
}).start();
break;
}
}
/*
* 由於使用者的輸入事件(點選button, 觸控螢幕....)是由主執行緒負責處理的,如果主執行緒處於工作狀態,
* 此時使用者產生的輸入事件如果沒能在5秒內得到處理,系統就會報“應用無響應”錯誤。
* 所以在主執行緒裡不能執行一件比較耗時的工作,否則會因主執行緒阻塞而無法處理使用者的輸入事件,
* 導致“應用無響應”錯誤的出現。耗時的工作應該在子執行緒裡執行。
*/
private DownloadTask task;
private void exit() {
if (task != null)
task.exit();
}
private void download(String path, File savDir) {
task = new DownloadTask(path, savDir);
new Thread(task).start();
}
/**
*
* UI控制元件畫面的重繪(更新)是由主執行緒負責處理的,如果在子執行緒中更新UI控制元件的值,更新後的值不會重繪到螢幕上
* 一定要在主執行緒裡更新UI控制元件的值,這樣才能在螢幕上顯示出來,不能在子執行緒中更新UI控制元件的值
*
*/
private final class DownloadTask implements Runnable {
private String path;
private File saveDir;
private FileDownloader loader;
public DownloadTask(String path, File saveDir) {
this.path = path;
this.saveDir = saveDir;
}
/**
* 退出下載
*/
public void exit() {
if (loader != null)
loader.exit();
}
DownloadProgressListener downloadProgressListener = new DownloadProgressListener() {
@Override
public void onDownloadSize(int size) {
Message msg = new Message();
msg.what = PROCESSING;
msg.getData().putInt("size", size);
handler.sendMessage(msg);
}
};
public void run() {
try {
// 例項化一個檔案下載器
loader = new FileDownloader(getApplicationContext(), path,
saveDir, 3);
// 設定進度條最大值
progressBar.setMax(loader.getFileSize());
loader.download(downloadProgressListener);
} catch (Exception e) {
e.printStackTrace();
handler.sendMessage(handler.obtainMessage(FAILURE)); // 傳送一條空訊息物件
}
}
}
}
// 進度改變
class SeekBarChangeEvent implements OnSeekBarChangeListener {
int progress;
@Override
public void onProgressChanged(SeekBar seekBar, int progress,
boolean fromUser) {
// 原本是(progress/seekBar.getMax())*player.mediaPlayer.getDuration()
this.progress = progress * player.mediaPlayer.getDuration()
/ seekBar.getMax();
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
// seekTo()的引數是相對與影片時間的數字,而不是與seekBar.getMax()相對的數字
player.mediaPlayer.seekTo(progress);
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (player != null) {
player.stop();
player = null;
}
}
}
相關文章
- matlab-播放音樂Matlab
- Android UI 實現廣告 Banner 輪播效果AndroidUI
- Vue之網易雲音樂PC版輪播圖的實現Vue
- 教你如何實現 Android TextView 文字輪播效果AndroidTextView
- Android自定義View實現文字輪播效果AndroidView
- 【.NET 與樹莓派】讓喇叭播放音樂樹莓派
- html網頁無法自動播放音樂HTML網頁
- Android中用SoundPool播放音訊Android音訊
- Android開源線上音樂播放器——波尼音樂Android播放器
- 在自己網站中插入網易雲音樂的外鏈,播放音樂網站
- 設計師研發智慧揹包Keeback,可實時定位和播放音樂
- Android中用MediaPlay播放音訊和影片Android音訊
- react實戰-線上音樂播放器React播放器
- 如何實現 Android 短影片跨頁面的流暢續播?Android
- ViewFlipper探索與使用——順便實現Android圖片輪播ViewAndroid
- Axure實現輪播效果
- ViewPage實現輪播圖View
- Banner實現輪播圖
- js實現輪播圖JS
- jQuery實現輪播效果jQuery
- Kusion智慧枕頭:是枕頭也是充電寶還能放音樂
- 線上視訊教學線上教育直播程式教育培訓系統線上錄播、回播
- Android QQ音樂/酷狗音樂鎖屏控制實現原理,酷狗鎖屏Android
- 在Linux終端命令列下播放音樂的命令(Ubuntu)Linux命令列Ubuntu
- iOS- 關於AVAudioSession的使用——後臺播放音樂iOSSession
- css實現滾動輪播CSS
- 原生js實現輪播圖JS
- 圖片輪播元件實現元件
- Android開源音樂播放器之線上音樂列表自動載入更多Android播放器
- Java 基於UDP 實現單播、組播、廣播 Socket 程式設計JavaUDP程式設計
- Android實現收款成功金額的語音播報功能(Nice tone)Android
- 線上直播系統原始碼,利用css和html實現首頁圖片輪播效果原始碼CSSHTML
- 使用微信API實現H5頁面播放音訊檔案APIH5音訊
- python實現播放音訊和錄音功能示例程式碼Python音訊
- LRC歌詞原理和實現高仿Android網易雲音樂Android
- C#實現任意源組播與特定源組播C#
- Bootstrap教程(26)–輪播的實現boot
- xbanner實現卡片式輪播