大家如果喜欢我的博客,请关注一下我的微博,请点击这里(http://weibo.com/kifile),谢谢
转载请标明出处(http://blog.csdn.net/kifile),再次感谢
大家在平时使用 Android 手机的时候,都会发现有些应用(例如 qq,微信,淘宝)为自己创建了账号系统,并且能够在设置页面看到他,可是当自己希望为自己的软件写一个账号系统的时候总是不知从何入手,现在我们就从头开始,一步一步打造属于自己应用的账号系统。
在进行设备账户管理的时候,我们会通过一个 AccountManager 类获取系统的账户管理类,获取的方法如下:
AccountManager mAccountManager = (AccountManager) getSystemService(ACCOUNT_SERVICE);
或者
AccountManager accountManager = AccountManager.get(context);
接下来我们需要通过 AccountManager 对象对账号系统进行操作。
1.获取账户信息
首先我们来查看一下如何获取用户已有的账户信息,如果你希望读取系统当前的账户信息,那么你首先需要在 manifest 文件中申明一个读取账户的权限,如下:
<uses-permission android:name="android.permission.GET_ACCOUNTS"/>
(1)获取所有账户信息
如果你希望获取到当前设备所有的账户信息,你可以使用:
accountManager.getAccounts();
(2)获取特定的账户信息
如果你只希望获取自己或者特定的账户信息,你就应该使用:
accountManager.getAccountsByType("com.kifile");
后面的参数是你自己定义的账户类型,怎么设置我会在接下来的文章中写出来。
然后,我们就可以通过上面获取到的工具类,读取到手机上的账户信息了,这里我写了一个 ListView 的 Adapter 用于展示当前系统中的账号信息:
public class AccountAdapter extends BaseAdapter { private Account[] mAccounts; public AccountAdapter(Account[] accounts) { this.mAccounts = accounts; } @Override public int getCount() { return mAccounts != null ? mAccounts.length : 0; } @Override public Object getItem(int position) { return mAccounts[position]; } @Override public long getItemId(int position) { return 0; } @Override public View getView(int position, View convertView, ViewGroup parent) { TextView tv = new TextView(getBaseContext()); tv.setText(mAccounts[position].name + " " + mAccounts[position].type); return tv; } }
获取当前所有系统账户信息以及设置 ListView 的 Adapter 的方法如下:
mListView.setAdapter(new AccountAdapter(mAccountManager.getAccounts()));
显示效果如下:
同你当前设备进行对比,你就会发现我们已经将当前所有的账户信息显示了出来,同样地如果你只希望显示部分的账户信息,你就可以通过 getAccountByType 获取对应的账户列表。
2.建立自己的账号服务
通过上面的部分,我们已经知道了如何获取 Android 本身现有的账号信息,现在我们就开始着手建立属于自己的账号系统吧。
你需要知道的是,如果你希望建立自己的账号系统,那么你得在 manifest 文件中声明一个关于账号的Service,如下
<service android:name="com.kifile.account.app.account.AccountService" android:enabled="true" android:exported="true"> <intent-filter> <action android:name="android.accounts.AccountAuthenticator"/> </intent-filter> <meta-data android:name="android.accounts.AccountAuthenticator" android:resource="@xml/authenticator"/> </service>
在上面的代码中,我们通过设置 intent-filter 告知系统,我们当前应用中有一个账号服务,至于具体的账号信息则放在 meta-data 中的 android:resource 文件中提供, 该文件为authenticator.xml,放置路径为 res/xml,内容如下:
<?xml version="1.0" encoding="utf-8"?> <account-authenticator xmlns:android="http://schemas.android.com/apk/res/android" android:accountType="com.kifile" android:icon="@drawable/ic_launcher" android:smallIcon="@drawable/ic_launcher" android:label="@string/app_name"/>
在这里,我们就可以向系统提供相关的账户信息,用于在 Android Setting 目录下显示对应的账号信息。例如这里,我们就定义了当前的账户类型为"com.kifile",在账户系统中显示的标签为@string/app_name 对应的 String 对象,显示的 icon 为ic_launcher。
Ok,到了这里,我们已经向系统声明了一个账户相关的服务,现在让我们来具体实现他。
Android 为我们提供了一个叫做 AbstractAccountAuthenticator 的抽象类,也是通过它来实现:
public static class Authenticator extends AbstractAccountAuthenticator { public Authenticator(Context context) { super(context); } @Override public Bundle editProperties(AccountAuthenticatorResponse response, String accountType) { return null; } @Override public Bundle addAccount(AccountAuthenticatorResponse response, String accountType, String authTokenType, String[] requiredFeatures, Bundle options) throws NetworkErrorException { return null; } @Override public Bundle confirmCredentials(AccountAuthenticatorResponse response, Account account, Bundle options) throws NetworkErrorException { return null; } @Override public Bundle getAuthToken(AccountAuthenticatorResponse response, Account account, String authTokenType, Bundle options) throws NetworkErrorException { return null; } @Override public String getAuthTokenLabel(String authTokenType) { return null; } @Override public Bundle updateCredentials(AccountAuthenticatorResponse response, Account account, String authTokenType, Bundle options) throws NetworkErrorException { return null; } @Override public Bundle hasFeatures(AccountAuthenticatorResponse response, Account account, String[] features) throws NetworkErrorException { return null; } }
从上面的代码中,我们可以看出,Android 需要从我们自定义的 Service 中获取一个 AbstractAccountAuthenticator 对象,然后再调用对象中的方法来实现账号系统的具体操作,每一个方法的意义,我们在本文中就不具体探讨了,如果你有兴趣,可以去看看 API 文档。
当我们创建好了一个AbstractAccountAuthenticator 类后,我们需要从 Service 中取得这个类的对象,代码如下:
private Authenticator authenticator; @Override public void onCreate() { super.onCreate(); authenticator = new Authenticator(this); } @Override public IBinder onBind(Intent intent) { return authenticator.getIBinder(); }
AccountService 类在 onCreate 的时候创建一个 Authenticator 对象,然后再 bindService 的时候,将 Authenticator 的 IBinder 传递回去,以供调用。
当你完成上面的步骤之后,你就会发现,在你的设置页面点击添加账户时就会出现你自定义的账户了,如下:
3.添加账户
虽然通过上面的步骤,我们已经能够在添加账户的界面看到属于我们自己的账户类别了,但是你会发现当你点击它们的时候,没有任何作用,那么我们应该怎么在设备上完成添加账户的操作呢?
(1)加入添加账户的权限
添加账户也需要对应的权限,你应该在 manifest 文件中加入
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS"/>
(2)重写 Authenticator 的 addAccount 方法
当用户在添加账户页面选择账户进行添加或者调用accountManager.addAccount 的时候,系统会默认调用 AbstractAccountAuthenticator 中的 addAccount 方法,因此你需要重写 addAccount 方法,直接添加默认账户,或者跳转到某个页面,让用户填写用户信息,然后添加账户。
(3)使用 addAccountExplicitly 直接添加账户
如果你希望直接添加账户信息,你可以使用以下方法:
Account account = new Account("Kifile,,,,12","com.kifile"); accountManager.addAccountExplicitly(account,password,userdata);
使用之后,你就会发现在设置页面出现你所建立的账户,点击进去,会发现账户名什么的也已经成功设置:
本文暂时就到这里了,通过本文,你可以初步了解到 Android 的账户信息的建立流程,希望对大家有所帮助。