• ListView局部更新(非notifyDataSetChanged)


    package com.example.test;
    
    import java.util.ArrayList;
    import java.util.List;
    
    import android.app.Activity;
    import android.content.Context;
    import android.os.Bundle;
    import android.view.LayoutInflater;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.view.MotionEvent;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.view.ViewGroup;
    import android.view.ViewStub;
    import android.view.Window;
    import android.widget.BaseAdapter;
    import android.widget.Button;
    import android.widget.FrameLayout;
    import android.widget.ListView;
    import android.widget.TextView;
    import android.widget.Toast;
    
    //
    /**ListView性能优化</br>
     * 当为ListView中每一个条目中某一控件设置点击事件时。通常能够在Adapter中每次getView中为该控件setOnClickListener,</br>
     * 但ListView中条目是复用的,不是必需每次都设置setOnClickListener,仅仅需在getView中推断convertView是否为null,</br>
     * 若为null则为控件设置setOnClickListener,然后每次getView时都为该控件的某个特殊字段设置值为position。(例 : holder.tv.setText(position+"");),</br>
     * 当点击该控件时,取出设置的该值则就得到了点击的条目的位置,然后就能够获取整个条目(能够通过多次调用点击的控件的getParent()方法就可以得到被点击的控</br>
     * 件所在的条目布局,或者通过listview.getChildAt(该条目的位置减去-listview.getFirstVisiblePosition())也</br>
     * 能够获取被点击的控件所在的条目。这样就能够仅仅改动该条目的部分信息了)</br></br>
     * 详见 getView代码
     * @author Young
     *
     */
    public class MainActivity extends Activity  {
    	@Override
    	protected void onCreate(Bundle savedInstanceState) {
    		// TODO Auto-generated method stub
    		super.onCreate(savedInstanceState);
    		
    		final ListView listview=new ListView(getApplicationContext());
    		final List<String>data=new ArrayList< >();
    		for (int i = 0; i < 50; i++) {
    			data.add("but "+i);
    		}
    		setContentView(listview);
    		
    		listview.setAdapter(new BaseAdapter() {
    			class ViewHolder{
    				TextView tv;
    				Button but;
    			}
    			
    			@Override
    			public View getView(final int position, View convertView, ViewGroup parent) {
    			 
    				if (convertView==null) {
    					final ViewHolder	holder=new ViewHolder();//该holder必须是局部final的。否则在以下的onClick中调用holder.tv得到的不是被点击的控件
    					
    					
    					convertView=LayoutInflater.from(getApplicationContext()).inflate(R.layout.list_item, parent,false);// ps:inflate中第二个參数是parent。否则条目高度是包裹内容,第三个參数一定是false
    					holder.but=(Button) convertView.findViewById(R.id.but);
    					holder.tv=(TextView) convertView.findViewById(R.id.tv);
    					
    					convertView.setTag(holder);
    					System.out.println("setOnClickListener-----");
    					holder.tv.setOnClickListener(new OnClickListener() {//仅仅需在这设置一次OnClickListener
    						
    						@Override
    						public void onClick(View v) {
    							
    							Toast.makeText(getApplicationContext(), holder.tv.getText(), 0).show();
    							System.out.println("点击位置  "+holder.tv.getText()  +"   firstVisizable "+listview.getFirstVisiblePosition());
    							
    							//该Demo仅仅是简单的把位置设置为了TextView的文字,实际使用中通常要设置其它属性为位置
    							int index=Integer.parseInt(holder.tv.getText().toString())-listview.getFirstVisiblePosition();
    							
    							ViewGroup viewGroup=(ViewGroup) listview.getChildAt(index);//获取点击的控件所在的条目布局
    							
    							String s=((TextView)viewGroup.getChildAt(0)).getText().toString();	
    							((Button)viewGroup.getChildAt(1)).setText("but-->"+s);//改动被点击控件所在条目布局中的还有一个控件的内容(不须要调用notifyDataSetChanged()来刷新整个可见区域)
    							
    							data.set(Integer.parseInt(holder.tv.getText().toString()), "but-->"+s);	//把数据存放到List中。以便下次滚动到该条目时正确显示					
    							System.out.println("index  "+index+"   "+s);
    						
    						}
    					});
    				}
    				ViewHolder	holder =(ViewHolder) convertView.getTag();
    				holder.tv.setText(position+"");//该Demo仅仅是简单的把位置设置为了TextView的文字,实际使用中通常要设置其它属性为位置(例:能够自己定义控件继承自TextView,然后为该控件添加一个专门记录位置的字段。或者直接setTag())
    				holder.but.setText(data.get(position) );//Button的内容事实上终于是依据data中信息来显示的
    				
    				
    				//主要优化以下这样的每次setOnClickListener,点击后通过notifyDataSetChanged来刷新整个可见区域
    //				holder.tv.setOnClickListener(new OnClickListener() {
    //					
    //					@Override
    //					public void onClick(View v) {
    //						// TODO Auto-generated method stub
    //						data.set(position, "but-->"+position);
    //						notifyDataSetChanged();
    //					}
    //				});
    				return convertView;
    			}
    			
    			@Override
    			public long getItemId(int position) {
    				// TODO Auto-generated method stub
    				return 0;
    			}
    			
    			@Override
    			public Object getItem(int position) {
    				// TODO Auto-generated method stub
    				return null;
    			}
    			
    			@Override
    			public int getCount() {
    				// TODO Auto-generated method stub
    				return data.size();
    			}
    			
    			
    			@Override
    			public void notifyDataSetChanged() {
    				// TODO Auto-generated method stub
    				super.notifyDataSetChanged();
    				System.out.println("notifyDataSetChanged====");//能够看到从没调用过该函数来刷新
    			}
    		});
    	}
     
    }
    
    </pre><pre name="code" class="java">
    <pre name="code" class="html"><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="60dp"
     	 android:background="#fff"
        android:orientation="horizontal" >
    
        <TextView android:id="@+id/tv"
            android:layout_gravity="center"
            
            android:layout_width="wrap_content"
            android:padding="15dp"
            android:layout_height="wrap_content"
            android:background="#eee"
            android:textColor="#ff0000"/>
       
        <Button android:id="@+id/but"
            android:layout_gravity="center"
            android:padding="15dp"
            android:layout_marginLeft="20dp"
            android:layout_width="wrap_content"
          	android:text="xx"
            android:layout_height="wrap_content"
        	android:background="#eee"
            android:textColor="#ff0000"/>
       
    </LinearLayout>
    


    
    

    能够看到有50个条目,但仅仅设置了8次Listener

    =============================================================

    方法二:

    package com.example.listviewtest;
    
    import java.util.ArrayList;
    import java.util.List;
    
    import android.app.Activity;
    import android.database.DataSetObserver;
    import android.os.Bundle;
    import android.view.LayoutInflater;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.view.ViewGroup;
    import android.view.ViewParent;
    import android.widget.AbsListView;
    import android.widget.BaseAdapter;
    import android.widget.ListAdapter;
    import android.widget.ListView;
    import android.widget.TextView;
    
    public class MainActivity extends Activity {
    
    	List<String>data;
    	ListView listView;
    	@Override
    	protected void onCreate(Bundle savedInstanceState) {
    		super.onCreate(savedInstanceState);
    		  listView=new ListView(getApplicationContext());
    		
    		setContentView(listView);
    		data=new ArrayList<>();
    		for (int i = 0; i < 5000; i++) {
    			data.add(""+i);
    		}
    		listView.setAdapter(new MyAdapter());
    	}
    	
    	
    	class MyAdapter extends BaseAdapter {
    
    		@Override
    		public int getCount() {
    			// TODO Auto-generated method stub
    			return data.size();
    		}
    
    		@Override
    		public Object getItem(int position) {
    			// TODO Auto-generated method stub
    			return null;
    		}
    
    		@Override
    		public long getItemId(int position) {
    			// TODO Auto-generated method stub
    			return 0;
    		}
    
    		@Override
    		public View getView(int position, View convertView, ViewGroup parent) {
    			// TODO Auto-generated method stub
    			if (convertView==null) {
    				  ViewHolder holder=new ViewHolder();
    				convertView=LayoutInflater.from(getApplicationContext()).inflate(R.layout.activity_main, parent,false);
    				holder.tv1=(TextView) convertView.findViewById(R.id.tv1);
    				holder.tv2=(TextView) convertView.findViewById(R.id.tv2);
    				holder.tv1.setOnClickListener(new OnClickListener() {
    					
    					@Override
    					public void onClick(View v) {
    //						int po=(int) holder.tv1.getTag();
    						
    						int po=getRootItemPosition(v, listView);
    						 ViewGroup group=(ViewGroup) listView.getChildAt(po-listView.getFirstVisiblePosition());
    						 ViewHolder holder=(ViewHolder) group.getTag();
    						 holder.tv2.setText(po+"--------------------->");
    						 data.set(po, po+"--------------------->");
    						 
    						
    					}
    				});
    				convertView.setTag(holder);
    			}
    			ViewHolder holder=(ViewHolder) convertView.getTag();
    			holder.tv2.setText(data.get(position));
    //			holder.tv1.setTag(position);
    			holder.tv1.setText(""+position);
    			return convertView;
    		}
    
    	 
    		public int getRootItemPosition(View v,AbsListView absListView){
    			ViewParent parent;
    			
    			while(! ((parent=v.getParent()) instanceof AbsListView) ){
    				v=(View) parent;
    			}
    			
    			int f=absListView.getFirstVisiblePosition();
    			int l=absListView.getLastVisiblePosition();
    			for (int i = f; i <= l; i++) {
     
    				if (absListView.getChildAt(i-f)==v) {
    					return i;
    				}
    			}
    			
    			return -1;
    		}
    		class ViewHolder{
    			TextView tv1;
    			TextView tv2;
    		}
    	}
    
    	
    }
    



    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        android:padding="10dp"
        tools:context="${relativePackage}.${activityClass}" >
    
        <FrameLayout 
            android:layout_width="wrap_content"
            android:layout_height="wrap_content">
            <LinearLayout 
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                >
                <RelativeLayout 
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content">
                    
                      <TextView android:id="@+id/tv1"
            android:background="#ccc"
            android:padding="15dp"
            android:textColor="#000"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="OK" />
                </RelativeLayout>
            </LinearLayout>
        </FrameLayout>
      
        <TextView android:id="@+id/tv2"
            android:padding="10dp"
            android:background="#ccc"
             android:textColor="#000"
            android:layout_marginLeft="10dp"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/hello_world" />
    
    </LinearLayout>
    


  • 相关阅读:
    洛谷[P1002]过河卒
    ACM-Teleportation
    ACM-Team Tic Tac Toe
    Data_Structure04-树
    Data_Structure03-栈和队列
    Data_Structure02-线性表
    Data_Structure01-绪论
    C语言第二次实验报告
    C语言第一次实验报告
    mysql
  • 原文地址:https://www.cnblogs.com/yangykaifa/p/7291239.html
Copyright © 2020-2023  润新知