2011-07-13 5 views
3

私はアンドロイド用のアプリケーションを実装していますが、インターネットからのデータをリストビューに読み込みながらメモリの問題にぶつかります。発生する奇妙なエラーは、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が正しく動作するようなものがありませんか? 過去に誰かがそれに遭遇しましたか?

お時間をいただきありがとうございます。

+0

解決策はわかりませんが、ListActivityはアクティビティを制限しているので、ListActivityに他のものをすべて追加することもできます。 – dbrettschneider

+0

私が理解しているように、listview 'recycling'は画面外を移動したビューの再利用です。それで、2番目のログがどのように問題を示しているのか分かりません。リストに50個のアイテムを追加しているので、最悪のシナリオは50個のビューになります。画面の価値について(私は推測している)6に相当します。確実に確認するには、アダプターが新しいビューを作成するたびにリファレンスを保存してみてください。そうすることで、アダプターが作成したビューの総数が得られます。 –

答えて

1

ActivityListActivityの両方が問題ありません。私はあなたがmContext参照を通して活動全体を漏らしていると信じています。それを取り除き、何が起こるかを見てください!