/*《获取所有应用信息的列表,并实现点击事件打开应用(有顶部图片,优化)》 ListView展示列表数据: 在layout中创建ListView,然后是创建适配器(前提条件:创建每一行的layout即Image+TextView,创建每一行的数据 (数据可以为集合,需建一个自定义数据类),在getView里把布局资源拿出来放入一个convertView即转换型视图中 ,再通过这个view和position去set设置数据),最后把适配器与ListView相匹配。 其中适配器获取数据:获取所有应用的信息List<ResolveInfo>,通过构造方法把数据传进来,可以通过几个方法获取传入数据的信息, getView()来处理view和data适配,数据和视图的绑定,填充数据,设置点击事件,点击视图(应用的图标) ,用intent开启activity(getView方法实现详情见代码注释),创建一个ViewHolder类存放控件,作为ListView的缓存类 便于控件取用,提高性能。 viewHolder 的作用是因为Android有个recycler的反复循环器,viewHolder就是借助他来做到循环利用itemView。 convertView 把从XML文件读出来的信息装到一个view里,当这个convertView不存在时,即第一次使用它 ,我们就创建一个item布局的View对象并赋给convertView,以后使用convertView时 ,只需从convertView中getTag取出来就可以,不需要再次创建item的布局对象了,这样便提高了性能。 convertView和viewHolder打配合通过实现循环和控件便捷复用来实现性能优化。 每次显示一个item都调用一次getview方法但是每次调用的时候covertview为空(因为还没有旧的view) ,当显示完了之后。如果屏幕移动了之后,并且导致有些Item(也可以说是view)跑到屏幕外面 ,此时如果还有新的item需要产生,则这些item显示时调用的getview方法中的convertview参数就不是null ,而是那些移出屏幕的view(旧view),我们所要做的就是将需要显示的item填充到这些回收的view(旧view)中去 ,最后注意convertview为null的不仅仅是初始显示的那些item,还有一些是已经开始移入屏幕但是还没有view被回收的那些item。 LayoutInflater 加载布局文件 getView ListView 针对List中每个item,要求 adapter “给我一个视图” (getView),一个新的视图被返回并显示 */
public class AppListActivity extends AppCompatActivity {
private ListView mListView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//
mListView = (ListView) findViewById(R.id.list_view_demo);
//获取布局填充服务
LayoutInflater layoutInflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
//将标题布局文件添加到View的对象
View headerView = layoutInflater.inflate(R.layout.header_list_demo, null);
//和ListView连接,添加标题控件
mListView.addHeaderView(headerView);
//实例化List数据对象,用于下面适配器传送数据
List<ResolveInfo> infos = getAppInfos();
//给ListView绑定适配器
mListView.setAdapter(new AppListAdapter(this, infos));
}
//获取所有应用信息,返回值为List,ResolveInfo 类 所有应用的信息List<ResolveInfo>
private List<ResolveInfo> getAppInfos() {
//系统方法:就是为了实现获取所有应用信息
Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);
mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
return getPackageManager().queryIntentActivities(mainIntent, 0);//查询
}
//自定义适配器,内部类,包含数据(通过构造方法传进来)和视图
public class AppListAdapter extends BaseAdapter {
private Context mContext;
private List<ResolveInfo> mInfos;//包含所有应用信息的List集合
//通过构造方法把数据传进来
public AppListAdapter(Context context, List<ResolveInfo> infos) {
mContext = context;
mInfos = infos;
}//下面可以通过几个方法获取传入数据的信息
//多少条数据
@Override
public int getCount() {
return mInfos.size();
}
//
@Override
public Object getItem(int position) {
return mInfos.get(position);
}
//
@Override
public long getItemId(int position) {
return position;
}
//处理view和data适配,数据和视图的绑定,填充数据
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
//viewholder里有image和textview
ViewHolder viewHolder;
//把xml文件读到java文件里成一个view对象来操作
LayoutInflater layoutInflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
//如果为空就创建
if (convertView == null) {
//实例化控件包
viewHolder = new ViewHolder();
//把xml读出来,显示成一个view
convertView = layoutInflater.inflate(R.layout.item_demo_list, null);
//用convertView.findViewById把两个控件读出来存到Holder中,方便以后调用
viewHolder.nameTextView = (TextView) convertView.findViewById(R.id.title_text_view);
viewHolder.avatarImageView = (ImageView) convertView.findViewById(R.id.icon_image_view);
//为空就先创建再设置给缓存器viewholder,建立对应联系
convertView.setTag(viewHolder);
} else {
//不为空直接取出来给viewholder做缓存
viewHolder = (ViewHolder) convertView.getTag();
}
// 和数据之间进行绑定,把数据装到工具包的控件上,从集合取出来数据设置到控件上
viewHolder.nameTextView.setText(mInfos.get(position).activityInfo.loadLabel(mContext.getPackageManager()));
viewHolder.avatarImageView.setImageDrawable(mInfos.get(position).activityInfo.loadIcon(mContext.getPackageManager()));
//设置点击事件,点击视图(应用的图标),用intent开启activity
convertView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//定义出所有应用信息的位置
ResolveInfo info = mInfos.get(position);
//拿到该应用的包名
String pkg = info.activityInfo.packageName;
//拿到应用的主activity类
String cls = info.activityInfo.name;
//创建一个主键(包名,类名)
ComponentName componet = new ComponentName(pkg, cls);
//实例化Intent对象来开启应用
Intent intent = new Intent();
//设置主键
intent.setComponent(componet);
//通过设置了主键的Intent对象开始运行这个activity
startActivity(intent);
}
});
//返回这个视图
return convertView;
}
//存放控件,作为ListView的缓存类,便于控件取用,提高性能
class ViewHolder {
ImageView avatarImageView;
TextView nameTextView;
}
}
}
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingBottom="@dimen/activity_vertical_margin"
tools:context=".AppListActivity">
<ListView
android:id="@+id/list_view_demo"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/white"></ListView>
</RelativeLayout>
item_demo_list.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="60dp">
<ImageView
android:id="@+id/icon_image_view"
android:layout_width="60dp"
android:layout_height="60dp"
android:src="@mipmap/ic_launcher"/>
<TextView
android:id="@+id/title_text_view"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="@string/app_name"
android:minHeight="30dp"
android:layout_centerVertical="true"
android:gravity="left|center"
android:textColor="#494949"
android:singleLine="true"
android:layout_marginLeft="16dp"
android:layout_toRightOf="@+id/breakfast_imageview"
android:textSize="18sp"/>
</LinearLayout>
header_list_demo.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="60dp">
<ImageView
android:id="@+id/icon_image_view"
android:layout_width="match_parent"
android:layout_height="80dp"
android:background="@drawable/banner"/>
</LinearLayout>