Музыкальный сервис всегда возвращает ноль

Я сделал список музыкальных файлов и сервис для воспроизведения музыки в фоновом режиме, но когда я нажимаю на музыку из списка, она всегда возвращает ноль

У меня никогда раньше не было работы с музыкальным сервисом, так что, может быть, я ошибаюсь, я пропускаю неправильную позицию в приемной или что-то еще?

оказание услуг

public class MusicService extends Service implements
        MediaPlayer.OnPreparedListener, MediaPlayer.OnErrorListener,
        MediaPlayer.OnCompletionListener {

    //media player
    private MediaPlayer player;
    //song list
    private ArrayList<SongModel> songs;
    //current position
    private int songPosn;
    //binder
    private final IBinder musicBind = new MusicBinder();
    //title of current song
    private String songTitle = "";
    //notification id
    private static final int NOTIFY_ID = 1;
    //shuffle flag and random
    private boolean shuffle = false;
    private Random rand;

    public void onCreate() {
        //create the service
        super.onCreate();
        //initialize position
        songPosn = 0;
        //random
        rand = new Random();
        //create player
        player = new MediaPlayer();
        //initialize
        initMusicPlayer();
    }

    public void initMusicPlayer() {
        //set player properties
        player.setWakeMode(getApplicationContext(),
                PowerManager.PARTIAL_WAKE_LOCK);
        player.setAudioStreamType(AudioManager.STREAM_MUSIC);
        //set listeners
        player.setOnPreparedListener(this);
        player.setOnCompletionListener(this);
        player.setOnErrorListener(this);
    }

    //pass song list
    public void setList(ArrayList<SongModel> theSongs) {
        songs = theSongs;
    }

    //binder
    public class MusicBinder extends Binder {
        public MusicService getService() {
            return MusicService.this;
        }
    }

    //activity will bind to service
    @Override
    public IBinder onBind(Intent intent) {
        return musicBind;
    }

    //release resources when unbind
    @Override
    public boolean onUnbind(Intent intent) {
        player.stop();
        player.release();
        return false;
    }

    //play a song
    public void playSong() {
        //play
        player.reset();
        //get song
        SongModel playSong = songs.get(songPosn);
        //get title
        songTitle = playSong.getSongTitle();
        //get id
        long currSong = playSong.getmSongID();
        //set uri
        Uri trackUri = ContentUris.withAppendedId(
                android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
                currSong);
        //set the data source
        try {
            player.setDataSource(getApplicationContext(), trackUri);
        } catch (Exception e) {
            Log.e("MUSIC SERVICE", "Error setting data source", e);
        }
        player.prepareAsync();
    }

    //set the song
    public void setSong(int songIndex) {
        Log.e("click", " service" + songIndex);
        for (int i = 0; i < songs.size(); i++) {
            Log.e("click", songs.get(i).getArtistname() + "  " + songs.get(i).getmSongID());
        }
        songPosn = songIndex;
    }

    @Override
    public void onCompletion(MediaPlayer mp) {
        //check if playback has reached the end of a track
        if (player.getCurrentPosition() > 0) {
            mp.reset();
            playNext();
        }
    }

    @Override
    public boolean onError(MediaPlayer mp, int what, int extra) {
        Log.v("MUSIC PLAYER", "Playback Error");
        mp.reset();
        return false;
    }

    @Override
    public void onPrepared(MediaPlayer mp) {
        //start playback
        mp.start();
        //notification
        Intent notIntent = new Intent(this, MainActivity.class);
        notIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        PendingIntent pendInt = PendingIntent.getActivity(this, 0,
                notIntent, PendingIntent.FLAG_UPDATE_CURRENT);

        Notification.Builder builder = new Notification.Builder(this);

        builder.setContentIntent(pendInt)
                .setSmallIcon(R.mipmap.ic_play_circle_filled_black_24dp)
                .setTicker(songTitle)
                .setOngoing(true)
                .setContentTitle("Playing")
                .setContentText(songTitle);
        Notification not = builder.build();
        startForeground(NOTIFY_ID, not);
    }

    //playback methods
    public int getPosn() {
        return player.getCurrentPosition();
    }

    public int getDur() {
        return player.getDuration();
    }

    public boolean isPng() {
        return player.isPlaying();
    }

    public void pausePlayer() {
        player.pause();
    }

    public void seek(int posn) {
        player.seekTo(posn);
    }

    public void go() {
        player.start();
    }

    //skip to previous track
    public void playPrev() {
        songPosn--;
        if (songPosn < 0) songPosn = songs.size() - 1;
        playSong();
    }

    //skip to next
    public void playNext() {
        if (shuffle) {
            int newSong = songPosn;
            while (newSong == songPosn) {
                newSong = rand.nextInt(songs.size());
            }
            songPosn = newSong;
        } else {
            songPosn++;
            if (songPosn >= songs.size()) songPosn = 0;
        }
        playSong();
    }

    @Override
    public void onDestroy() {
        stopForeground(true);
    }

    //toggle shuffle
    public void setShuffle() {
        if (shuffle) shuffle = false;
        else shuffle = true;
    }}

класс деятельности

public static MusicService musicSrv;
    private Intent playIntent;
    private boolean musicBound=false;

    public static ArrayList<SongModel> songList;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        load();
    @Override
    protected void onStart() {
        super.onStart();
        if(playIntent==null){
            playIntent = new Intent(this, MusicService.class);
            bindService(playIntent, musicConnection, Context.BIND_AUTO_CREATE);
            startService(playIntent);
        }
    }

private ServiceConnection musicConnection = new ServiceConnection() {

    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        MusicService.MusicBinder binder = (MusicService.MusicBinder) service;
        //get service
        musicSrv = binder.getService();
        //pass list
        musicSrv.setList(songList);
        musicBound = true;
    }

    @Override
    public void onServiceDisconnected(ComponentName name) {
        musicBound = false;
    }
};

    private void initLayout() {
        final Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
        final String[] cursor_cols = {MediaStore.Audio.Media._ID,
                MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.ALBUM,
                MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.DATA,
                MediaStore.Audio.Media.ALBUM_ID,
                MediaStore.Audio.Media.DURATION};
        final String where = MediaStore.Audio.Media.IS_MUSIC + "=1";
        final Cursor cursor = getContentResolver().query(uri,
                cursor_cols, where, null, null);


        while (cursor.moveToNext()) {
            String artist = cursor.getString(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.ARTIST));
            String album = cursor.getString(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.ALBUM));
            String track = cursor.getString(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.TITLE));
            String data = cursor.getString(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
            Long albumId = cursor.getLong(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.ALBUM_ID));

            int duration = cursor.getInt(cursor
                    .getColumnIndexOrThrow(MediaStore.Audio.Media.DURATION));

            Uri sArtworkUri = Uri
                    .parse("content://media/external/audio/albumart");
            Uri albumArtUri = ContentUris.withAppendedId(sArtworkUri, albumId);

            Log.e("art", albumArtUri.toString());
            /*Bitmap bitmap = null;
            try {
                bitmap = MediaStore.Images.Media.getBitmap(
                        getActivity().getContentResolver(), albumArtUri);
                bitmap = Bitmap.createScaledBitmap(bitmap, 30, 30, true);

            } catch (FileNotFoundException exception) {
                exception.printStackTrace();
                bitmap = BitmapFactory.decodeResource(getActivity().getResources(),
                        R.mipmap.ic_music_note_black_24dp);
            } catch (IOException e) {

                e.printStackTrace();
            }*/

            SongModel audioListModel = new SongModel();
            audioListModel.setArtistname(artist);
            //  audioListModel.setBitmap(bitmap);
            audioListModel.setmSongTitle(album);
            audioListModel.setmSongTitle(track);
            audioListModel.setData(data);
            audioListModel.setmSongID(albumId);
            audioListModel.setSongLength(String.valueOf(duration));
            audioListModel.setUri(String.valueOf(albumArtUri));
            Log.e("data", "artist :" + artist + "  album  " + album + "  track  " + track + "  data  " + data + "  " + "albumId  " + albumId
                    + "  " + "duration  " + duration + " art:  " + albumArtUri);

            songList.add(audioListModel);

        }

    }


    private class GetAudioListAsynkTask extends AsyncTask<Void, Void, Boolean> {

        private Context context;

        public GetAudioListAsynkTask(Context context) {

            this.context = context;
        }

        @Override
        protected void onPreExecute() {
            super.onPreExecute();

        }

        @Override
        protected Boolean doInBackground(Void... params) {
            try {
                initLayout();

                return true;
            } catch (Exception e) {
                return false;

            }

        }

        @Override
        protected void onPostExecute(Boolean result) {
            Collections.sort(songList, new Comparator<SongModel>() {
                public int compare(SongModel a, SongModel b) {
                    return a.getSongTitle().compareTo(b.getSongTitle());
                }
            });

        }
    }

    public void load() {
        songList = new ArrayList<>();
        new GetAudioListAsynkTask(this).execute((Void) null);


    }

класс адаптера

@Override
public void onBindViewHolder(final Holder holder, final int position) {
    final SongModel songModel = list.get(holder.getAdapterPosition());
    Glide.with(holder.itemView.getContext()).load(songModel.getUri()).error(R.mipmap.ic_music_note_black_24dp).into(holder.imageView);
    holder.imageView.setImageURI(Uri.parse(songModel.getUri()));
    holder.artistname.setText(songModel.getArtistname());
    holder.duration.setText(songModel.getSongLength());
    holder.songname.setText(songModel.getSongTitle());
    holder.itemView.setTag(position);

    holder.itemView.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            songPicked(position);
        }
    });


}
 private void songPicked(int position) {
    musicSrv.setSong(position);
    musicSrv.playSong();
}

журнал ошибок

    04-21 19:01:37.847 3983-29870/? E/DatabaseUtils: Writing exception to parcel
                                                 java.lang.SecurityException: Permission Denial: reading com.android.providers.media.MediaProvider uri content://media/external/audio/media/1 from pid=2608, uid=1000 requires android.permission.READ_EXTERNAL_STORAGE, or grantUriPermission()
                                                     at android.content.ContentProvider.enforceReadPermissionInner(ContentProvider.java:646)
                                                     at android.content.ContentProvider$Transport.enforceReadPermission(ContentProvider.java:493)
                                                     at android.content.ContentProvider$Transport.enforceFilePermission(ContentProvider.java:484)
                                                     at android.content.ContentProvider$Transport.openFile(ContentProvider.java:380)
                                                     at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:240)
                                                     at android.os.Binder.execTransact(Binder.java:453)


                                                 [ 04-21 19:01:37.847  2264:16266 D/         ]
                                                 openContentUri(content://media/external/audio/media/1) caught exception -1
04-21 19:01:37.847 2264-16266/? E/MediaPlayerService: Couldn't open fd for content://media/external/audio/media/1
04-21 19:01:37.847 19273-19273/? E/MediaPlayer: Unable to create media player
04-21 19:01:37.847 19273-19273/? E/MUSIC SERVICE: Error setting data source
                                                  java.io.IOException: setDataSource failed.: status=0x80000000
                                                      at android.media.MediaPlayer.nativeSetDataSource(Native Method)
                                                      at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1231)
                                                      at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1215)
                                                      at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1169)
                                                      at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1105)
                                                      at com.nowsip.musicplayer.service.MusicService.playSong(MusicService.java:118)
                                                      at com.nowsip.musicplayer.adapter.SongAdapter.songPicked(SongAdapter.java:86)
                                                      at com.nowsip.musicplayer.adapter.SongAdapter.access$000(SongAdapter.java:24)
                                                      at com.nowsip.musicplayer.adapter.SongAdapter$1.onClick(SongAdapter.java:52)
                                                      at android.view.View.performClick(View.java:5721)
                                                      at android.view.View$PerformClick.run(View.java:22620)
                                                      at android.os.Handler.handleCallback(Handler.java:739)
                                                      at android.os.Handler.dispatchMessage(Handler.java:95)
                                                      at android.os.Looper.loop(Looper.java:148)
                                                      at android.app.ActivityThread.main(ActivityThread.java:7409)
                                                      at java.lang.reflect.Method.invoke(Native Method)
                                                      at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1230)
                                                      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1120)
04-21 19:01:37.847 19273-19273/? E/MediaPlayer: prepareAsync called in state 1
04-21 19:01:37.847 19273-19273/? E/AndroidRuntime: FATAL EXCEPTION: main
                                                   Process: com.nowsip.musicplayer, PID: 19273
                                                   java.lang.IllegalStateException
                                                       at android.media.MediaPlayer.prepareAsync(Native Method)
                                                       at com.nowsip.musicplayer.service.MusicService.playSong(MusicService.java:122)
                                                       at com.nowsip.musicplayer.adapter.SongAdapter.songPicked(SongAdapter.java:86)
                                                       at com.nowsip.musicplayer.adapter.SongAdapter.access$000(SongAdapter.java:24)
                                                       at com.nowsip.musicplayer.adapter.SongAdapter$1.onClick(SongAdapter.java:52)
                                                       at android.view.View.performClick(View.java:5721)
                                                       at android.view.View$PerformClick.run(View.java:22620)
                                                       at android.os.Handler.handleCallback(Handler.java:739)
                                                       at android.os.Handler.dispatchMessage(Handler.java:95)
                                                       at android.os.Looper.loop(Looper.java:148)
                                                       at android.app.ActivityThread.main(ActivityThread.java:7409)
                                                       at java.lang.reflect.Method.invoke(Native Method)
                                                       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1230)
                                                       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1120)
04-21 19:01:37.887 19790-19790/? E/Zygote: v2
04-21 19:01:37.887 2608-19795/? E/android.os.Debug: ro.product_ship = true
04-21 19:01:37.887 2608-19795/? E/android.os.Debug: ro.debug_level = 0x4f4c
04-21 19:01:37.887 2608-19795/? E/android.os.Debug: sys.mobilecare.preload = false
04-21 19:01:37.897 19790-19790/? E/Zygote: accessInfo : 0
04-21 19:01:37.897 2608-2803/? E/Qmage: isQIO : stream is not a QIO file
04-21 19:01:38.457 11589-11589/? E/Qmage: isQIO : stream is not a QIO file
04-21 19:01:38.497 11589-11589/? E/Qmage: isQIO : stream is not a QIO file
04-21 19:01:38.497 11589-11589/? E/Qmage: isQIO : stream is not a QIO file

2 ответа

Подтвердите, что когда ваш onServiceConnected Вызов метода, и вы устанавливаете список песен в вашем сервисе, в то время ваш songList Прапрада или нет?

Я имею в виду, что ваш список песен может быть нулевым, когда вы звоните ниже строки.

    //pass list
    musicSrv.setList(songList);

В случае, когда услуга подключена до вашего initLayout Метод завершен.

Обновить

В соответствии с приведенной ниже строкой журналов ошибок вы должны добавить READ_EXTERNAL_STORAGE разрешение в вашем файле манифеста.

java.lang.SecurityException: Отказ в разрешении: чтение содержимого uri com.android.providers.media.MediaProvider://media/external/audio/media/1 из pid=2608, uid=1000 требует android.permission.READ_EXTERNAL_STORAGE или grantUriPermission()

Задайте источник данных, используя файл FileDescriptor:

//play a song
public void playSong() {
    //play
    player.reset();
    //get song
    SongModel playSong = songs.get(songPosn);
    //get title
    songTitle = playSong.getSongTitle();
    //get id
    long currSong = playSong.getmSongID();
    //set uri
    Uri trackUri = ContentUris.withAppendedId(
            android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
            currSong);
    //set the data source
    try {
        String filePath = getPathFromContentUri(context, trackUri);
        File file = new File(filePath);
        FileInputStream inputStream = new FileInputStream(file);    
        player.setDataSource(inputStream.getFD());
        inputStream.close();
    } catch (Exception e) {
        Log.e("MUSIC SERVICE", "Error setting data source", e);
    }
    player.prepareAsync();
}


public static String getPathFromContentUri(Context context, Uri contentUri) {
    String[] proj = { MediaStore.Audio.Media.DATA };
    Cursor cursor = context.getContentResolver().query(contentUri,
            proj, null, null, null);
    cursor.moveToFirst();

    String path = cursor.getString(cursor
            .getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));

    cursor.close();
    return path;
}

Проверьте этот ответ для получения дополнительной информации. Это может быть полезно для вас.

Это не проблема музыкального сервиса, это проблема разрешения

Во-первых, вам нужно дать имя сервиса для манифеста Android, как

После этого, если вы работаете с 6.0 или более, вам нужно запросить разрешение на запись во внешнее хранилище для чтения при первом запуске приложения в lollipop или kit kat или другом, которое дает разрешение во время установки, но в зефире или больше, которое вы даете вручную спросить пользователя

проверить это Ошибка разрешения хранения в Зефир

Другие вопросы по тегам