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>

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

  1. Solr 18 - 通过SolrJ局部更新Solr中的文档 (原子操作、非覆盖操作)

    目录 1 需求分析 2 需求实现 2.1 pom.xml依赖 2.2 Java代码示例 3 补充说明 3.1 关于文档中_version_的取值说明 3.2 store=true/false的区别 1 ...

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

    在android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用notifyDataSetChanged()这个函数,但是它会更新listview中所有可 ...

  3. 转:android listview局部刷新和模拟应用下载

    在 android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用 notifyDataSetChanged()这个函数,但是它会更新listview中所 ...

  4. Android 结合实际项目学会ListView局部刷新和相关知识《一》

    转载本专栏博客,请注明出处:道龙的博客 最近在公司参与的项目中有一个界面需要做局部UI更新处理,把其化烦为简为Demoi形式写在这里.我们还是运行该Demo,知道ListView局部刷新的使用场景:( ...

  5. Android 如何在 ListView 中更新 ProgressBar 进度

    =======================ListView原理============================== Android 的 ListView 的原理打个简单的比喻就是: 演员演 ...

  6. ListView只更新某个item

    方案1:针对需要更新的item调用public View getView(int position, View convertView, ViewGroup parent)即可.如: public c ...

  7. WebApiClient的JsonPatch局部更新

    1. 文章目的 随着WebApiClient的不断完善,越来越多开发者选择WebApiClient替换原生的HttpClient,本文将介绍使用WebApiClient来完成JsonPatch提交的新 ...

  8. 通过Solrj实现对索引库中数据的局部更新操作

    for (UpdateIndexDTO updateIndexDTO : data) { // 局部更新 SolrInputDocument doc = new SolrInputDocument() ...

  9. .Net页面局部更新的思考

    最近在修改以前做的模块,添加一个新功能.整理了下才发现重用率很低,大部分的东西还是需要重新写.功能里用到了局部更新,所有整理一下一路来实现局部更新的解决方案及改进. 我接触的项目开发大多是以Asp.n ...

随机推荐

  1. [转]解决Eclipse更新ADT插件时遇到的Eclipse reports rendering library more recent than ADT plug-in问题

    使用 SDK Manager 工具更新下载新版本后,无法显示可视化布局,同时提示 This version of the rendering library is more recent than y ...

  2. 支持解析GitHub Flavored Markdown(GFM)的PHP库-Parsedown

    网上搜索PHP的markdown解析库,只能找得到Michel的PHP Markdown,这个库很不错,但是他只能支持标准markdown和他自己定义的一套扩展php Markdown Extra.这 ...

  3. mysql安装以后无法登陆的的解决方法((ERROR 1698 (28000): Access denied for user 'root'@'localhost'))

    mysql安装以后无法登陆的的解决方法((ERROR 1698 (28000): Access denied for user 'root'@'localhost')) 解决步骤: [====> ...

  4. 基于QTP的自己主动化測试框架介绍

    继前面用了七章介绍了基于QTP的自己主动化測试框架,以下再用几个视频再补充一下.        视频一:基本框架特点介绍说明 .框架的特点从正反两面进行了分析以及主要思想      http://v. ...

  5. curl错误码大全

    CURL状态码列表 状态码 状态原因 解释 0 正常访问 访问地址未返回结果 1 错误的协议 未支持的协议.此版cURL 不支持这一协议. 2 初始化代码失败 初始化失败. 3 URL格式不正确 UR ...

  6. Spring 自定义配置类bean

    <!-- 引入配置文件 --> <bean id="propertyConfigurer" class="org.springframework.bea ...

  7. 将思维转向rss

    本屌丝因为穷住在了离市区比较远的农民房,平时上下班单程地铁时间接近一小时.在这漫长的一小时里,总得干点什么来蹉跎这段时光,看手机是最容易实现的事情.最地铁信号不好,手机也没什么好看的. 经过高人指点说 ...

  8. 通过web php 执行shell脚本,获取的结果与直接在命令行下获取的结果不同。

    公司项目中的一项小功能,统计设备的连接数.其中用到shell脚本来获取已连接设备的统计.使用命令 /bin/netstat -an| grep ESTABLISHED | awk '{print $4 ...

  9. Java死锁举例

    死锁: 在多线程竞争使用共享资源的情况下.就有可能出现死锁的情况.比方,当一个线程等待还有一个线程所持有的锁时.那个线程又可能在等待第一个线程所持有的锁.此时.这两个线程会陷入无休止的相互等待状态.这 ...

  10. Guava缓存使用

    public class GuavaCache { /** * LoadingCache当缓冲中不存在时,可自动加载 * */ private static LoadingCache<Integ ...