Xamarin.Android开发实践(十四)

简介: 原文:Xamarin.Android开发实践(十四) Xamarin.Android之ListView和Adapter 一、前言 如今不管任何应用都能够看到列表的存在,而本章我们将学习如何使用Xamarin去实现它,以及如何使用适配器和自定义适配器(本文中的适配器的主要内容就是将原始的数据转换成了能够供列表控件显示的项)。

原文:Xamarin.Android开发实践(十四)

Xamarin.Android之ListView和Adapter

一、前言

如今不管任何应用都能够看到列表的存在,而本章我们将学习如何使用Xamarin去实现它,以及如何使用适配器和自定义适配器(本文中的适配器的主要内容就是将原始的数据转换成了能够供列表控件显示的项)。

 

二、简介适配器

在开始之前我们需要先了解下适配器,首先是提供给我们使用的适配器之间的关系:

下面我们将上面的适配器进行简单的介绍:

BaseAdapter:顾名思义,就是所以适配器的基类,但是我们不能将其实例化,因为它是一个虚类,一般我们都是继承该类并实现其中的方法,形成形成自定义的列表(大多数情况下我们都会使用到它)。

ArrayAdapterArrayAdapter<T>:就是专门用于将列表数据的适配器,该适配器内部已经实现了BaseAdapter的方法,所以我们只需要指定对应的数据项以及列表项资源即可。

CursorAdapter:上面的适配器只是用于列表数据,而该适配器则可以用在将数据库返回的结果集显示到列表中去,当然在难度上也要比上面的适配器高。

 

三、正文

 

1.简单列表

下面我们将利用ListActivityArrayAdapater<T>去实现一个简单的列表界面,下面为该示例的代码(MainActivity.cs):

 1     public class MainActivity : ListActivity
 2  {  3 protected string[] items;  4  5 protected override void OnCreate(Bundle bundle)  6  {  7 base.OnCreate(bundle);  8 items = new string[]{  9 "First","Second","Third" 10  }; 11 ListAdapter = new ArrayAdapter<string>(this, Android.Resource.Layout.SimpleListItem1, items); 12  } 13 14 protected override void OnListItemClick(ListView l, View v, int position, long id) 15  { 16 var t = items[position]; 17 Toast.MakeText(this, t, ToastLength.Short).Show(); 18  } 19 }

这里有一个重要的关键词就是ListActivity,如果存在一个界面整个界面都是列表,那么我们就可以继承这个特殊的活动,并且可以通过ListView属性和ListAdapter去控制,同时还可以响应关于列表的事件。其中我们利用了ArrayAdapter给列表指定了一个适配器,而这个适配器的第一个参数是当前的上下文,第二个是列表中的项的界面,最后一个就是对应的数据了。

 

最后将显示如下的界面:

 

当我们点击不同的项后,还能看到底部显示了当前我们选择的项。这个功能就是在我们重写了OnListItemClick实现了,正如代码中所示,我们根据position获取指定的数据,然后通过Toast将其显示出来。

 

 

2.自定义一个适配器

简单介绍过如何使用适配器后,我们将开始学习如何利用BaseAdapter自定义一个适配器,能够促使我们理解适配器内部的工作原理,首先我们来看下笔者的写的代码:

 1         public class MyCustomeAdapter : BaseAdapter<string>
 2  {  3 string[] items;  4  Activity activity;  5  6 public MyCustomeAdapter(Activity context, string[] values)  7 : base()  8  {  9 activity = context; 10 items = values; 11  } 12 13 public override string this[int position] 14  { 15 get { return items[position]; } 16  } 17 18 public override int Count 19  { 20 get { return items.Length; } 21  } 22 23 public override long GetItemId(int position) 24  { 25 return position; 26  } 27 28 public override View GetView(int position, View convertView, ViewGroup parent) 29  { 30 View v = convertView; 31 if (v == null) 32 v = activity.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, null); 33 v.FindViewById<TextView>(Android.Resource.Id.Text1).Text = items[position]; 34 return v; 35  } 36 }

其中最主要的是GetView方法,ListView和ListActivity主要是通过调用该方法获取对应项的视图,然后将其添加为子控件,从而显示,如果我们需要显示例如淘宝等app那种复杂的列表就需要通过重写该方法达到,细心的读者可以发现笔者是先判断convertView是否为null,如果为null才通过Inflate方法重新实例化一个视图,关于这部分我们可以参考该文章

 

关于其他的方法相信大家靠名称就能够明白了,这里就不多做解释,具体的使用方法如下所示:

1         protected override void OnCreate(Bundle bundle)
2  { 3 base.OnCreate(bundle); 4 string[] items = new string[]{ 5 "First","Second","Third" 6  }; 7 ListAdapter = new MyCustomeAdapter(this, items); 8 }

 

 

3.关键字索引

在手机的使用中,大家一定经常使用着电话簿,当我们点击右边的滚动条进行滑动的时候还会显示a-z的字母并且列表的内容也会根据这些字母的变化发生变化,将快速定位到联系人姓名为指定字母开头的位置,本节我们将会学习如何实现该功能。

 

提供这个功能我们需要实现一个接口,这个接口就是ISectionIndexer,下面是关于该接口的代码:

1     public interface ISectionIndexer : IJavaObject, IDisposable
2  { 3 int GetPositionForSection(int section); 4 int GetSectionForPosition(int position); 5  Object[] GetSections(); 6 }

关于这些接口简单的介绍下:

GetPositionForSection:根据关键字的索引获取该关键字的起始数据索引。

GetSectionForPosition:根据数据索引获取关键字索引。

GetSections:返回关键字数组。

 

笔者为了能够节约时间,所以利用上节的示例代码,在MyCustomeAdapter中实现了ISectionIndexer接口,下面是笔者的代码:


 1 public class MyCustomeAdapter : BaseAdapter<string> , ISectionIndexer
 2  {  3 string[] items;  4  Activity activity;  5  6 Dictionary<string, int> alphaindex;  7  Java.Lang.Object[] sectionsObjects;  8 string[] sections;  9 10 public MyCustomeAdapter(Activity context, string[] values) 11 : base() 12  { 13 activity = context; 14 items = values; 15 16 alphaindex = new Dictionary<string, int>(); 17 //获取每种关键字的起始数据索引 18 for (int i = 0; i < items.Length; i++) 19  { 20 string key = items[i][0].ToString(); 21 if (!alphaindex.ContainsKey(key)) 22  alphaindex.Add(key, i); 23  } 24 25 //将关键字转换成数据 26 sections = new string[alphaindex.Keys.Count]; 27 alphaindex.Keys.CopyTo(sections, 0); 28 29 //将关键字转换成Java.Lang.String类型 30 sectionsObjects = new Java.Lang.Object[alphaindex.Keys.Count]; 31 for (int i = 0; i < sections.Length; i++) 32  { 33 sectionsObjects[i] = new Java.Lang.String(sections[i]); 34  } 35  } 36 37 public override string this[int position] 38  { 39 get { return items[position]; } 40  } 41 42 public override int Count 43  { 44 get { return items.Length; } 45  } 46 47 public override long GetItemId(int position) 48  { 49 return position; 50  } 51 52 public override View GetView(int position, View convertView, ViewGroup parent) 53  { 54 View v = convertView; 55 if (v == null) 56 v = activity.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, null); 57 v.FindViewById<TextView>(Android.Resource.Id.Text1).Text = items[position]; 58 return v; 59  } 60 61 public int GetPositionForSection(int section) 62  { 63 //根据关键字索引获取关键字,然后在根据关键字从alphaindex获取对应的value,即该关键字的起始数据索引 64 return alphaindex[sections[section]]; 65  } 66 67 public int GetSectionForPosition(int position) 68  { 69 int preposition = 0; 70 //循环关键字 71 for (int i = 0; i < sections.Length; i++) 72 { 73 //判断当前的索引是否在i所在关键字的范围内 74 if (GetPositionForSection(i) > position) 75 break; 76 preposition = i; 77 } 78 return preposition; 79 } 80 81 public Java.Lang.Object[] GetSections() 82 { 83 return sectionsObjects; 84 } 85 }

 

关于该接口方法中的实现可以看代码中的注释,在使用的时候也要写如下代码:

 1         protected override void OnCreate(Bundle bundle)
 2  {  3 base.OnCreate(bundle);  4 string[] items = new string[]{  5 "Apple","As","Banner",  6 "BMW","Beatf","Bad",  7 "Day","Dazy","Eat","Else",  8 "Feel","Fly","Go","Hourse"  9  }; 10 ListAdapter = new MyCustomeAdapter(this, items); 11 ListView.FastScrollEnabled = true; 12 ListView.FastScrollAlwaysVisible = true; 13 }

 

按照官方的说法其实只要将FastScrollEnabled设置为true即可,但是笔者在虚拟机中测试的时候并不会显示,所以笔者还要再将FastScrollAlwaysVisible一起设置为true才可以看到效果,下面为笔者的虚拟机上的截图:

 

 

4.Cursor列表

前面我们都是通过数组来填充列表的,但是实际情况下数据大多数都是来自于数据库,而数据库通过查询后将会得到ICursor类型的返回值,当然我们也可以通过将ICursor转换成数组或者键值对,但这些都是没有必要的,因为有自带的适配器(CursorAdapter)。

 

首先我们需要创建一个数据库,下面是笔者的代码,想学习数据库的可以参考本人的以下随笔:

《Xamarin.Android之SQLiteOpenHelper》

《Xamarin.Android之ContentProvider》

《Xamarin.Android之SQLite.NET ORM》

 

代码如下所示:

 1     public class TestDatabase : SQLiteOpenHelper
 2  {  3 public TestDatabase(Context context):base(context,"testdatabase.db",null,1)  4  {}  5  6 public override void OnCreate(SQLiteDatabase db)  7  {  8 db.ExecSQL("CREATE TABLE TestTable ( _id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,name TEXT NOT NULL)");  9 db.ExecSQL("INSERT INTO TestTable (name) values('Vegetables')"); 10 db.ExecSQL("INSERT INTO TestTable (name) values('Fruits')"); 11 db.ExecSQL("INSERT INTO TestTable (name) values('Flower Buds')"); 12 db.ExecSQL("INSERT INTO TestTable (name) values('Legumes')"); 13 db.ExecSQL("INSERT INTO TestTable (name) values('Bulbs')"); 14 db.ExecSQL("INSERT INTO TestTable (name) values('Tubers')"); 15  } 16 17 public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) 18  { 19 20  } 21 }

 

数据库创建部分比较简单,主要的重点是下面的代码。我们将实例化SimpleCursorAdapter并赋值给ListAdapter中去,代码如下所示:

 1     [Activity(Label = "CursorAdapter", MainLauncher = true, Icon = "@drawable/icon")]  2 public class MainActivity : ListActivity  3  {  4  TestDatabase vdb;  5  ICursor cursor;  6  7 protected override void OnCreate(Bundle bundle)  8  {  9 base.OnCreate(bundle); 10 vdb = new TestDatabase(this); 11 //通过SQL查询数据库 12 cursor = vdb.ReadableDatabase.RawQuery("SELECT * FROM TestTable", null); 13  StartManagingCursor(cursor); 14 //创建适配器 15 ListAdapter = new SimpleCursorAdapter(this, Android.Resource.Layout.SimpleListItem1, cursor, 16 new string[] { "name" }, //指定使用结果集中的哪个字段数据 17 new int[] { Android.Resource.Id.Text1 }); //填充到哪个控件中 18  } 19 20 protected override void OnDestroy() 21  { 22 base.OnDestroy(); 23  StopManagingCursor(cursor); 24  cursor.Close(); 25  } 26 }

通过上面的代码我们可以看到SimpleCursorAdapter相比之前的适配器,需要的参数更多。同时还需要对Cursor这个重要资源进行管理,对应的可以看到StartManagingCursorStopManagingCursor,希望读者在实际开发中一定要注意资源的释放,以上的结果笔者就不截图了。

 

5.自定义CursorAdapter

为了某些效果我们不一定非要继承BaseAdapter从而实现很多的方法,如果 数据是来自于数据库的我们还可以继承自CursorAdapter,这样我们能够节省实现很多的方法,从而只需要关注重要部分,下面我们继承 CursorAdapter来查看需要实现哪些方法:

 1     public class CustomeCursorAdapter : Android.Widget.CursorAdapter
 2  {  3  4 public override void BindView(Android.Views.View view, Android.Content.Context context, Android.Database.ICursor cursor)  5  {  6 throw new NotImplementedException();  7  }  8  9 public override Android.Views.View NewView(Android.Content.Context context, Android.Database.ICursor cursor, Android.Views.ViewGroup parent) 10  { 11 throw new NotImplementedException(); 12  } 13 }

可以看到我们只需要实现两个方法,相比BaseAdapter而言减少了很多,其中BindView就是将ICursor中的数据与View进行绑定,而NewView则是创建所需的界面。下面我们自定一个适配器实现和上一节一样的效果:

 

 1     public class CustomeCursorAdapter : Android.Widget.CursorAdapter
 2  {  3  Activity context;  4  5 public CustomeCursorAdapter(Activity context, ICursor cursor)  6 : base(context, cursor)  7  {  8 this.context = context;  9  } 10 11 public override void BindView(Android.Views.View view, Android.Content.Context context, Android.Database.ICursor cursor) 12  { 13 TextView tv = view.FindViewById<TextView>(Android.Resource.Id.Text1); 14 tv.Text = cursor.GetString(1); 15  } 16 17 public override Android.Views.View NewView(Android.Content.Context context, Android.Database.ICursor cursor, Android.Views.ViewGroup parent) 18  { 19 return this.context.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, parent, false); 20  } 21 }

 

 6.注意点

 需要读者需要使用系统自带的SimpleAdapter时,传递数据时候不能以Dictionary类型添加,必须以JavaDictionary,否则将会出现无法将其转换为java.util.map的错误。

目录
相关文章
|
20天前
|
Java Android开发
Android 开发获取通知栏权限时会出现两个应用图标
Android 开发获取通知栏权限时会出现两个应用图标
12 0
|
25天前
|
调度 数据库 Android开发
构建高效Android应用:Kotlin协程的实践与优化
在Android开发领域,Kotlin以其简洁的语法和平台友好性成为了开发的首选语言。其中,Kotlin协程作为处理异步任务的强大工具,它通过提供轻量级的线程管理机制,使得开发者能够在不阻塞主线程的情况下执行后台任务,从而提升应用性能和用户体验。本文将深入探讨Kotlin协程的核心概念,并通过实例演示如何在实际的Android应用中有效地使用协程进行网络请求、数据库操作以及UI的流畅更新。同时,我们还将讨论协程的调试技巧和常见问题的解决方法,以帮助开发者避免常见的陷阱,构建更加健壮和高效的Android应用。
32 4
|
27天前
|
移动开发 Java Android开发
构建高效Android应用:Kotlin协程的实践之路
【2月更文挑战第31天】 在移动开发领域,性能优化和流畅的用户体验一直是开发者追求的目标。随着Kotlin语言的流行,其异步编程解决方案——协程(Coroutines),为Android应用带来了革命性的并发处理能力。本文将深入探讨Kotlin协程的核心概念、设计原理以及在Android应用中的实际应用案例,旨在帮助开发者掌握这一强大的工具,从而提升应用的性能和响应能力。
|
28天前
|
移动开发 调度 Android开发
构建高效Android应用:探究Kotlin协程的优势与实践
【2月更文挑战第30天】 在移动开发领域,尤其是针对Android平台,性能优化和应用流畅度始终是开发者关注的重点。近年来,Kotlin语言凭借其简洁性和功能性成为Android开发的热门选择。其中,Kotlin协程作为一种轻量级的线程管理解决方案,为异步编程提供了强大支持,使得编写非阻塞性代码变得更加容易。本文将深入分析Kotlin协程的核心优势,并通过实际案例展示如何有效利用协程提升Android应用的性能和响应速度。
|
1月前
|
XML 缓存 Android开发
Android开发,使用kotlin学习多媒体功能(详细)
Android开发,使用kotlin学习多媒体功能(详细)
94 0
|
1月前
|
数据库 Android开发 开发者
构建高性能微服务架构:从理论到实践构建高效Android应用:探究Kotlin协程的优势
【2月更文挑战第16天】 在当今快速迭代和竞争激烈的软件市场中,微服务架构以其灵活性、可扩展性和独立部署能力而受到企业的青睐。本文将深入探讨如何构建一个高性能的微服务系统,涵盖从理论基础到具体实现的各个方面。我们将重点讨论服务拆分策略、通信机制、数据一致性以及性能优化等关键主题,为读者提供一个清晰、实用的指南,以便在复杂多变的业务环境中构建和维护健壮的微服务体系结构。 【2月更文挑战第16天】 在移动开发领域,性能优化和流畅的用户体验是至关重要的。随着技术的不断进步,Kotlin作为一种现代编程语言,在Android开发中被广泛采用,尤其是其协程特性为异步编程带来了革命性的改进。本文旨在深入
239 5
|
1月前
|
设计模式 人工智能 开发工具
安卓应用开发:构建未来移动体验
【2月更文挑战第17天】 随着智能手机的普及和移动互联网技术的不断进步,安卓应用开发已成为一个热门领域。本文将深入探讨安卓平台的应用开发流程、关键技术以及未来发展趋势。通过分析安卓系统的架构、开发工具和框架,本文旨在为开发者提供全面的技术指导,帮助他们构建高效、创新的移动应用,以满足不断变化的市场需求。
18 1
|
17天前
|
Java Android开发 开发者
构建高效Android应用:Kotlin协程的实践与优化
在响应式编程范式日益盛行的今天,Kotlin协程作为一种轻量级的线程管理解决方案,为Android开发带来了性能和效率的双重提升。本文旨在探讨Kotlin协程的核心概念、实践方法及其在Android应用中的优化策略,帮助开发者构建更加流畅和高效的应用程序。通过深入分析协程的原理与应用场景,结合实际案例,本文将指导读者如何优雅地解决异步任务处理,避免阻塞UI线程,从而优化用户体验。
|
1月前
|
机器学习/深度学习 调度 Android开发
安卓应用开发:打造高效通知管理系统
【2月更文挑战第14天】 在移动操作系统中,通知管理是影响用户体验的关键因素之一。本文将探讨如何在安卓平台上构建一个高效的通知管理系统,包括服务、频道和通知的优化策略。我们将讨论最新的安卓开发工具和技术,以及如何通过这些工具提高通知的可见性和用户互动性,同时确保不会对用户造成干扰。
33 1
|
11天前
|
移动开发 API Android开发
构建高效Android应用:探究Kotlin协程的优势与实践
【4月更文挑战第7天】 在移动开发领域,性能优化和应用响应性的提升一直是开发者追求的目标。近年来,Kotlin语言因其简洁性和功能性在Android社区中受到青睐,特别是其对协程(Coroutines)的支持,为编写异步代码和处理并发任务提供了一种更加优雅的解决方案。本文将探讨Kotlin协程在Android开发中的应用,揭示其在提高应用性能和简化代码结构方面的潜在优势,并展示如何在实际项目中实现和优化协程。