如何获取Android设备的主要电子邮件地址

420

如何获取Android的主要电子邮件地址(或电子邮件地址列表)?

据我所知,在2.0及以上版本的操作系统中,支持多个电子邮件地址,但在2.0以下版本中,每个设备只能有一个电子邮件地址。


你是在谈论获取联系人的电子邮件地址吗? - Austyn Mahoney
1
不,这是设备的主要电子邮件地址。 - Brandon O'Rourke
有一个或多个电子邮件地址与Android设备相关联,对吧?那就是我想要的。 - Brandon O'Rourke
2
@BrandonO'Rourke,您是指“设备的主要电子邮件地址”是与Android市场相关联的那个吗?因为与Android市场相关联的Gmail ID和其他电子邮件之间存在差异。请参阅此问题https://dev59.com/R2PVa4cB1Zd3GeqP9uDT#tkwEoYgBc1ULPQZFUU6U - Gaurav Agarwal
13个回答

750

有几种方法可以实现这一点,如下所示。

友情提示:在处理帐户、配置文件和联系人数据时,请向用户提供谨慎和诚实的服务。如果您滥用用户的电子邮件地址或其他个人信息,可能会发生糟糕的事情。

方法A:使用AccountManager(API级别5+)

您可以使用AccountManager.getAccountsAccountManager.getAccountsByType获取设备上所有帐户名称的列表。对于某些帐户类型(包括com.google),帐户名称是电子邮件地址。下面是一个示例片段。

Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
Account[] accounts = AccountManager.get(context).getAccounts();
for (Account account : accounts) {
    if (emailPattern.matcher(account.name).matches()) {
        String possibleEmail = account.name;
        ...
    }
}

请注意,这需要GET_ACCOUNTS权限:

<uses-permission android:name="android.permission.GET_ACCOUNTS" />

关于如何使用AccountManager的更多信息,请参阅SDK中的Contact Manager示例代码。

方法B:使用ContactsContract.Profile(API级别14+)

从Android 4.0(Ice Cream Sandwich)开始,您可以通过访问用户配置文件来获取用户的电子邮件地址。访问用户配置文件有点繁重,因为它需要两个权限(下面会详细介绍),但电子邮件地址是相当敏感的数据,所以这就是门槛。

以下是一个完整的示例,它使用CursorLoader来检索包含电子邮件地址的配置文件数据行。

public class ExampleActivity extends Activity implements LoaderManager.LoaderCallbacks<Cursor> {
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getLoaderManager().initLoader(0, null, this);
    }

    @Override
    public Loader<Cursor> onCreateLoader(int id, Bundle arguments) {
        return new CursorLoader(this,
                // Retrieve data rows for the device user's 'profile' contact.
                Uri.withAppendedPath(
                        ContactsContract.Profile.CONTENT_URI,
                        ContactsContract.Contacts.Data.CONTENT_DIRECTORY),
                ProfileQuery.PROJECTION,

                // Select only email addresses.
                ContactsContract.Contacts.Data.MIMETYPE + " = ?",
                new String[]{ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE},

                // Show primary email addresses first. Note that there won't be
                // a primary email address if the user hasn't specified one.
                ContactsContract.Contacts.Data.IS_PRIMARY + " DESC");
    }

    @Override
    public void onLoadFinished(Loader<Cursor> cursorLoader, Cursor cursor) {
        List<String> emails = new ArrayList<String>();
        cursor.moveToFirst();
        while (!cursor.isAfterLast()) {
            emails.add(cursor.getString(ProfileQuery.ADDRESS));
            // Potentially filter on ProfileQuery.IS_PRIMARY
            cursor.moveToNext();
        }

        ...
    }

    @Override
    public void onLoaderReset(Loader<Cursor> cursorLoader) {
    }

    private interface ProfileQuery {
        String[] PROJECTION = {
                ContactsContract.CommonDataKinds.Email.ADDRESS,
                ContactsContract.CommonDataKinds.Email.IS_PRIMARY,
        };

        int ADDRESS = 0;
        int IS_PRIMARY = 1;
    }
}

这需要同时具备READ_PROFILEREAD_CONTACTS权限:

<uses-permission android:name="android.permission.READ_PROFILE" />
<uses-permission android:name="android.permission.READ_CONTACTS" />

4
我有一个类似的问题,使用你的代码,我能够获取与我的手机关联的所有Gmail ID,但我想要主邮箱。我发现一种解决方法是,当我们添加更多的邮件ID以与手机同步时,它们会进入一个堆栈,如果我获取第0个位置的"com.google" ID,我就能得到主邮箱,因为它首先进入并占据了堆栈中的第0个位置。这是我的一些代码: Account[] accounts=AccountManager.get(this).getAccountsByType("com.google"); String myEmailid=accounts[0].toString(); Log.d("我想要的邮箱", myEmailid);我知道这不是正确的方式。 - PiyushMishra
61
我认为,Profile Method (个人资料权限) 存在严重的缺陷。与一个需要我的电子邮件地址的应用相比,一个要读取我所有联系人的应用程序才是大问题,但你们让这两者都需要同样的权限。因此,作为用户,我无法区分一个将要读取我的电子邮件的应用和一个将要读取我500多个联系人的应用。随着滥用你的联系人的应用数量增长,这是一个非常现实、实际的问题! - Tom
3
@Muzikant 这不是官方声明,但这是一件相当不可能改变的事情。话虽如此,“正确”的访问用户电子邮件地址的方法是方法B。这更“正式”,而且它要求一些重量级权限,这表明你应该以多么谨慎的态度来处理这样的数据敏感性。 - Roman Nurik
15
我同意@Tom的观点。仅仅因为想要用户的名字和姓氏就请求访问手机上所有联系人的数据是荒谬的。 - tasomaniac
4
在 Android 4.4 上,我复制了所有的示例代码,但 Method B 对我无效。cursor.isAfterLast() 总是返回 true。有什么想法吗? - cprcrack
显示剩余15条评论

55

1
这是一个非常有帮助的答案,我认为这应该是首选选项,因为“主电子邮件”通常意味着Google账户,而Google账户则将与Google Play一起使用。 - Alex.F
@Alex.F 这个适用于安卓6.0及以上版本吗? - Eswar

28

我会使用Android的AccountPicker,该功能在ICS中引入。

Intent googlePicker = AccountPicker.newChooseAccountIntent(null, null, new String[]{GoogleAuthUtil.GOOGLE_ACCOUNT_TYPE}, true, null, null, null, null);
startActivityForResult(googlePicker, REQUEST_CODE);

然后等待结果:

protected void onActivityResult(final int requestCode, final int resultCode,
                                final Intent data) {
    if (requestCode == REQUEST_CODE && resultCode == RESULT_OK) {
        String accountName = data.getStringExtra(AccountManager.KEY_ACCOUNT_NAME);
    }
}

2
请注意,这需要您使用play-services,并且在某些情况下会显示一个对话框,用户需要选择帐户。 - android developer
1
使用 AccountManager.newChooseAccountIntent() 可以完成同样的工作,且不需要 play-services 库。 - Greelings
这是否会在最新的Android版本中向用户弹出帐户验证的弹窗?如果是,那么对于只有一个帐户的设备,我该如何绕过它? - Eswar
这是最好的答案,刚刚获得了一个真实账户,带有用户交互。 - Noor Hossain

14
public String getUsername() {
    AccountManager manager = AccountManager.get(this);
    Account[] accounts = manager.getAccountsByType("com.google");
    List<String> possibleEmails = new LinkedList<String>();

    for (Account account : accounts) {
        // TODO: Check possibleEmail against an email regex or treat
        // account.name as an email address only for certain account.type values.
        possibleEmails.add(account.name);
    }

    if (!possibleEmails.isEmpty() && possibleEmails.get(0) != null) {
        String email = possibleEmails.get(0);
        String[] parts = email.split("@");

        if (parts.length > 1)
            return parts[0];
    }
    return null;
}

简单易懂的方法。谢谢 :) - Talha Q
2
请注意,此操作需要 android.permission.GET_ACCOUNTS 权限,该权限被定义为“危险”权限(需要在运行时请求):https://developer.android.com/reference/android/Manifest.permission.html#GET_ACCOUNTS - SagiLow
@SagiLow 你是怎么处理的?我不想再向用户请求另一个权限,只是为了让他懒得输入他的电子邮件地址 :) - codebased
1
@codebased 我不知道... 据我所知,这是不可能的。 - SagiLow
3
manager.getAccountsByType("com.google"); 在较新版本的 Android 中无法使用。 - powder366

8

有一种Android API可以让用户选择他们的电子邮件地址,而无需权限。请查看:https://developers.google.com/identity/smartlock-passwords/android/retrieve-hints

HintRequest hintRequest = new HintRequest.Builder()
        .setHintPickerConfig(new CredentialPickerConfig.Builder()
                .setShowCancelButton(true)
                .build())
        .setEmailAddressIdentifierSupported(true)
        .setAccountTypes(IdentityProviders.GOOGLE)
        .build();

PendingIntent intent = mCredentialsClient.getHintPickerIntent(hintRequest);
try {
    startIntentSenderForResult(intent.getIntentSender(), RC_HINT, null, 0, 0, 0);
} catch (IntentSender.SendIntentException e) {
    Log.e(TAG, "Could not start hint picker Intent", e);
}

这将显示一个选择器,让用户选择电子邮件地址。结果将在 onActivityResult() 中返回。


我可以确认我能够使用这个,感谢你提供了我找到的唯一有效的答案。 - csga5000

7

很遗憾,已接受的答案无效。

我来晚了,但这是内部Android电子邮件应用程序的解决方案,除非内容URI由提供者更改:

Uri EMAIL_ACCOUNTS_DATABASE_CONTENT_URI = 
              Uri.parse("content://com.android.email.provider/account");

public ArrayList<String> GET_EMAIL_ADDRESSES ()
{
    ArrayList<String> names = new ArrayList<String>();
    ContentResolver cr      = m_context.getContentResolver();
    Cursor cursor           = cr.query(EMAIL_ACCOUNTS_DATABASE_CONTENT_URI ,null, 
                             null, null, null);

    if (cursor == null) {
        Log.e("TEST", "Cannot access email accounts database");
        return null;
    }

    if (cursor.getCount() <= 0) {
        Log.e("TEST", "No accounts");
        return null;
    }

    while (cursor.moveToNext()) {
        names.add(cursor.getString(cursor.getColumnIndex("emailAddress")));
        Log.i("TEST", cursor.getString(cursor.getColumnIndex("emailAddress")));
    }
    return names;
}

6

3
请使用这个方法:
 public String getUserEmail() {
    AccountManager manager = AccountManager.get(App.getInstance());
    Account[] accounts = manager.getAccountsByType("com.google");
    List<String> possibleEmails = new LinkedList<>();
    for (Account account : accounts) {
        possibleEmails.add(account.name);
    }
    if (!possibleEmails.isEmpty() && possibleEmails.get(0) != null) {
        return possibleEmails.get(0);
    }
    return "";
}

请注意,这需要GET_ACCOUNTS权限:
<uses-permission android:name="android.permission.GET_ACCOUNTS" />

然后:

editTextEmailAddress.setText(getUserEmail());

这似乎只返回与当前应用程序相关联的帐户 - 因此在测试中我得到了“none”。 - csga5000
1
manager.getAccountsByType("com.google")在较新版本的Android中无法使用。而App.getInstance()是从哪里来的? - powder366

1

适用于 Android 8 及以上版本 -

步骤 1 - 在 AndroidManifest.xml 文件中添加以下代码 -

<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.READ_CONTACTS" />

步骤2 - 在您的活动中添加以下代码,以在运行时请求权限。

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if ((ActivityCompat.checkSelfPermission(this, Manifest.permission.GET_ACCOUNTS) == PackageManager.PERMISSION_GRANTED) && (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_CONTACTS) == PackageManager.PERMISSION_GRANTED)) {
                getGoogleAccounts();
            }

            else {
                ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.GET_ACCOUNTS, Manifest.permission.READ_CONTACTS}, 1);
                //return false;
            }
        }

步骤 3 - 添加onRequestPermissionsResult的代码 -

 @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){

            getGoogleAccounts();
        }
    }

步骤 4 - 最后,添加代码使用 AccountManager 检索帐户。

private void getGoogleAccounts(){

   AccountManager am = AccountManager.get(this); // "this" references the current Context
    Account[] accounts = am.getAccountsByType("com.google");

    for (Account acc : accounts){
        System.out.println("http accounts " + acc);
    }
}

请参考以下链接了解Android 8的更改 - https://developer.android.com/about/versions/oreo/android-8.0-changes#aaad

1

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接