私はアンドロイド用のアプリケーションを実装していますが、インターネットからのデータをリストビューに読み込みながらメモリの問題にぶつかります。発生する奇妙なエラーは、Google I/O 2009に記載されているように、recycling of the rows as it should normally happenの欠如によって引き起こされたようです。ListActivityとActivityの奇妙な振る舞い
http://android.amberfog.com/?p=296でコードを実行すると、すべてがスムーズに実行され、行ビューがリサイクルされ、listViewが最適に使用されます。
もっと多くのものが内部にある別のアクティビティ内でlistViewを使用したいので、ListActivityだけを拡張したクラスでは不十分です。だから、私は次のコードを持って活動を持っている:
public class MultipleItemsList extends Activity {
private Context mContext;
private MyCustomAdapter mAdapter;
private ListView listView;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mContext = this;
listView = (ListView) findViewById(R.id.listView);
mAdapter = new MyCustomAdapter(mContext);
for (int i = 1; i < 50; i++) {
mAdapter.addItem("item " + i);
}
listView.setAdapter(mAdapter);
}
}
main.xmlも何も空想です:以前のリンクが提案していることを
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/hello"/>
<ListView
android:id="@+id/listView"
android:layout_width="fill_parent"
android:layout_height="wrap_content"/>
</LinearLayout>
ListActivityは次のとおりです。
public class MultipleItemsList extends ListActivity {
private Context mContext;
private MyCustomAdapter mAdapter;
private ListView listView;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mContext = this;
mAdapter = new MyCustomAdapter(mContext);
for (int i = 1; i < 50; i++) {
mAdapter.addItem("item " + i);
}
setListAdapter(mAdapter);
}
}
アダプタはどちらの場合も以下の通りです:
public class MyCustomAdapter extends BaseAdapter {
private static final int TYPE_ITEM = 0;
private static final int TYPE_SEPARATOR = 1;
private static final int TYPE_MAX_COUNT = TYPE_SEPARATOR + 1;
private ArrayList<String> mData = new ArrayList<String>();
private LayoutInflater mInflater;
private TreeSet<Integer> mSeparatorsSet = new TreeSet<Integer>();
public static class ViewHolder {
public TextView textView;
}
public MyCustomAdapter(Context context) {
mInflater = LayoutInflater.from(context);
}
public void addItem(final String item) {
mData.add(item);
notifyDataSetChanged();
}
public void addSeparatorItem(final String item) {
mData.add(item);
// save separator position
mSeparatorsSet.add(mData.size() - 1);
notifyDataSetChanged();
}
@Override
public int getItemViewType(int position) {
return mSeparatorsSet.contains(position) ? TYPE_SEPARATOR : TYPE_ITEM;
}
@Override
public int getViewTypeCount() {
return TYPE_MAX_COUNT;
}
@Override
public int getCount() {
return mData.size();
}
@Override
public String getItem(int position) {
return mData.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
int type = getItemViewType(position);
Log.d("myCA", "getView " + position + " " + convertView + " type = " + type);
if (convertView == null) {
holder = new ViewHolder();
switch (type) {
case TYPE_ITEM:
convertView = mInflater.inflate(R.layout.item1, null);
holder.textView = (TextView)convertView.findViewById(R.id.text);
break;
case TYPE_SEPARATOR:
convertView = mInflater.inflate(R.layout.item2, null);
holder.textView = (TextView)convertView.findViewById(R.id.textSeparator);
break;
}
convertView.setTag(holder);
} else {
holder = (ViewHolder)convertView.getTag();
}
holder.textView.setText(mData.get(position));
return convertView;
}
}
私の質問は、ListActivityの2番目のケースでは、ビューのリサイクルは問題なく行われますが、アクティビティの実装時にリサイクルは失敗するのはなぜですか?
ログListActivityを使用する場合、すべてが正しくリサイクルされていることを示しています
07-13 10:14:25.277: DEBUG/myCA(2336): getView 0 null type = 0
07-13 10:14:25.277: DEBUG/myCA(2336): getView 1 null type = 0
07-13 10:14:25.277: DEBUG/myCA(2336): getView 2 null type = 0
07-13 10:14:25.277: DEBUG/myCA(2336): getView 3 null type = 0
07-13 10:14:25.287: DEBUG/myCA(2336): getView 4 null type = 0
07-13 10:14:25.287: DEBUG/myCA(2336): getView 5 null type = 0
07-13 10:14:27.887: DEBUG/myCA(2336): getView 6 null type = 0
07-13 10:14:28.047: DEBUG/myCA(2336): getView 7 [email protected] type = 0
07-13 10:14:28.267: DEBUG/myCA(2336): getView 8 [email protected] type = 0
...
...
...
活動の私の実装では、私は、以下を参照してください。
に問題が明確に存在し07-13 10:11:47.517: DEBUG/myCA(2296): getView 0 null type = 0
07-13 10:11:47.517: DEBUG/myCA(2296): getView 1 [email protected] type = 0
07-13 10:11:47.517: DEBUG/myCA(2296): getView 2 [email protected] type = 0
07-13 10:11:47.517: DEBUG/myCA(2296): getView 3 [email protected] type = 0
07-13 10:11:47.527: DEBUG/myCA(2296): getView 4 [email protected] type = 0
07-13 10:11:47.527: DEBUG/myCA(2296): getView 5 [email protected] type = 0
07-13 10:11:47.547: DEBUG/myCA(2296): getView 0 [email protected] type = 0
07-13 10:11:47.547: DEBUG/myCA(2296): getView 1 null type = 0
07-13 10:11:47.547: DEBUG/myCA(2296): getView 2 null type = 0
07-13 10:11:47.557: DEBUG/myCA(2296): getView 3 null type = 0
...
...
...
リサイクル。
親アクティビティが単純なアクティビティであり、ListActivityでない場合、listViewが正しく動作するようなものがありませんか? 過去に誰かがそれに遭遇しましたか?
お時間をいただきありがとうございます。
解決策はわかりませんが、ListActivityはアクティビティを制限しているので、ListActivityに他のものをすべて追加することもできます。 – dbrettschneider
私が理解しているように、listview 'recycling'は画面外を移動したビューの再利用です。それで、2番目のログがどのように問題を示しているのか分かりません。リストに50個のアイテムを追加しているので、最悪のシナリオは50個のビューになります。画面の価値について(私は推測している)6に相当します。確実に確認するには、アダプターが新しいビューを作成するたびにリファレンスを保存してみてください。そうすることで、アダプターが作成したビューの総数が得られます。 –