Получение доступа к кешу медиаплеера
Я хочу переместить прогрессивно потоковый mp3-файл на SD-карту после полной загрузки. Есть ли способ достичь этого.
Я видел, что MediaPlayer
полностью загружает весь файл во время прогрессивной потоковой передачи, а затем мы можем искать любую часть файла. Я хочу переместить полностью потоковый файл на внешнее хранилище, чтобы в будущем воспроизведение не тратило данные и батарею.
3 ответа
Комментарий к исходному сообщению указывает вам правильное направление, но я подумал, что это может быть полезно немного разъяснить...
Я сделал простой прокси-сервер, используя HTTP-библиотеки Apache. Там должно быть много примеров, чтобы получить основы этой части. Укажите MediaPlayer соответствующий локальный URL-адрес, чтобы он открывал сокет для вашего прокси. Когда MediaPlayer делает запрос, используйте прокси-сервер для отправки эквивалентного запроса на фактический медиа-хост. Вы получите данные byte[] в методе packageReceived прокси-сервера, который я использую для создания HttpGet и отправки его в путь с помощью AndroidHttpClient.
Вы получите обратно HttpResponse и можете использовать HttpEntity внутри для доступа к данным потоковых байтов. Я использую ReadableByteChannel, вот так:
HttpEntityWrapper entity = (HttpEntityWrapper)response.getEntity();
ReadableByteChannel src = Channels.newChannel(entity.getContent());
Делайте с данными все, что захотите, когда читаете их обратно (например, кешируйте их в файл на SD-карте). Чтобы передать правильные данные в MediaPlayer, получите SocketChannel из клиентского сокета, сначала запишите заголовки ответа непосредственно в этот канал, а затем приступайте к записи байтовых данных объекта. Я использую NIO ByteBuffer в цикле while (клиент - это Socket, а буфер - это ByteBuffer).
int read, written;
SocketChannel dst = client.getChannel();
while (dst.isConnected() &&
dst.isOpen() &&
src.isOpen() &&
(read = src.read(buffer)) >= 0) {
try {
buffer.flip();
// This is one point where you can access the stream data.
// Just remember to reset the buffer position before trying
// to write to the destination.
if (buffer.hasRemaining()) {
written = dst.write(buffer);
// If the player isn't reading, wait a bit.
if (written == 0) Thread.sleep(15);
buffer.compact();
}
}
catch (IOException ex) {
// handle error
}
}
Возможно, вам придется изменить заголовок узла в ответе, прежде чем передать его проигрывателю, чтобы он выглядел так, как будто ваш прокси-сервер является отправителем, но я имею в виду запатентованную реализацию MediaPlayer, чтобы поведение могло быть немного другим. Надеюсь, это поможет.
Идея состоит в том, чтобы создать прокси-сервер, с которого медиаплеер может читать, а не читать данные непосредственно из Интернета.
Я использовал ht tps://github.com/danikula/AndroidVideoCache, который очень прост в сборке / использовании. Я использовал его для аудио, а не видео, но все равно.
Уже поздно, но я обнаружил, что большинству людей все еще нужно решение. Мое решение основано на DiskLruCache от JakeWharton. Нам нужны две вещи
AsyncTask для чтения файла или загрузки из сети и кэширования его
Обратный вызов для получения InputStram/FileDescriptor из кэша
Шаг 1:
import android.content.Context;
import android.os.AsyncTask;
import org.apache.commons.io.IOUtils;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
// you can use FileDescriptor as
// extends AsyncTask<String, Void, FileDescriptor>
public class AudioStreamWorkerTask extends AsyncTask<String, Void, FileInputStream> {
private OnCacheCallback callback = null;
private Context context = null;
public AudioStreamWorkerTask(Context context, OnCacheCallback callback) {
this.context = context;
this.callback = callback;
}
@Override
protected FileInputStream doInBackground(String... params) {
String data = params[0];
// Application class where i did open DiskLruCache
DiskLruCache cache = MyApplication.getDiskCache(context);
if (cache == null)
return null;
String key = hashKeyForDisk(data);
final int DISK_CACHE_INDEX = 0;
long currentMaxSize = cache.getMaxSize();
float percentageSize = Math.round((cache.size() * 100.0f) / currentMaxSize);
if (percentageSize >= 90) // cache size reaches 90%
cache.setMaxSize(currentMaxSize + (10 * 1024 * 1024)); // increase size to 10MB
try {
DiskLruCache.Snapshot snapshot = cache.get(key);
if (snapshot == null) {
Log.i(getTag(), "Snapshot is not available downloading...");
DiskLruCache.Editor editor = cache.edit(key);
if (editor != null) {
if (downloadUrlToStream(data, editor.newOutputStream(DISK_CACHE_INDEX)))
editor.commit();
else
editor.abort();
}
snapshot = cache.get(key);
} else
Log.i(getTag(), "Snapshot found sending");
if (snapshot != null)
return (FileInputStream) snapshot.getInputStream(DISK_CACHE_INDEX);
} catch (IOException e) {
e.printStackTrace();
}
Log.i(getTag(), "File stream is null");
return null;
}
@Override
protected void onPostExecute(FileInputStream fileInputStream) {
super.onPostExecute(fileInputStream);
if (callback != null) {
if (fileInputStream != null)
callback.onSuccess(fileInputStream);
else
callback.onError();
}
callback = null;
context = null;
}
public boolean downloadUrlToStream(String urlString, OutputStream outputStream) {
HttpURLConnection urlConnection = null;
try {
final URL url = new URL(urlString);
urlConnection = (HttpURLConnection) url.openConnection();
InputStream stream = urlConnection.getInputStream();
// you can use BufferedInputStream and BufferOuInputStream
IOUtils.copy(stream, outputStream);
IOUtils.closeQuietly(outputStream);
IOUtils.closeQuietly(stream);
Log.i(getTag(), "Stream closed all done");
return true;
} catch (final IOException e) {
e.printStackTrace();
} finally {
if (urlConnection != null)
IOUtils.close(urlConnection);
}
return false;
}
private String getTag() {
return getClass().getSimpleName();
}
private String hashKeyForDisk(String key) {
String cacheKey;
try {
final MessageDigest mDigest = MessageDigest.getInstance("MD5");
mDigest.update(key.getBytes());
cacheKey = bytesToHexString(mDigest.digest());
} catch (NoSuchAlgorithmException e) {
cacheKey = String.valueOf(key.hashCode());
}
return cacheKey;
}
private String bytesToHexString(byte[] bytes) {
// http://stackru.com/questions/332079
StringBuilder sb = new StringBuilder();
for (byte aByte : bytes) {
String hex = Integer.toHexString(0xFF & aByte);
if (hex.length() == 1)
sb.append('0');
sb.append(hex);
}
return sb.toString();
}
}
Шаг 2:
public interface OnCacheCallback {
void onSuccess(FileInputStream stream);
void onError();
}
пример
final String path = "http://www.example.com/test.mp3";
new AudioStreamWorkerTask (TestActivity.this, new OnCacheCallback() {
@Override
public void onSuccess(FileInputStream fileInputStream) {
Log.i(getClass().getSimpleName() + ".MediaPlayer", "now playing...");
if (fileInputStream != null) {
// reset media player here if necessary
mediaPlayer = new MediaPlayer();
try {
mediaPlayer.setDataSource(fileInputStream.getFD());
mediaPlayer.prepare();
mediaPlayer.setVolume(1f, 1f);
mediaPlayer.setLooping(false);
mediaPlayer.start();
fileInputStream.close();
} catch (IOException | IllegalStateException e) {
e.printStackTrace();
}
} else {
Log.e(getClass().getSimpleName() + ".MediaPlayer", "fileDescriptor is not valid");
}
}
@Override
public void onError() {
Log.e(getClass().getSimpleName() + ".MediaPlayer", "Can't play audio file");
}
}).execute(path);
Замечания:
Это проверенный, но грубый пример для кэширования аудиофайлов, могут возникнуть проблемы, если вы найдете что-нибудь, пожалуйста, сообщите мне:)