我正在尝试使用 GoogleAccountCredential 登录我的应用程序进行身份验证:
mGoogleAccountCredential = GoogleAccountCredential.usingOAuth2(context, Arrays.asList(Scopes.EMAIL, Scopes.PLUS_LOGIN));
mGoogleAccountCredential.setSelectedAccountName(accountName);
String token = mGoogleAccountCredential.getToken();
它在真实设备上运行良好,但在 android 模拟器上mGoogleAccountCredential.getToken()
失败,出现以下异常:
java.lang.IllegalArgumentException: the name must not be empty: null
03-01 19:41:31.604 3203-3361/com.myapp W/System.err: at android.accounts.Account.<init>(Account.java:48)
03-01 19:41:31.604 3203-3361/com.myapp W/System.err: at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
03-01 19:41:31.604 3203-3361/com.myapp W/System.err: at com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAccountCredential.getToken(GoogleAccountCredential.java:255)
- 模拟器上存在的谷歌播放服务(
GoogleApiAvailability.isGooglePlayServicesAvailable(context)
返回 0) -
accountName
在传递到setSelectedAccountName
时设置并正确(设置为"myuser@gmail.com"
) - 所有权限、依赖项和配置都存在于项目中(事实上,它适用于所有真实设备)
任何线索为什么它不在模拟器上运行?
上级:
在谷歌的代码中挖掘了一些内容后:问题出现在setSelectedAccountName(accountName)
方法中。此方法要求GoogleAccountManager
给他一个与给定帐户名称关联的帐户。如果没有此类帐户,则帐户名称设置为 null
:
public final GoogleAccountCredential setSelectedAccountName(String accountName) {
selectedAccount = accountManager.getAccountByName(accountName);
// check if account has been deleted
this.accountName = selectedAccount == null ? null : accountName;
return this;
}
反过来,AccountManager
遍历所有现有帐户,并将其名称与给定的帐户名称进行比较。如果存在匹配项,则返回相应的帐户:
public Account getAccountByName(String accountName) {
if (accountName != null) {
for (Account account : getAccounts()) {
if (accountName.equals(account.name)) {
return account;
}
}
}
return null;
}
public Account[] getAccounts() {
return manager.getAccountsByType("com.google");
}
问题是getAccounts()
在模拟器上返回空数组。但是,在真实设备上,它会返回正确的列表。
好吧,和往常一样,事情比看起来容易。
感谢这篇文章和 b1izzar 指出正确的答案。
我检查的所有真实设备都运行Android 5.1 Lollipop。
我检查的所有模拟器都运行Android 6.0 Marshmallow。
在棉花糖上,即在我的模拟器上,仅仅在清单中指定GET_ACCOUNTS
权限是不够的。必须在运行时使用特定代码请求此权限:
请求权限:
// Here, thisActivity is the current activity if (ContextCompat.checkSelfPermission(thisActivity, Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) { // Should we show an explanation? if (ActivityCompat.shouldShowRequestPermissionRationale(thisActivity, Manifest.permission.READ_CONTACTS)) { // Show an expanation to the user *asynchronously* -- don't block // this thread waiting for the user's response! After the user // sees the explanation, try again to request the permission. } else { // No explanation needed, we can request the permission. ActivityCompat.requestPermissions(thisActivity, new String[]{Manifest.permission.READ_CONTACTS}, MY_PERMISSIONS_REQUEST_READ_CONTACTS); // MY_PERMISSIONS_REQUEST_READ_CONTACTS is an // app-defined int constant. The callback method gets the // result of the request. } }
注意:在棉花糖GET_ACCOUNTS
中,WRITE_CONTACTS
和READ_CONTACTS
权限位于同一权限组中,因此一旦授予READ_CONTACTS
,也会授予GET_ACCOUNTS
。
注意2:在Android中,牛轧糖GET_ACCOUNTS
已被弃用,因此即使在棉花糖中也使用READ_CONTACTS
而不是GET_ACCOUNT
是有意义的。
也许模拟器运行的是旧版本的Google服务。 看起来最新版本会抛出GoogleAuthException而不是IllegalArgumentException。
接口文档
public String getToken()
throws IOException,
com.google.android.gms.auth.GoogleAuthException
Returns an OAuth 2.0 access token.
Must be run from a background thread, not the main UI thread.
Throws:
IOException
com.google.android.gms.auth.GoogleAuthException
问题是您必须使用物理设备进行开发和测试,因为无法在模拟器上安装Google Play服务。
我没有看到其他原因,但这里有一个代码片段取自 tasks-android-sample,它也使用 GoogleAccountCredential。