android-BaseAdapter自定义控件深刻理解
一、自定义控件的实现
自定义控件需要继承BaseAdapter抽象类,该类实现了ListAdapter, SpinnerAdapter两个接口,这两个接口继承了Adapter接口类,没错、是继承Adapter接口;而自定义控件实现的四个方法也来自Adapter接口;
1、下面对这四个方法进行详细说明:
<1>public int getCount()
返回记录的数量,如果数据源是数组,返回的就是数组长度;如果数据原是集合,返回的就是集合的大小,即决定了显示了多少行;
<2>Object getItem(int position)
返回指定下标对应的对象;
<3>long getItemId(int position)
返回指定下标对应的id;
<4>View getView(int position, View convertView, ViewGroup parent);
返回下标对应的View视图对象
position:下标
convertView:可复用的缓存视图对象
为null:没有可复用的,必须加载一个item的布局文件,并赋值给convertView;
不为null:直接使用此视图对象。
parent:父容器(常用ListView)
2、给ListView的item设置监听器:
<1>item的点击监听:listView.setOnItemClickListener(listener)
<2>item的长按监听:listView.setOnItemLongClickListener(listener)
3、初步优化---防止内存溢出
a.内存中最多存在n+1个convertView对象
b.只有当convertView为null时才去加载item的布局文件。
练习:
从以上运行截图可知分别采用了ListView、GridView布局,界面布局过于简单不允给出;
1、ListView+BaseAdapter
该程序中用到一个AppInfo类,对查询出来的信息进行封装;
自定义适配器类继承BaseAdapter类,实现上述说明的四种方法;
为每一个item添加单击和长按事件监听;
getAllAppInfos()方法用于获取手机应用程序信息,packagename、label、icon、占用大小等,获取应用程序信息的详细介绍待续...
界面布局过程中使用到的重要属性:
scaleType="fitXY" //图片填充满View的宽度
minHeight="70dp" //最小高度
layout_gravity //控件本身相对于父视图的位置
gravity //指定控件中的内容的基本位置
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.AdapterView.OnItemLongClickListener;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
public class MainActivity extends Activity implements OnItemLongClickListener{
private ListView lv_main;
private List<AppInfo> data;
private AppAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//初始化成员变量
lv_main=(ListView) findViewById(R.id.lv_main);
data=getAllAppInfos();
adapter=new AppAdapter();
//显示列表
lv_main.setAdapter(adapter);
//设置监听器
lv_main.setOnItemClickListener(new OnItemClickListener() {
/*
* parent:为当前的ListView
* view:当前Item视图对象
* position:当前行的下标
*
* */
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
//当前行应用程序名称
String appName=data.get(position).getAppName();
//提示
Toast.makeText(MainActivity.this, appName, 0).show();
}
});
lv_main.setOnItemLongClickListener(this);
}
class AppAdapter extends BaseAdapter{
@Override
public int getCount() {
return data.size();
}
@Override
public Object getItem(int position) {
return data.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
//返回带数据当前行的iten视图对象
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//1、判断如果convertView空,加载item布局文件
if (convertView==null) {
Log.e("TAG","谢荣康");
convertView=View.inflate(MainActivity.this, R.layout.item_main, null);
}
//2、得到当前行数据对象
AppInfo appInfo=data.get(position);
//3、得到当前行需要更新的子View对象
ImageView imageView=(ImageView)convertView.findViewById(R.id.iv_item_icon);
TextView textView=(TextView)convertView.findViewById(R.id.tv_item_name);
//4、给视图设置数据
imageView.setImageDrawable(appInfo.getIcon());
textView.setText(appInfo.getAppName());
//返回convertView
return convertView;
}
}
/*
* 得到手机所有应用信息
* Drawable:可绘制的图片
*
* */
protected List<AppInfo> getAllAppInfos(){
List<AppInfo> list=new ArrayList<>();
//得到应用的PackageManager---获得已经安装应用程序的信息
PackageManager packageManager=this.getPackageManager();
//创建主题界面的intent
Intent intent=new Intent();
intent.setAction(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
//得到包含应用信息的列表
List<ResolveInfo> ResolveInfos=packageManager.queryIntentActivities(intent, 0);
//List<ResolveInfo> ResolveInfos=packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
//Collections.sort(ResolveInfos,new ResolveInfo.DisplayNameComparator(packageManager));
//遍历
for (ResolveInfo ri:ResolveInfos) {
//得到包名
String packageName=ri.activityInfo.packageName;
//得到图标
Drawable icon=ri.loadIcon(packageManager);
//得到应用的名称
String appName=ri.loadLabel(packageManager).toString();
//封装应用信息对象
AppInfo appInfo=new AppInfo(icon,appName,packageName);
//添加到list
list.add(appInfo);
}
return list;
}
@Override
public boolean onItemLongClick(AdapterView<?> parent, View view,
int position, long id) {
/*
* 返回值false,长按和单击都会执行同样的功能
* */
//删除当前行并更新列表
data.remove(position);
/*
* 通知更新列表使用所有缓存的item视图对象,设置更新列表不使用
*
* */
adapter.notifyDataSetChanged();
//lv_main.setAdapter(adapter);
return true;
}
}
2、GridView+BaseAdapter
除item使用的父容器不同外基本和以上一致;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.GridView;
import android.widget.Toast;
public class MainActivity extends Activity {
private GridView gridView;
private MainAdapter adapter;
List<AppInfoBean> list;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
gridView=(GridView) findViewById(R.id.gv_main);
adapter=new MainAdapter(this.getAllAppInfos(),this);
gridView.setAdapter(adapter);
gridView.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
// TODO Auto-generated method stub
Toast.makeText(MainActivity.this, list.get(position).appName, 0).show();
}
});
}
protected List<AppInfoBean> getAllAppInfos(){
list=new ArrayList<>();
//得到应用的PackageManager---获得已经安装应用程序的信息
PackageManager packageManager=this.getPackageManager();
//创建主题界面的intent
Intent intent=new Intent();
intent.setAction(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
//得到包含应用信息的列表
List<ResolveInfo> ResolveInfos=packageManager.queryIntentActivities(intent, 0);
//List<ResolveInfo> ResolveInfos=packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
//Collections.sort(ResolveInfos,new ResolveInfo.DisplayNameComparator(packageManager));
//遍历
for (ResolveInfo ri:ResolveInfos) {
//得到图标
Drawable icon=ri.loadIcon(packageManager);
//得到应用的名称
String appName=ri.loadLabel(packageManager).toString();
//封装应用信息对象
AppInfoBean appInfo=new AppInfoBean(icon,appName);
//添加到list
list.add(appInfo);
}
return list;
}
}
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;
public class MainAdapter extends BaseAdapter {
List<AppInfoBean> data;
Context context;
public MainAdapter(List<AppInfoBean> list,Context context) {
this.context=context;
data=list;
}
@Override
public int getCount() {
// TODO Auto-generated method stub
return data.size();
}
@Override
public Object getItem(int position) {
// TODO Auto-generated method stub
return data.get(position);
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//1、判断如果convertView空,加载item布局文件
if (convertView==null) {
Log.e("TAG","谢荣康");
convertView=View.inflate(context, R.layout.item_main, null);
}
//2、得到当前行数据对象
AppInfoBean appInfo=data.get(position);
//3、得到当前行需要更新的子View对象
ImageView imageView=(ImageView)convertView.findViewById(R.id.iv_item_icon);
TextView textView=(TextView)convertView.findViewById(R.id.tv_item_name);
//4、给视图设置数据
imageView.setImageDrawable(appInfo.getIcon());
textView.setText(appInfo.getAppName());
return convertView;
}
}
二、问题
1、空指针异常
ImageView imageView=(ImageView)convertView.findViewById(R.id.iv_item_icon);
ImageView imageView=(ImageView)findViewById(R.id.iv_item_icon);---出现控指针异常
android-BaseAdapter自定义控件深刻理解的更多相关文章
- android:Layout_weight的深刻理解
最近写Demo,突然发现了Layout_weight这个属性,发现网上有很多关于这个属性的有意思的讨论,可是找了好多资料都没有找到一个能够说的清楚的,于是自己结合网上资料研究了一下,终于迎刃而解,写出 ...
- Android中自定义控件TextSize属性问题
本文主要说明一个自定义控件添加TextSize属性的坑,刚刚从坑里面爬出来,写个随笔,记录一下: *************************************************** ...
- Atitit 图像处理 深刻理解梯度原理计算.v1 qc8
Atitit 图像处理 深刻理解梯度原理计算.v1 qc8 1.1. 图像处理 梯度计算 基本梯度 内部梯度 外部梯度 方向梯度1 2. 图像梯度就是图像边缘吗?2 1.1. 图像处理 梯度计算 ...
- 深刻理解Java中final的作用(一):从final的作用剖析String被设计成不可变类的深层原因
声明:本博客为原创博客,未经同意,不得转载!小伙伴们假设是在别的地方看到的话,建议还是来csdn上看吧(原文链接为http://blog.csdn.net/bettarwang/article/det ...
- for循环,你深刻理解了吗?
前几天,有一个面试机会,去看了看,遇到一个认为不错的面试题! 过了几天看到csdn上说华为的一道面试题,看了下和我遇到的很相似! 我分享出来希望大家有帮助! 你真的深刻理解for循环了吗? ...
- 【转】Android菜单详解——理解android中的Menu--不错
原文网址:http://www.cnblogs.com/qingblog/archive/2012/06/08/2541709.html 前言 今天看了pro android 3中menu这一章,对A ...
- 深刻理解iosBlock
深刻理解iosBlock ///一个控制器里的方法 - (void)setRefreshHeader { ACWeakSelf(self); self.tableView.mj_header = [M ...
- javascript 面向对象程序设计--深刻理解对象
javascript中,每个对象都是基于一个引用类型创建的,我们可以把ECMAScript 的对象想象成散列表:无非就是一组名值对,其中值可以是数据或函数. 深刻理解对象 创建自定义对象的最简单方式就 ...
- [转]深刻理解Python中的元类(metaclass)以及元类实现单例模式
使用元类 深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例 ...
随机推荐
- android 消息推送
android 消息推送 极光推送百度云推送(语音)友盟消息推送
- 【转】最近搞Hadoop集群迁移踩的坑杂记
http://ju.outofmemory.cn/entry/237491 Overview 最近一段时间都在搞集群迁移.最早公司的hadoop数据集群实在阿里云上的,机器不多,大概4台的样子,据说每 ...
- 从零开始---控制台用c写俄罗斯方块游戏(1)
从零开始---控制台用c写俄罗斯方块游戏(1) 很少写博文,一来自身知识有限,二来自己知道,已经有很多这样的博文了,三就是因为懒,文笔也一般,四来刚出来工作,时间也不多 之所以写这篇博文,是因为应群里 ...
- Python与Hack之Unix口令
1.在实验时候,先导入crypt库:必须在Unix环境下才能实现这个模块 2.代码贴一下,以后有了Unix环境试试吧: import cryptimport syssys.modules['Crypt ...
- Angular JS学习之表达式
1.Angular JS使用表达式把数据绑定到HTML: 2.Angular JS表达式写在双大括号中:{{expression}} **Angular JS表达式把数据绑定到HTML,这与ng-bi ...
- Windows下安装 使用coreseek
1.安装 1.01:到官网下载 coreseek-3.2.14 1.01_1 原理 缓存服务器: 准备数据 来自数据库 配置连接 生成索引 开启服务 流程:用户-> web->sphin ...
- 【noip2014T3】
上文有提到noip2014还有没A的嘛..就先把这个坑给填了 flappy bird好sad啊 还是先做解方程 八中的数据好强了,然而我最后凑了四个质数就A了,感谢shy! 作为联赛最后一题,学习它的 ...
- CentOS7安装docker出错(Transaction check error)
1. 出错内容: Transaction check error: :-.el7_2..x86_64 conflicts with :-.el7.x86_64 :-.el7_2..x86_64 con ...
- HIVE 创建外部分区表--利用HUE不能创建外部表
Create EXTERNAL table obd_data_2( imei string, ts timestamp, fuel_instant float, gps_speed float, gp ...
- dplyr包--数据操作与清洗
1.简介 在我们数据分析的实际应用中,我们可能会花费大量的时间在数据清洗上,而如果使用 R 里面自带的一些函数(base 包的 transform 等),可能会觉得力不从心,或者不是很人性化.好在我们 ...