Форум программистов, компьютерный форум CyberForum.ru

Не могу разобраться с потоками - Android

Восстановить пароль Регистрация
 
JIyrApy
31 / 31 / 1
Регистрация: 22.08.2010
Сообщений: 242
07.04.2013, 18:55     Не могу разобраться с потоками #1
Здравствуйте. Никак не могу понять принцип выноса задач в отдельный поток. Смысл прост:
1)есть метод который обновляет адаптер для ListView (назовем метод updateAdapter - делает запрос к БД, обновляет массив и обновляет адаптер, привязанный к листвью) - собственно обновляется и сам лситВью.
2)есть обычный progressDialog - который по логике должен запускаться (progressDialog.show()) перед updateAdapter и закрываться после обновления адаптера (progressDialog.dismiss());
Главное чтобы код притормаживал на момент обновления адаптера, то есть
...код...
запускаем прогрессДиалог \
запускаем обновление адаптера | в это время код программы дальше не выполняется
закрываем прогрессдиалог /
...код...

(данный кусок кода вызываю в onResume() - чтобы обновлять при каждом возвращении к активности)
Вопрос в том - как это лучше реализовать? Понятно, что нужно запускать отдельный поток, в котором и нужно запускать обновление адаптера. Но как это лучше сделать? Можно хотя бы пример?

PS пробовал через AsyncTask - иногда приложение вылетает, ругаясь, что типа адаптер был изменен, но не было сообщено листВью. Создается впечатление, что код продолжает выполняться при запуске второго потока, когда он еще не успел обновить адаптер. Или я что то не понимаю

AsyncTask такой:
Java
1
2
3
4
5
6
7
8
9
10
11
12
protected void onPreExecute() {
             прогрессДиалог.show();
        }
 
        protected Void doInBackground(Void... params) {
                  обновляем массив
}
protected void onPostExecute(Void result) {
            адаптер.notifyDataSetChanged();
            прогрессДиалог.dismiss();
    
        }
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
V0v1k
1157 / 981 / 1
Регистрация: 28.06.2012
Сообщений: 3,462
07.04.2013, 18:59     Не могу разобраться с потоками #2
Цитата Сообщение от JIyrApy Посмотреть сообщение
Создается впечатление, что код продолжает выполняться при запуске второго потока
да, так и есть.
JIyrApy
31 / 31 / 1
Регистрация: 22.08.2010
Сообщений: 242
07.04.2013, 19:18  [ТС]     Не могу разобраться с потоками #3
тогда как можно приостановить выполнение основного потока. Ну то есть появился у меня прогрессДиалог, запустился второй поток, а первый поток ждет, пока второй не исполнится и не закроется прогресс диалог.
V0v1k
1157 / 981 / 1
Регистрация: 28.06.2012
Сообщений: 3,462
07.04.2013, 20:29     Не могу разобраться с потоками #4
можно, но тогда все повиснет и через 5 сек у пользователя появится диалог "приложение не отвечает. завершить? да\нет".
в том то и суть потоков, чтобы один не ждал другого.
JIyrApy
31 / 31 / 1
Регистрация: 22.08.2010
Сообщений: 242
07.04.2013, 20:36  [ТС]     Не могу разобраться с потоками #5
Хм... Пытался реализовать такую структуру, уже без второго потока, тупо по схеме:
-запускаю прогресдиалог
-запускаю метод обновления адаптера
-убираю диалог
В результат прогресдиалог не показывается. Думал - может не успевает, поставил специально большой цикл - приложение зависает на время выполнения цикла, но прогрессдиалог не показывает. Почему?
V0v1k
1157 / 981 / 1
Регистрация: 28.06.2012
Сообщений: 3,462
07.04.2013, 20:38     Не могу разобраться с потоками #6
потому что вы тормозите UI поток своим циклом.
JIyrApy
31 / 31 / 1
Регистрация: 22.08.2010
Сообщений: 242
07.04.2013, 20:56  [ТС]     Не могу разобраться с потоками #7
Вот основная реализация класса активности. Еще хочу добавить, что в эту активность могут возвращаться из др активностей либо по методу finish() дочерней активности, либо по прямой загрузке через интент с указанием этой активности.
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
private ListView collectionList;
    Context context;
    private ArrayList<MyCollections> collectionItems;
    private CollectionItemAdapter aa;
    
    public CollectorDBAdapter collectorDBAdapter;
    Cursor collectionListCursor;
        ProgressDialog progressUpd;
 
@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        context=getApplicationContext();        
        collectionList=(ListView)findViewById(R.id.collectionList);     
        
        // создаем соединение с БД
        collectorDBAdapter = new CollectorDBAdapter(this);
            collectorDBAdapter.open();     
        collectionItems = new ArrayList<MyCollections>();       
                int resID=R.layout.collectionlist;
               // адаптер для привязывания к ListView
              aa = new CollectionItemAdapter(this, resID, collectionItems);
              // Привяжите адаптер с массивом к ListView.
             collectionList.setAdapter(aa);
        
        // обрабатываем нажатие на элемент списка
        
              collectionList.setOnItemClickListener(new OnItemClickListener() {
        @Override
            public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
                final MyCollections curC = aa.getItem(position);
                Intent intent = new Intent(Collector.this, TabCol.class);
                intent.putExtra("curId",curC.getIdx());
                intent.putExtra("typeC", 1);
                startActivity(intent);
 
            }
            
        });
        registerForContextMenu(collectionList);
        collectionList.setFastScrollEnabled(true);
    }
 
 
     protected void onResume() {
        
        super.onResume();
    
                 new newListTask().execute(new Void[]{});
        if(collectionListCursor.getCount()<=0){
            // иначе выводим сообщение о пустом списке и предлагаем добавить новую коллекцию
            emptyCollection();
        }
        
    }
 
 
void getProgressUpd(CharSequence stt){
        progressUpd = new ProgressDialog(this);
                 progressUpd.setCanceledOnTouchOutside(false);
        progressUpd.setCancelable(false);
        progressUpd.setTitle(R.string.app_name);
        progressUpd.setMessage(stt);
        progressUpd.show();
    }
    class newListTask extends AsyncTask<Void, Void, Void>{
 
 
        @Override
        protected void onPreExecute() {
            collectionListCursor = collectorDBAdapter.getAllCollectionsItemsCursor();
            getProgressUpd(getText(R.string.updatinglist));
        }
 
        @Override
        protected Void doInBackground(Void... params) {
 
            collectionItems.clear();
            if (collectionListCursor.moveToFirst()){
 
                do {
                    String title = collectionListCursor.getString(collectionListCursor.getColumnIndex(CollectorDBAdapter.TABLE_COLLECTIONS_TITLE));
                    long created = collectionListCursor.getLong(collectionListCursor.getColumnIndex(CollectorDBAdapter.TABLE_COLLECTIONS_CREATION_DATE));
                    long update = collectionListCursor.getLong(collectionListCursor.getColumnIndex(CollectorDBAdapter.TABLE_COLLECTIONS_UPDATE_DATE));
                    long idx = collectionListCursor.getLong(collectionListCursor.getColumnIndex(CollectorDBAdapter.TABLE_COLLECTIONS_ID));
                    String descdesc = collectionListCursor.getString(collectionListCursor.getColumnIndex(CollectorDBAdapter.TABLE_COLLECTIONS_DESCRIPTION));
                    MyCollections newItem = new MyCollections(title, idx, new Date(created), new Date(update), descdesc);
                    collectionItems.add(newItem);
                    publishProgress(new Void[]{});
 
 
                } while(collectionListCursor.moveToNext());
 
             }
            
            return null;
        }
        
 
 
        @Override
        protected void onPostExecute(Void result) {
 
            aa.notifyDataSetChanged();
            progressUpd.dismiss();
        
                        
        }
        
    }
Проблема в том, что иногда появляется ошибка при возвращении в эту активность
java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. [in ListView(2130968678, class android.widget.ListView) with Adapter(class com.app.collector.CollectionItemAdapter)]
Что может быть не так?
V0v1k
1157 / 981 / 1
Регистрация: 28.06.2012
Сообщений: 3,462
07.04.2013, 21:05     Не могу разобраться с потоками #8
нельзя менять UI елементы с не UI потока. для этого и сделалм асинктаски, хендлены и метод runOnUiThread

Добавлено через 6 минут
наполняйте отдельный ArrayList новыми айтемами, а в onPostExecute например
Java
1
2
3
4
collectionItems.clear();
collectionItems.addAll(newList);
aa.notifyDataSetChanged();
progressUpd.dismiss();
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
07.04.2013, 21:38     Не могу разобраться с потоками
Еще ссылки по теме:

Не могу разобраться в логе ошибки Android
Android Не могу разобраться с вёрсткой
не могу разобраться . java . игровой движок. игры Android
Android Не могу разобраться в логах, где ошибка?
Android Не могу разобраться с разметкой xml - layout'a

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
JIyrApy
31 / 31 / 1
Регистрация: 22.08.2010
Сообщений: 242
07.04.2013, 21:38  [ТС]     Не могу разобраться с потоками #9
Отлично! То что доктор прописал! Спасибо, а то уже два дня голову ломаю
Yandex
Объявления
07.04.2013, 21:38     Не могу разобраться с потоками
Ответ Создать тему
Опции темы

Текущее время: 05:00. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru