Android音樂播放器簡單實(shí)現(xiàn)案例
音樂播放器開發(fā)
MediaPlayer 是Android 控制音頻和視頻文件播放類
1.創(chuàng)建MediaPlayer 對象 的Create方法
2.無can構(gòu)造方法 -> setDataSorce -> prepare()加載創(chuàng)造文件
注意訪問SDK需要授予權(quán)限
當(dāng)Mediaplay.stop() 資源后需要重新加載資源,使用Mediaplay.setDataSource() 方法進(jìn)行加載 ,最后還需要釋放資源,防止內(nèi)存泄漏
缺點(diǎn):
1.延遲低長,占有資源多
2. 不支持同時(shí)播放多個音頻
package com.example.myapplication;
import android.media.MediaPlayer;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.view.View;
import android.widget.ImageButton;
import android.widget.Toast;
import androidx.annotation.RequiresApi;
import androidx.appcompat.app.AppCompatActivity;
import java.io.File;
import java.io.IOException;
/**
* MediaPlay 實(shí)現(xiàn)簡易的播放器
*/
public class MainActivity extends AppCompatActivity {
private MediaPlayer mediaPlayer; //定義MediaPlayer對象
private boolean isPause = false; //定義是否暫停
private File file;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final ImageButton btn_play = (ImageButton) findViewById(R.id.btn_play); //獲取“播放/暫?!卑粹o
final ImageButton btn_stop = (ImageButton) findViewById(R.id.btn_stop); //獲取“停止”按鈕
;
file = new File("/res/raw/bird.ogg");
if (file.exists()) {
mediaPlayer = MediaPlayer.create(this, Uri.parse(file.getAbsolutePath()));
} else {
Toast.makeText(MainActivity.this, "要播放的音頻文件不存在!", Toast.LENGTH_SHORT).show();
return;
}
// 為MediaPlayer 添加監(jiān)聽事件 完成事件監(jiān)聽器,當(dāng)media.play.stop 后在重新play
mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mediaPlayer) {
play();
}
});
btn_play.setOnClickListener(new View.OnClickListener(){
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
@Override
public void onClick(View view) {
if(mediaPlayer.isPlaying() && !isPause){
mediaPlayer.pause();
isPause = true; // 暫停播放
// 更好播放圖標(biāo)
((ImageButton) view).setImageDrawable(getResources().getDrawable(R.drawable.play, null));
}else{
mediaPlayer.start(); //繼續(xù)播放
// 更換為暫停圖標(biāo)
((ImageButton) view).setImageDrawable(getResources().getDrawable(R.drawable.pause, null));
isPause = false; //設(shè)置為播放狀態(tài)
}
}
});
btn_stop.setOnClickListener(new View.OnClickListener() { //單擊停止按鈕,實(shí)現(xiàn)停止播放音頻
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
@Override
public void onClick(View v) {
mediaPlayer.stop(); //停止播放
//更換為播放圖標(biāo)
btn_play.setImageDrawable(getResources().getDrawable(R.drawable.play, null));
}
});
}
private void play(){
// 重置MediaPlay
mediaPlayer.reset();
try {
mediaPlayer.setDataSource(file.getAbsolutePath());
mediaPlayer.prepare(); // 預(yù)加載
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
protected void onDestroy() {
if(mediaPlayer.isPlaying()){
mediaPlayer.stop();
}
mediaPlayer.release();
super.onDestroy();
}
}SoundPool 播放多個音頻
優(yōu)點(diǎn):
1. 延遲低
2.占有資源少
3. 支持多個音頻播放
缺點(diǎn): 不能播放大文件音頻
應(yīng)用場景: 小型游戲中多個飛機(jī)被擊破聲音
創(chuàng)建SoundPool 對象,load , play
package com.example.myapplication;
import android.media.AudioAttributes;
import android.media.SoundPool;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ListView;
import androidx.appcompat.app.AppCompatActivity;
import java.util.HashMap;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ListView listView = findViewById(R.id.listView);
// 創(chuàng)建SoundPool 對象設(shè)置音頻相關(guān)特性
//1.音頻對象
AudioAttributes attr = new AudioAttributes.Builder() //設(shè)置音效相關(guān)屬性
.setUsage(AudioAttributes.USAGE_GAME) // 設(shè)置音效使用場景
.setContentType(AudioAttributes.CONTENT_TYPE_MUSIC) // 設(shè)置音效的類型
.build();
final SoundPool soundpool = new SoundPool.Builder() // 創(chuàng)建SoundPool對象
.setAudioAttributes(attr) // 設(shè)置音效池的屬性
.setMaxStreams(10) // 設(shè)置最多可容納10個音頻流,
.build();
// 要播放音頻保存到HashMap中
final HashMap<Integer, Integer> soundmap = new HashMap<Integer, Integer>();
soundmap.put(0, soundpool.load(this, R.raw.cuckoo, 1));
soundmap.put(1, soundpool.load(this, R.raw.chimes, 1));
soundmap.put(2, soundpool.load(this, R.raw.notify, 1));
soundmap.put(3, soundpool.load(this, R.raw.ringout, 1));
soundmap.put(4, soundpool.load(this, R.raw.bird, 1));
soundmap.put(5, soundpool.load(this, R.raw.water, 1));
soundmap.put(6, soundpool.load(this, R.raw.cock, 1));
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
soundpool.play(soundmap.get(i),1,1,0,0,1);
}
});
}
}video View播放視頻
在XML中添加Videoview 組件:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<VideoView
android:id="@+id/video"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</RelativeLayout>
讀取SDK上數(shù)據(jù) 設(shè)置Mainifest.xml 設(shè)置訪問權(quán)限,同時(shí)在Android 機(jī)器上設(shè)置權(quán)限打開
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
-----------------------------------------------------------
<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.mingrisoft"
xmlns:android="http://schemas.android.com/apk/res/android">
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity" android:screenOrientation="landscape">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>
package com.example.myapplication;
import android.media.MediaPlayer;
import android.os.Bundle;
import com.google.android.material.floatingactionbutton.FloatingActionButton;
import com.google.android.material.snackbar.Snackbar;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import android.os.Environment;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.MediaController;
import android.widget.Toast;
import android.widget.VideoView;
import java.io.File;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
VideoView videoView = findViewById(R.id.video);
File file = new File(Environment.getExternalStorageDirectory()+"/video.mp4");
if(file.exists()){
videoView.setVideoPath(file.getAbsolutePath());
}else{
return ;
}
// 控制視頻播放
MediaController mc = new MediaController(MainActivity.this);
videoView.setMediaController(mc);
// videoiew 獲取焦點(diǎn)
videoView.requestFocus();
videoView.start();
// 完成監(jiān)聽器
videoView.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mediaPlayer) {
Toast.makeText(MainActivity.this,"視頻播放完成",Toast.LENGTH_SHORT).show();
}
});
}
}控制攝像頭攝像
<!-- 授予程序可以向SD卡中保存文件的權(quán)限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<!-- 授予程序使用攝像頭的權(quán)限 -->
<uses-permission android:name="android.permission.CAMERA"/>
<uses-feature android:name="android.hardware.camera.autofocus"/>
<uses-feature android:name="android.hardware.camera"/>
package com.example.myapplication;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.PixelFormat;
import android.hardware.Camera;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.WindowManager;
import android.widget.ImageButton;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
public class MainActivity extends AppCompatActivity {
private Camera camera;
private boolean isPreview = false; // 設(shè)置是否是預(yù)覽狀態(tài)
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//設(shè)置全屏顯示
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
if (!android.os.Environment.getExternalStorageState().equals( //判斷手機(jī)是否安裝SD卡
android.os.Environment.MEDIA_MOUNTED)) {
Toast.makeText(this, "請安裝SD卡!", Toast.LENGTH_SHORT).show(); // 提示安裝SD卡
}
// 設(shè)置預(yù)覽
final SurfaceView surfaceView = findViewById(R.id.surfaceView);
final SurfaceHolder surfaceHolder = surfaceView.getHolder();
// 設(shè)置自己維護(hù)緩存
surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
ImageButton preview = (ImageButton) findViewById(R.id.preview); //獲取“預(yù)覽”按鈕
ImageButton takePicture = (ImageButton) findViewById(R.id.takephoto); //獲取“拍照”按鈕
preview.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (!isPreview) {
camera = Camera.open();
isPreview = true;
}
try {
camera.setPreviewDisplay(surfaceHolder); // 設(shè)置用于顯示預(yù)覽的SurfaceView
Camera.Parameters parameters = camera.getParameters(); //獲取相機(jī)參數(shù)
parameters.setPictureFormat(PixelFormat.JPEG); //指定圖片為JPEG圖片
parameters.set("jpeg-quality", 80); //設(shè)置圖片的質(zhì)量
camera.setParameters(parameters); //重新設(shè)置相機(jī)參數(shù)
camera.startPreview(); //開始預(yù)覽
camera.autoFocus(null); //設(shè)置自動對焦
} catch (IOException e) { //輸出異常信息
e.printStackTrace();
}
}
});
// 保存拍照
//實(shí)現(xiàn)相機(jī)拍照功能
takePicture.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (camera != null) { //相機(jī)不為空
camera.takePicture(null, null, jpeg); //進(jìn)行拍照
}
}
});
}
final Camera.PictureCallback jpeg = new Camera.PictureCallback() {
@Override
public void onPictureTaken(byte[] data, Camera camera) {
// 根據(jù)拍照所得的數(shù)據(jù)創(chuàng)建位圖
final Bitmap bm = BitmapFactory.decodeByteArray(data, 0,
data.length);
camera.stopPreview(); //停止預(yù)覽
isPreview = false; //設(shè)置為非預(yù)覽狀態(tài)
//獲取sd卡根目錄
File appDir = new File(Environment.getExternalStorageDirectory(), "/DCIM/Camera/");
if (!appDir.exists()) { //如果該目錄不存在
appDir.mkdir(); //就創(chuàng)建該目錄
}
String fileName = System.currentTimeMillis() + ".jpg"; //將獲取當(dāng)前系統(tǒng)時(shí)間設(shè)置為照片名稱
File file = new File(appDir, fileName); //創(chuàng)建文件對象
try { //保存拍到的圖片
FileOutputStream fos = new FileOutputStream(file); //創(chuàng)建一個文件輸出流對象
bm.compress(Bitmap.CompressFormat.JPEG, 100, fos); //將圖片內(nèi)容壓縮為JPEG格式輸出到輸出流對象中
fos.flush(); //將緩沖區(qū)中的數(shù)據(jù)全部寫出到輸出流中
fos.close(); //關(guān)閉文件輸出流對象
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
// 將照片插入到圖庫當(dāng)中
try {
MediaStore.Images.Media.insertImage(MainActivity.this.getContentResolver(),
file.getAbsolutePath(),fileName,null);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
// 更新通知圖庫
MainActivity.this.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://" + "")));
restCamera();
}
};
private void restCamera(){
if(!isPreview){
camera.startPreview();
isPreview = true;
}
}
@Override
protected void onPause() { //當(dāng)暫停Activity時(shí),停止預(yù)覽并釋放資源
if (camera != null) { //如果相機(jī)不為空
camera.stopPreview(); //停止預(yù)覽
camera.release(); //釋放資源
}
super.onPause();
}
}
到此這篇關(guān)于Android音樂播放器簡單實(shí)現(xiàn)案例的文章就介紹到這了,更多相關(guān)Android音樂播放器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android實(shí)現(xiàn)系統(tǒng)語言切換功能
這篇文章主要為大家詳細(xì)介紹了Android系統(tǒng)語言切換功能的實(shí)現(xiàn)方法,感興趣的小伙伴們可以參考一下2016-03-03
利用Android實(shí)現(xiàn)一種點(diǎn)贊動畫效果的全過程
最近做項(xiàng)目需要實(shí)現(xiàn)點(diǎn)贊動畫,下面這篇文章主要給大家介紹了關(guān)于利用Android實(shí)現(xiàn)一種點(diǎn)贊動畫效果的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-12-12
android實(shí)現(xiàn)注冊頁面開發(fā)
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)注冊頁面開發(fā),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
Flutter實(shí)現(xiàn)PopupMenu彈出式菜單按鈕詳解
這篇文章主要介紹了Flutter實(shí)現(xiàn)PopupMenu彈出式菜單按鈕,PopupMenuButton是一個用于創(chuàng)建彈出菜單的小部件,當(dāng)用戶點(diǎn)擊觸發(fā)按鈕時(shí),PopupMenuButton會在屏幕上方或下方彈出一個菜單,感興趣想要詳細(xì)了解可以參考下文2023-05-05
Android自定義控件實(shí)現(xiàn)按鈕滾動選擇效果
這篇文章主要為大家詳細(xì)介紹了Android自定義控件實(shí)現(xiàn)按鈕滾動選擇效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07
Android編程獲取手機(jī)后臺運(yùn)行服務(wù)的方法
這篇文章主要介紹了Android編程獲取手機(jī)后臺運(yùn)行服務(wù)的方法,涉及Android針對系統(tǒng)服務(wù)的相關(guān)操作技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-12-12
Android中實(shí)現(xiàn)長按照片彈出右鍵菜單功能的實(shí)例代碼
這篇文章主要介紹了Android中實(shí)現(xiàn)長按照片彈出右鍵菜單功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-01-01
解決android studio卡頓,提升studio運(yùn)行速度的方法
這篇文章主要介紹了解決android studio卡頓,提升studio運(yùn)行速度的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03
Android?Choreographer源碼詳細(xì)分析
Choreographer的作用主要是配合Vsync,給上層App的渲染提供一個穩(wěn)定的Message處理的時(shí)機(jī),也就是Vsync到來的時(shí)候,系統(tǒng)通過對Vsync信號周期的調(diào)整,來控制每一幀繪制操作的時(shí)機(jī)2022-08-08

