안드로이드 기기의 기본 이메일 주소를 얻는 방법


412

Android의 기본 이메일 주소 (또는 이메일 주소 목록)는 어떻게 얻습니까?

OS 2.0 이상에서는 여러 전자 메일 주소를 지원하지만 2.0 미만에서는 장치 당 전자 메일 주소를 하나만 가질 수 있다는 것을 이해합니다.


연락처 이메일 주소 검색에 대해 이야기하고 있습니까?
Austyn Mahoney

1
아니요, 기기의 기본 이메일 주소입니다.
Brandon O'Rourke

Android 기기와 연결된 이메일 주소가 하나 이상 있습니까? 그것이 내가 원하는 것입니다.
Brandon O'Rourke

2
@ BrandonO'Rourke Android 마켓과 연결된 "기기의 기본 이메일 주소"를 의미합니까? Android 마켓과 연결된 Gmail ID와 다른 이메일간에 차이가 있기 때문입니다. 이 질문을 살펴보십시오. stackoverflow.com/questions/10606976/…
Gaurav Agarwal

답변:


749

아래와 같이 여러 가지 방법이 있습니다.

친숙한 경고로서 계정, 프로필 및 연락처 데이터를 다룰 때 사용자에게주의를 기울이고 선결하십시오. 사용자의 이메일 주소 나 기타 개인 정보를 오용하면 나쁜 일이 발생할 수 있습니다.

방법 A : AccountManager 사용 (API 레벨 5 이상)

장치에서 AccountManager.getAccounts또는 AccountManager.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" />

사용에 대한 자세한 내용 AccountManagerContact Manager 에서 찾을 수 있습니다. 은 SDK 샘플 코드에서 .

방법 B : ContactsContract.Profile 사용 (API 레벨 14+)

Android 4.0 (Ice Cream Sandwich)부터 사용자 프로필에 액세스하여 사용자의 이메일 주소를 얻을 수 있습니다. 사용자 프로필에 액세스하는 데는 두 가지 권한이 필요하기 때문에 약간 무겁지만 (아래에 더 자세히 설명되어 있음) 전자 메일 주소는 상당히 민감한 데이터 조각이므로 입장료입니다.

아래는 a를 사용하여 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" />

3
나는 귀하의 코드를 사용하여 비슷한 질문을 가지고 있습니다. 모든 Gmail ID를 내 전화와 연결할 수는 있지만 기본 질문을 원합니다. 전화 번호와 동기화하기 위해 더 많은 메일 ID를 추가 할 때와 같은 하나의 솔루션을 찾았습니다 .0 번째 위치 com.google ID를 얻는 경우 스택에 들어옵니다. 첫 번째 주소가 처음 입력되어 스택에서 0 번째 위치를 얻으므로 기본 솔루션을 얻습니다. . 다음은 코드입니다. Account [] accounts = AccountManager.get (this) .getAccountsByType ( "com.google"); 문자열 myEmailid = accounts [0] .toString (); Log.d ( "내가 원하는 이메일 아이디", myEmailid); 나는 그것이 올바른 방법을 알지 못한다.
PiyushMishra

59
프로파일 방법에는 결함이 있습니다 (제 생각에는). 내 전자 메일을 원하거나 필요로하는 응용 프로그램은 모든 연락처를 읽으려는 응용 프로그램과 비교할 때 큰 문제는 아니지만 둘 다 동일한 권한을 요구하도록 만들었습니다. 따라서 사용자는 전자 메일을 읽을 앱과 500+ 연락처를 읽는 앱의 차이점을 알 수 없습니다. 연락처를 남용하는 앱의 수가 증가함에 따라 이는 매우 실제적이고 실질적인 문제입니다!
Tom

3
@Muzikant 공식적인 진술은 아니지만 변경이 거의 불가능합니다. 그러나 사용자의 전자 메일 주소에 액세스하는 "올바른"방법은 방법 B입니다. 더 "공식적인"것이며 일부 권한이있는 것은 그러한 데이터에 접근해야하는 감도를 나타내야합니다.
로마 Nurik

15
나는 이것에 @Tom에 동의합니다. 사용자의 이름과성에 대해서만 전화의 모든 연락처 데이터에 대한 권한을 요청하는 것은 어리 석습니다.
tasomaniac

3
모든 예제 코드를 복사하는 Android 4.4에서 방법 B가 작동하지 않습니다. cursor.isAfterLast()항상 true를 반환합니다. 어떤 생각?
cprcrack

55

이것은 다른 사람들에게 유용 할 수 있습니다.

AccountPicker를 사용하여 전체 권한없이 사용자의 이메일 주소를 가져오고 프로세스를 인식하고 권한을 부여하거나 취소 할 수 있습니다.


1
이것은 매우 유용한 답변입니다. 기본 이메일은 일반적으로 Google 계정을 의미하기 때문에 선호하는 옵션이어야한다고 생각 합니다. Google 계정
Alex.F

@ Alex.F 이것은 marshmellow 이후 / 안드로이드 버전에서 작동합니까?
Eswar

27

ICS에 도입 된 Android의 AccountPicker를 사용 합니다.

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
플레이 서비스를 사용해야하며 경우에 따라 사용자가 계정을 선택해야한다는 대화 상자가 표시됩니다.
Android 개발자

AccountManager.newChooseAccountIntent ()를 사용하면 동일한 작업을 수행하며 play-services 라이브러리가 필요하지 않습니다.
Denis

최신 Android 버전의 맥락에서 계정 인증을 위해 팝업이 사용자에게 표시됩니까? 그렇다면 계정이 하나 뿐인 장치에 대해서는 어떻게 우회합니까?
Eswar

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'위험'의 허가로 정의되는, (런타임 요청 필요) : developer.android.com/reference/android/...
SagiLow

@SagiLow 어떻게 처리 했습니까? 다른 사용자에게 다른 이메일 주소를 입력하기 위해 게으른 만들기를 요청하고 싶지 않습니다. :)
codebased

1
@ codebased 나는하지 못했습니다 ... 아는 한 불가능합니다.
SagiLow

2
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

유일한 작업 솔루션 덕분에 wirling
Zulqarnain

7

안타깝게도 대답이 작동하지 않습니다.

늦었지만 제공자가 컨텐츠 URI를 변경하지 않는 한 내부 Android 이메일 애플리케이션에 대한 솔루션은 다음과 같습니다.

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;
}


2

이 방법을 사용하십시오 :

 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());

이것은 현재 앱과 관련된 계정 만 반환하는 것 같습니다. 따라서 테스트에서 "없음"을 얻습니다
csga5000

manager.getAccountsByType ( "com.google")은 이후 버전의 Android에서 작동하지 않습니다. 그리고 App.getInstance ()는 무엇입니까?
powder366


0

안드로이드 잠겨 GET_ACCOUNTS 최근에 으므로 일부 답변이 저에게 효과적이지 않았습니다. 사용자가 권한 대화 상자를 견뎌야한다는 경고와 함께 Android 7.0 에서이 작업을 수행했습니다.

AndroidManifest.xml

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

MainActivity.java

package com.example.patrick.app2;
import android.content.pm.PackageManager;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.accounts.AccountManager;
import android.accounts.Account;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.*;

public class MainActivity extends AppCompatActivity {

    final static int requestcode = 4; //arbitrary constant less than 2^16

    private static String getEmailId(Context context) {
        AccountManager accountManager = AccountManager.get(context);
        Account[] accounts = accountManager.getAccountsByType("com.google");
        Account account;
        if (accounts.length > 0) {
            account = accounts[0];
        } else {
            return "length is zero";
        }
        return account.name;
    }

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

                    String emailAddr = getEmailId(getApplicationContext());
                    ShowMessage(emailAddr);

                } else {
                    ShowMessage("Permission Denied");
                }
        }
    }

    public void ShowMessage(String email)
    {
        AlertDialog alertDialog = new AlertDialog.Builder(MainActivity.this).create();
        alertDialog.setTitle("Alert");
        alertDialog.setMessage(email);
        alertDialog.setButton(AlertDialog.BUTTON_NEUTRAL, "OK",
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                    }
                });
        alertDialog.show();
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        Context context = getApplicationContext();

        if ( ContextCompat.checkSelfPermission( context, android.Manifest.permission.GET_ACCOUNTS )
                != PackageManager.PERMISSION_GRANTED )
        {
            ActivityCompat.requestPermissions( this, new String[]
                            {  android.Manifest.permission.GET_ACCOUNTS  },requestcode );
        }
        else
        {
            String possibleEmail = getEmailId(getApplicationContext());
            ShowMessage(possibleEmail);
        }
    }
}

이것은 다른 답변과 다르지 않으며 나에게 도움이되지 않는 것처럼 보입니다. 최신 버전의 Android에서는 앱과 관련된 계정 만 반환하고 더 이상 권한이 필요하지 않기 때문에
csga5000

0

MarshMallow 운영 체제에서 작업

    btn_click=(Button) findViewById(R.id.btn_click);

    btn_click.setOnClickListener(new View.OnClickListener() {

        @Override
        public void onClick(View arg0)
        {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
            {
                int permissionCheck = ContextCompat.checkSelfPermission(PermissionActivity.this,
                        android.Manifest.permission.CAMERA);
                if (permissionCheck == PackageManager.PERMISSION_GRANTED)
                {
                    //showing dialog to select image
                    String possibleEmail=null;

                     Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
                     Account[] accounts = AccountManager.get(PermissionActivity.this).getAccounts();
                     for (Account account : accounts) {
                         if (emailPattern.matcher(account.name).matches()) {
                             possibleEmail = account.name;
                             Log.e("keshav","possibleEmail"+possibleEmail);
                         }
                     }

                    Log.e("keshav","possibleEmail gjhh->"+possibleEmail);
                    Log.e("permission", "granted Marshmallow O/S");

                } else {                        ActivityCompat.requestPermissions(PermissionActivity.this,
                            new String[]{android.Manifest.permission.READ_EXTERNAL_STORAGE,
                                    android.Manifest.permission.READ_PHONE_STATE,
                                    Manifest.permission.GET_ACCOUNTS,
                                    android.Manifest.permission.CAMERA}, 1);
                }
            } else {
// Lower then Marshmallow

                    String possibleEmail=null;

                     Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
                     Account[] accounts = AccountManager.get(PermissionActivity.this).getAccounts();
                     for (Account account : accounts) {
                         if (emailPattern.matcher(account.name).matches()) {
                             possibleEmail = account.name;
                             Log.e("keshav","possibleEmail"+possibleEmail);
                     }

                    Log.e("keshav","possibleEmail gjhh->"+possibleEmail);


            }
        }
    });

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

1
계정 받기 답변이 저에게 도움이되지 않았습니다 (0 개의 계정 반환). 버튼 콜백에서 코드를 호출해도 아무런 차이가 없음을 확인할 수 있습니다.
csga5000

이 규범은 작동하지만 최근에 시간 문제가 있으므로 우리 편을 확인하십시오
Keshav Gera

0

이 단일 행을 매니페스트에 추가하십시오 ( 권한 ).

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

그런 다음이 코드를 활동에 붙여 넣으십시오.

private ArrayList<String> getPrimaryMailId() {
    ArrayList<String> accountsList = new ArrayList<String>();
    try {
        Account[] accounts = AccountManager.get(this).getAccountsByType("com.google");
        for (Account account : accounts) {
            accountsList.add(account.name);
            Log.e("GetPrimaryMailId ", account.name);
        }
    } catch (Exception e) {
        Log.e("GetPrimaryMailId", " Exception : " + e);
    }
    return accountsList;
}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.