IntentService работает через список, который может быть переупорядочен одновременно через onHandleIntent - PullRequest
0 голосов
/ 24 апреля 2011

Я использую IntentService для загрузки 200 больших JPG из списка.Во время загрузки пользователь может пропустить незагруженные файлы JPG и загрузить, например, JPG # 156, но после загрузки он должен продолжить загрузку остальных.Так что это похоже на Lazy Loader ... но он продолжается, когда он простаивает.

Я ранее использовал onHandleIntent и поместил цикл с # 1 на # 200 ... который, очевидно, не работает, когда я пытаюсь отправитьдругой вызов IntentService для JPG # 156.Таким образом, вызов к # 156 происходит только после того, как onHandleIntent выполняется с # 200.

Затем я изменил его так, чтобы onHandleIntent переупорядочил запрос # 156 на верхнюю часть списка, затем запросил верхнюю часть списка (изагружает JPG), затем удаляет его из списка.Затем он снова вызывает IntentService, что звучит довольно рискованно из-за рекурсивного переполнения стека.Иногда это работает, и я вижу, что файл # 156 помещается первым ... иногда.

Есть ли лучший способ сделать это?Я мог бы подумать о том, чтобы запустить все это через базу данных.

РЕДАКТИРОВАТЬ: Это то, что я придумал:

code

public class PBQDownloader extends IntentService {
    int currentWeight = 0;
    PriorityBlockingQueue<WeightedAsset> pbQueue = new PriorityBlockingQueue<WeightedAsset>(100, new CompareWeightedAsset());
    public PBQDownloader() {
        super("PBQDownloader");
    }
    public PBQDownloader(String name) {
        super(name);
    }
    @Override
    protected void onHandleIntent(Intent intent) {
        String downloadUrl = "-NULL-";
        Bundle extras = intent.getExtras();
        if (extras!=null) {
            downloadUrl = extras.getString("url");
            Log.d("onHandleIntent 1.1", "asked to download: " + downloadUrl);
        } else {
            Log.d("onHandleIntent 1.2", "no URL sent so let's start queueing everything");
            int MAX = 10;
            for (int i = 1; i <= MAX; i++) {
                // should read URLs from list
                WeightedAsset waToAdd = new WeightedAsset("url: " + i, MAX - i);
                if (pbQueue.contains(waToAdd)) {
                    Log.d("onStartCommand 1", downloadUrl + " already exists, so we are removing it and adding it back with a new priority");
                    pbQueue.remove(waToAdd); 
                }
                pbQueue.put(waToAdd);
            }
            currentWeight = MAX + 1;
        }
        while (!pbQueue.isEmpty()) {
            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
            }
            WeightedAsset waToProcess = pbQueue.poll();
            Log.d("onHandleIntent 2 DOWNLOADED", waToProcess.url);
        }
        Log.d("onHandleIntent 99", "finished all IntentService calls");
    }
    @Override
    public int onStartCommand(Intent intent, int a, int b) {
        super.onStartCommand(intent, a, b);
        currentWeight++;
        String downloadUrl = "-NULL-";
        Bundle extras = intent.getExtras();
        if (extras!=null) downloadUrl = extras.getString("url");
        Log.d("onStartCommand 0", "download: " + downloadUrl + " with current weight: " + currentWeight);
        WeightedAsset waToAdd = new WeightedAsset(downloadUrl, currentWeight);
        if (pbQueue.contains(waToAdd)) {
            Log.d("onStartCommand 1", downloadUrl + " already exists, so we are removing it and adding it back with a new priority");
            pbQueue.remove(waToAdd); 
        }
        pbQueue.put(waToAdd);
        return 0;
    }
    private class CompareWeightedAsset implements Comparator<WeightedAsset> {
        @Override
        public int compare(WeightedAsset a, WeightedAsset b) {
            if (a.weight < b.weight) return 1;
            if (a.weight > b.weight) return -1;
            return 0;
        }
    }
    private class WeightedAsset {
        String url;
        int weight;
        public WeightedAsset(String u, int w) {
            url = u;
            weight = w;
        }
    }
}

code

Тогда у меня есть это занятие:

code

public class HelloPBQ extends Activity {
    int sCount = 10;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        Button tv01 = (Button) findViewById(R.id.tv01);
        Button tv02 = (Button) findViewById(R.id.tv02);
        Button tv03 = (Button) findViewById(R.id.tv03);
        tv01.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                doPBQ();
            }
        });
        tv02.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                doInitPBQ();
            }
        });
        tv03.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                sCount = 0;
            }
        });
    }

    private void doInitPBQ() {
        Intent intent = new Intent(getApplicationContext(), PBQDownloader.class);
        //intent.putExtra("url", "url: " + sCount);
        startService(intent);
    }    
    private void doPBQ() {
        sCount++;
        Intent intent = new Intent(getApplicationContext(), PBQDownloader.class);
        intent.putExtra("url", "url: " + sCount);
        startService(intent);
    }
}

code

Теперь беспорядочный бит в том, что янужно держать постоянно увеличивающийся счетчик, который рискует выйти за пределы int (WeightedAsset.weight) - есть ли способ программно добавить в очередь, и он автоматически станет главой очереди?Я попытался заменить WeightedAsset на String, но он не опрашивал (), как мне хотелось, в качестве FIFO вместо стека LIFO.

1 Ответ

2 голосов
/ 24 апреля 2011

Вот как я бы попробовал сначала:

Шаг # 1: Удерживайте IntentService на PriorityBlockingQueue.

Шаг # 2: Итерируйте onHandleIntent()PriorityBlockingQueue, загружая каждый файл по очереди, когда он извлекается из очереди.

Шаг # 3: onStartCommand() проверяет, является ли команда командой «сбросить все загрузки» (в этом случае,цепочка к суперклассу).Если вместо этого это команда «установить приоритет этой загрузки», измените приоритет этой записи в PriorityBlockingQueue, поэтому она будет выбрана следующей onHandleIntent() по окончании текущей загрузки.

...