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

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-05 13:48:33

ListView局部更新(非notifyDataSetChanged)的相关文章

listView局部刷新

listview局部刷新关键逻辑代码:在activity中 /** * listview局部更新 */ private void updateListView(int position){ int firstVisiblePosition = mListView.getFirstVisiblePosition(); int lastVisiblePosition = mListView.getLastVisiblePosition(); if(position>=firstVisiblePosi

Android listview局部刷新和模拟应用下载(zhu)

在android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用notifyDataSetChanged()这个函数,但是它会更新listview中所有可视范围内的item,这样对性能肯定会有影响.比较常见的情景是android应用商店中的下载列表,当我们下载一款游戏的时候,只需要更新这款游戏对应的进度就可以了.本文就来模拟android应用商店的游戏下载,实现对listview的局部刷新,只实现一个简单的demo,不去真的下载文件.1. 首先来创建

Android 如何在 ListView 中更新 ProgressBar 进度

=======================ListView原理============================== Android 的 ListView 的原理打个简单的比喻就是: 演员演小品(假设演员都长一样,每个角色任何演员都可以演) 小品剧也不会为每个角色都招募一个演员.ListView 不会为每一个 Item 创建 View 对象. 小品剧的演员在一个角色表演完成后,会在后台换下一个角色的服装,等待需要表演的时候再出场. ListView 会让未显示的 View 填充数据后缓

Jquery Ajax 页面局部更新

Ajax 是一个异步过程,页面中的一部分代码发出一个网络请求,在里面设置一个回调函数,如果网络请求得到返回,那么执行回调函数.在回调函数中的上下文和时间触发的上下文会有所不同,在jquery中的应用需求大多数时候是这样的.比如我点击了一个按钮,这个按钮是一个 tr 中的 td 中的内容,我可以通过这个点击时间确定当前行的任何属性,或者确定当前也面中的任何一个属性,只要这个页面的布局是又某种可控的过滤的话.需要实用到this关键字.还有就是 find() 方法.比如this.parent().fi

经验总结16--EF局部更新数据

EF提供更新某个实体的某些数据,而其他数据保留的功能. 1.重新实例一个实体,并给予ID和需要修改的字段.注:此处的实体不能从数据库中查询,只能新的实例. Order order = new Order() { ID = id }; order.Name= "test222"; 2.包含该实体,更新策略为Unchanged,设置需要更新的字段. DbEntityEntry<Order> entry = db.Entry<Order>(order); entry.

局部更新document(partial updates to documents)

就像在Updating a whole document中所说的一样,更新document的步骤就是检索--修改--插入整个document.然而使用update,就能局部更新,就像在一个请求里增加一个计数器一样. 以前说过document是不可变的--不能被修改,只能被替换,update也必须遵守这个规则.在外部看来是局部更新的,在内部,update依然执行了检索--修改--重新插入这个流程.不同的是这个过程是发生在shard中,因此避免了多次网络请求的开销,通过减少检索和重新插入的时间,降低

ES使用脚本进行局部更新的排错记录

初学Elasticsearch,在按照<Elasticsearch服务器开发(第2版)>进行学习的过程中,在P17页中1.4.5 更新文档小节,使用脚本对文档进行局部更新的时候遇到了如下报错:   ~ curl -XPOST http://127.0.0.1:9200/blog/article/1/_update -d '{"script": "ctx._source.content=\"new content\""}' { &quo

【SSH网上商城项目实战28】使用Ajax技术局部更新商品数量和总价

昨天把项目部署了一下,玩了玩,今天完善了一下购物车中修改商品数量就能局部更新相应的总价的功能,大家都知道这得用Ajax实现,我之前也没学Ajax,刚好借助这个小功能,去简单学习一下Ajax的知识. 1.问题的分析 先看一下页面中的情况:  功能如上,在没有Ajax之前,一般都是根据用户修改的值去找Action,然后返回新的jsp页面重新加载整个页面,完成数字的更新.但是有了Ajax技术后,我们可以利用Ajax技术局部刷新要改变的地方,而不是重新加载整个页面.首先看一下上图对应的jsp部分的代码:

dojo组件使用---tree组件的局部更新

dojo的tree组件使用主要涉及的几个api: 1.dijit/Tree:树组件 2.dijit/Tree._TreeNode:树节点组件,局部更新时就是修改这个组件.(由于数组件的store使用Memory,所以需要手动同步更新store): 3.dijit/tree/ObjectStoreModel:数组件数据操作模块. 具体操作: 1.拿到dijit/Tree._TreeNode实例对象.树组件的每个节点都是一个dijit/Tree._TreeNode对象,通过selectedNode/