Android中联系人使用

我8月份的时候接触过联系人这里,看了很多文章,把我弄蒙了,今天突然发现这篇文章,不错,如果我以后涉及到这方面的业务,会多来学习下,作者博客地址和英文原文地址都放在最下面了。前阵子搞短信,发现Android1.x至2.0版本联系人数据库很多地方做了更改,且关于这方面的资料也比较少,所以找到一篇文章稍作翻译了下,以供大家参考,该文将分三部分发布。WorkingWithAndro

大家好,又见面了,我是你们的朋友全栈君。

我8月份的时候接触过联系人这里,看了很多文章,把我弄蒙了,今天突然发现这篇文章,不错,如果我以后涉及到这方面的业务,会多来学习下,作者博客地址和英文原文地址都放在最下面了。


前阵子搞短信,发现Android 1.x至2.0 版本联系人数据库很多地方做了更改,且关于这方面的资料也比较少,所以找到一篇文章稍作翻译了下,以供大家参考,该文将分三部分发布。

Working With Android Contacts

Introduction To Android Contacts

Learn to work with the Android contacts database. Basic knowledge of accessing SQLite in Android along with using Cursors is expected. See the Android SQLite and Cursor Article for more information. Google changed the contacts database moving from 1.x to 2.0 versions of Android. This tutorial will be broken into 3 sections. First covering accessing contacts in Android 2.0. The second page will deal with accessing the contacts in Android 1.6 and before. Third we’ll glue it all together with a class that abstracts specific classes for each version and a set of classes to manage the data from the contact records.

 

学习使用Android联系人数据库。要求懂得基本的SQLite的知识。可以查看 Android SQLite and Cursor Article相关文章以获取更多信息。从Android 1.x 至 2.0 版本谷歌改变了Android的联系人数据库。该手册主要分为三个部分:一是介绍2.0中访问名片夹;二是介绍1.6之前的版本;三我们综合了为每个版本给出一个抽象类和累积来管理名片记录数据。

 

Create a new project called TestContacts in Eclipse setup for Android 2.0.

Android Contact API For 2.0

Granting Access 授予权限

Before an application can query the contact records access must be granted through the AndroidManifest.xml file stored in the root of the project. Add the following uses-permission belows the uses-sdk statement. 在AndroidManifest.xml文件中授予以下权限

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

 

Querying The Android Contact Database 联系人数据库查询

Retrieving Contact Details 检索联系方式

Basic contact information stored in Contacts table with detailed information stored in individual tables for normalization. In Android 2.0 to query the base contact records the URI to query is stored in ContactsContract.Contacts.CONTENT_URI.

基本的个人信息存储在名片夹表,而详细的存储在个人表里。在Andoid2.0中查询相应联系记录的URI是ContactsContract.Contacts.CONTENT_URI。

  
  
  
package com.test; import android.app.Activity; import android.database.Cursor; import android.os.Bundle; import android.provider.ContactsContract; import android.provider.ContactsContract.Contacts; public class TestContacts extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main);
     ContentResolver cr = getContentResolver();  
Cursor cursor = cr.query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null); // 查询通讯录 if(cursor.getCount()>0){ while (cursor.moveToNext()) { String id = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID)); // 联系人id String name = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME)); // 联系人名称 if(cursor.getInt(cursor.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER))>0){ // Query phone here. Covered next 在该处查询电话号码 } } } } }

This application starts off as any other Android application. First create a ContentResolver isntance in cr. Then use the ContentResolver instance to query the database and return a Cursor with the contacts list. The query is perofrmed against the URI stored in ContactsContract.Contacts.CONTENT_URI. Next check if the cursor contains records and if so loop through them. The record ID field is stored in the id variable. This will be used as a where parameter later. Also the display name field is stored in the string name. For more details about working with cursors see Android Cursors Tutorial.

启动该应用程序时需关闭任何其他Android应用程序。首先,创建一个ContentResolver的实例cr。然后使用ContentResolver的实例查询数据库并返回联系人列表游标。该查询是针对ContactsContract.Contacts.CONTENT_URI 进行存储的URI。下一步检查游标是否包含记录,如果包含记录,侧记录ID字段的值存储在ID变量中。他将作为一个参数在后面的地方使用。也把名称字段的值存储在name变量中。对于游标的更多详细用法可以查看 Android的游标教程

Phone Numbers 电话号码

Phone numbers are stored in their own table and need to be queried separately. To query the phone number table use the URI stored in the SDK variable ContactsContract.CommonDataKinds.Phone.CONTENT_URI. Use a WHERE conditional to get the phone numbers for the specified contact.

电话号码存储在它们自己的表中,需要单独进行查询。要查询的电话号码表使用的是SDK中的变量ContactsContract.CommonDataKinds.Phone.CONTENT_URI存储的URI。使用WHERE条件得到指定联系人的电话号码。

  
  
  
// 根据ID查询出电话号码 Cursor pCur = cr.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, ContactsContract.CommonDataKinds.Phone.CONTACT_ID +" = ?", new String[]{id}, null); while (pCur.moveToNext()) { // Do something with phones } pCur.close();

Perform a second query against the Android contacts SQLite database. The phone numbers are queried against the URI stored in ContactsContract.CommonDataKinds.Phone.CONTENT_URI. The contact ID is stored in the phone table as ContactsContract.CommonDataKinds.Phone.CONTACT_ID and the WHERE clause is used to limit the data returned.

在Android联系人SQLite数据库中执行第二个查询。查询的电话号码是针对ContactsContract.CommonDataKinds.Phone.CONTENT_URI存储的URI。CONTACT_ID存储在电话表中,ContactsContract.CommonDataKinds.Phone.CONTACT_ID和where子句用于限制返回的数据。



Email Addresses 电子邮件地址

Querying email addresses is similar to phone numbers. A query must be performed to get email addresses from the database. Query the URI stored in ContactsContract.CommonDataKinds.Email.CONTENT_URI to query the email address table.

查询电子邮件地址类似电话号码。必须执行一个查询从数据库中获取电子邮件地址。根据存储在ContactsContract.CommonDataKinds.Email.CONTENT_URI的URI来查询电子邮件地址表。

  
  
  
Cursor emailCur = cr.query(ContactsContract.CommonDataKinds.Email.CONTENT_URI, null, ContactsContract.CommonDataKinds.Email.CONTACT_ID + " = ?", new String[]{id}, null); while (emailCur.moveToNext()) { // 如果email地址被保存在一个数组中,你将得到多个邮件地址 String email = emailCur.getString(emailCur.getColumnIndex(ContactsContract.CommonDataKinds.Email.DATA)); String emailType = emailCur.getString(emailCur.getColumnIndex(ContactsContract.CommonDataKinds.Email.TYPE)); } emailCur.close();

As with the phone query the field names for the email table are also stored under ContactsContract.CommonDataKinds. The email query is performed on the URI in ContactsContract.CommonDataKinds.Email.CONTENT_URI and the WHERE clause has to match the ContactsContract.CommonDataKinds.Email.CONTACT_ID field. Since multiple email addresses can be stored loop through the records returned in the Cursor.

正如手机查询email表中的字段名称也存在ContactsContract.CommonDataKinds。该email执行查询的URI ContactsContract.CommonDataKinds.Email.CONTENT_URI和where子句必须符合ContactsContract.CommonDataKinds.Email.CONTACT_ID领域。多个email地址可以通过存储在游标返回的记录循环。

Notes 注释

Custom notes can be attached to each contact record. As before these are stored in a separate table and are related based on the contact ID.

可以为每个联系人记录附加自定义注释。这些注释被存储在一个单独的表中,根据相关的联系人ID查询。

  
  
  
String noteWhere = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ?"; String[] noteWhereParams = new String[]{id,ContactsContract.CommonDataKinds.Note.CONTENT_ITEM_TYPE}; Cursor noteCur = cr.query(ContactsContract.Data.CONTENT_URI, null, noteWhere, noteWhereParams, null); if (noteCur.moveToFirst()) { String note = noteCur.getString(noteCur.getColumnIndex(ContactsContract.CommonDataKinds.Note.NOTE)); } noteCur.close();

Notes are stored in the Android Contacts generic data table. When accessing specific data the WHERE clause will need 2 conditionals. First the standard contact ID, second a MIMETYPE for the data that is being requested. The Android SDK comes with a series of auto-generated variables that take care of this. Use the ContactsContract.CommonDataKinds.Note.CONTENT_ITEM_TYPE variable to limit the query to note records. The data table URI is stored at ContactsContract.Data.CONTENT_URI. Finally the note field name is stored in ContactsContract.CommonDataKinds.Note.NOTE.

注释存储在Android联系人通用数据表中。当访问指定数据时where子句将需要2个条件。首先是标准的联系人ID,第二个是对那些被请求数据的媒体类型。在Android SDK中有一系列自动生成的变量处理这个。使用ContactsContract.CommonDataKinds.Note.CONTENT_ITEM_TYPE来限要查询的记录。数据表URI存放在ContactsContract.Data.CONTENT_URI。最后,注意字段名称存储在ContactsContract.CommonDataKinds.Note.NOTE。

Postal Addresses 邮政地址

Android can store multiple postal addresses per contact. Addresses are also stored in the data table like notes and queried via the URI stored in ContactsContract.Data.CONTENT_URI. Similar to the notes query a MIMETYPE must be added to the WHERE conditional. Also in Android 2.0 the Address record was split into multiple fields containing different parts of the address (PO-Box, stree, city, region, postal code). In earlier versions of the Android SDK this was a free-form string storage.

Android的每个联系人都可以多个邮政地址。地址也存储在Notes数据表中,并通过存储在ContactsContract.Data.CONTENT_URI的URI查询。类似于注释查询,媒体类型必须被添加到where条件。另外,在Android2.0中,邮政地址记录被分割成多个小地址(邮政信箱、应力[不知道什么意思]、城市、地区、邮政编码)。在早期的Android SDK版本中,该地址是由一个自由格式的字符串存储。

  
  
  
String addrWhere = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ?"; String[] addrWhereParams = new String[]{id,ContactsContract.CommonDataKinds.StructuredPostal.CONTENT_ITEM_TYPE}; Cursor addrCur = cr.query(ContactsContract.Data.CONTENT_URI, null, addrWhere, addrWhereParams, null); while(addrCur.moveToNext()) { String poBox = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.POBOX)); String street = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.STREET)); String city = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.CITY)); String region = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.REGION)); String postalCode = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.POSTCODE)); String country = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.COUNTRY)); String type = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.TYPE)); } addrCur.close();

This code is similar to the previous example. Notice the field names for the address pieces are stored in ContactsContract.CommonDataKinds.StructuredPostal.

此代码类似于前面的示例。请注意该地址块的字段名称都存储在ContactsContract.CommonDataKinds.StructuredPostal。

Instant Messenger (IM) 即时消息

The instant messenger query performs just as the notes and address queries. Important field names for IM related data are stored in ContactsContract.CommonDataKinds.Im.

即时消息查询仅作为注释及地址查询。IM相关数据的重要字段名称存储在ContactsContract.CommonDataKinds.Im。

  
  
  
String imWhere = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ?"; String[] imWhereParams = new String[]{id,ContactsContract.CommonDataKinds.Im.CONTENT_ITEM_TYPE}; Cursor imCur = cr.query(ContactsContract.Data.CONTENT_URI, null, imWhere, imWhereParams, null); if (imCur.moveToFirst()) { String imName = imCur.getString(imCur.getColumnIndex(ContactsContract.CommonDataKinds.Im.DATA)); String imType = imCur.getString(imCur.getColumnIndex(ContactsContract.CommonDataKinds.Im.TYPE)); } imCur.close();

Organizations 组织

The last part of the contact record to be covered is the Organizations data. The Android contact record can contain information about Employment, professional, and social memberships as well as roles and titles. These records are queried from the URI stored in ContactsContract.Data.CONTENT_URI. Important field names for the organization data are stored in ContactsContract.CommonDataKinds.Organization.

联系人记录的最后一部分是组织的数据。Android的联系人记录可以包含有关就业、职业信息、社会成员以及角色和职称。这些记录从C​​ontactsContract.Data.CONTENT_URI存储的URI查询。组织数据的重要字段名称存储在ContactsContract.CommonDataKinds.Organization。

  
  
  
String orgWhere = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ?"; String[] orgWhereParams = new String[]{id,ContactsContract.CommonDataKinds.Organization.CONTENT_ITEM_TYPE}; Cursor orgCur = cr.query(ContactsContract.Data.CONTENT_URI, null, orgWhere, orgWhereParams, null); if (orgCur.moveToFirst()) { String orgName = orgCur.getString(orgCur.getColumnIndex(ContactsContract.CommonDataKinds.Organization.DATA)); String title = orgCur.getString(orgCur.getColumnIndex(ContactsContract.CommonDataKinds.Organization.TITLE)); } orgCur.close();



API For 1.6 and Before  1.6之前的版本

If you just read the begining of this article the first bit of this page is going to look familiar. This page is designed to act as a standalone guide to working with the contacts API in Android 1.6 and before. 假如你刚刚阅读了该文的开头是否有种时曾相识的感觉。该文主要是用来介绍1.6以下版本联系人的API。

Granting Access 授予权限

Before an application can query the contact records access must be granted through the AndroidManifest.xml file stored in the root of the project. Add the following uses-permission belows the uses-sdk statement. 同Android Contacts的使用(一)

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

Querying the contact database 联系人数据库查询

Retrieving Contact Details 获取联系方式

Basic contact information stored in Contacts table with detailed information stored in individual tables for normalization. In Android 1.x to query the base contact records the URI to query is stored in People.CONTENT_URI. 基本的联系人信息存储在联系人表中,而详细信息存储在个人表中。在 Android1.x 中查询的联系人记录数据库的URI是People.CONTENT_URI。

    
    
    
package com.test; import android.app.Activity; import android.content.ContentResolver; import android.database.Cursor; import android.os.Bundle; import android.provider.Contacts.People; public class TestContacts extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.main); ContentResolver cr = getContentResolver(); Cursor cur = cr.query(People.CONTENT_URI, null , null , null , null ); if (cur.getCount() > 0 ){ while (cur.moveToNext()) { String id = cur.getString(cur.getColumnIndex(People._ID)); String name = cur.getString(cur.getColumnIndex(People.DISPLAY_NAME)); } } } }

Start off with the standard view loading. Then we create a ContentResolver instance that will be used to query the SQLite database that stores the contacts. The ContentResolver query returns a Cursor instance that holds the contact records queried from the database. Then take the ID field from the contact record and store it in the string id and take the DISPLAY_NAME field and place it in the string name. For more information about cursors see the Android Cursor Tutorial.  参考Android Contacts的使用(一)

Phone Numbers 电话号码

Phone numbers are stored in their own table and need to be queried separately. To query the phone number table use the URI stored in the SDK variable Contacts.Phones.CONTENT_URI. Use a WHERE conditional to get the phone numbers for the specified contact. 同Android Contact的使用(一),查询的URI换成Contacts.Phones.CONTENT_URI。

    
    
    
if (cur.getInt(cur.getColumnIndex(People.PRIMARY_PHONE_ID)) > 0 ) { Cursor pCur = cr.query(Contacts.Phones.CONTENT_URI, null , Contacts.Phones.PERSON_ID + " = ? " , new String[]{id}, null ); int i = 0 ; int pCount = pCur.getCount(); String[] phoneNum = new String[pCount]; String[] phoneType = new String[pCount]; while (pCur.moveToNext()) { phoneNum[i] = pCur.getString(pCur.getColumnIndex(Contacts.Phones.NUMBER)); phoneType[i] = pCur.getString(pCur.getColumnIndex(Contacts.Phones.TYPE)); i ++ ; } }

Query the phones table and get a Cursor stored in pCur. Since the Android contacts database can store multiple phone numbers per contact we need to loop through the returned results. In addition to returning the phone number the query also returned the type of number (home, work, mobile, etc). 查询电话表,返回一个游标pCur。由于Android中每个联系人可以存储多个电话号码,我们需要遍历返回的结果。查询除了返回电话号码外还返回了其他(家庭,工作,手机等)类型。

Email Addresses 邮件地址

Querying email addresses is similar to phone numbers. A special query must be performed to get email addresses from the database. Query the URI stored in Contacts.ContactMethods.CONTENT_EMAIL_URI to query the email addresses.  同Android Contact的使用(一),查询的URI换成Contacts.ContactMethods.CONTENT_EMAIL_URI。

    
    
    
Cursor emailCur = cr.query(Contacts.ContactMethods.CONTENT_EMAIL_URI, null , Contacts.ContactMethods.PERSON_ID + " = ? " , new String[]{id}, null ); while (emailCur.moveToNext()) { // This would allow you get several email addresses } emailCur.close();

Simple query Contacts.ContactMethods.CONTENT_EMAIL_URI with a conditional limiting the results to numbers that match the ID of the contact record matches the value in the field Contacts.ContactMethods.PERSON_ID. As with phone numbers each contact can contain multiple email addresses so we need to loop through the Cursor records. 查询Contacts.ContactMethods.CONTENT_EMAIL_URI,以联系人记录的ID为条件匹配Contacts.ContactMethods.PERSON_ID的值。正如电话号码每个联系人也可以有多个email地址,同样的我们也需要遍历返回的结果。

Notes 注释

Custom notes can be attached to each contact record. Notes though are stored in the main contact record and are simply accessed through the data stored in People.NOTES. 可以为每个联系人附加自定义注释。注释存储在联系人记录中,并通过People.NOTES存储的数据进行访问。

    
    
    
String notes = cur.getString(cur.getColumnIndex(People.NOTES));


Postal Addresses 邮政地址

Android can store multiple postal addresses per contact. Addresses are stored in the contact methods table and need to have a second conditional added to retrieve the data. Add a conditional Contacts.ContactMethods.KIND that matches Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE to only query postal addresses from Contacts.ContactMethods.CONTENT_URI. 每个联系人都可以存储多个邮政地址。地址存储在联系方式表中,要获取数据,需要有次要条件。添加适配Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE的条件Contacts.ContactMethods.KIND来访问Contacts.ContactMethods.CONTENT_URI传递过来的地址。

    
    
    
String addrWhere = Contacts.ContactMethods.PERSON_ID + " = ? AND " + Contacts.ContactMethods.KIND + " = ? " ; String[] addrWhereParams = new String[]{id, Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE}; Cursor addrCur = cr.query(Contacts.ContactMethods.CONTENT_URI, null , addrWhere, addrWhereParams, null ); while (addrCur.moveToNext()) { String addr = addrCur.getString( addrCur.getColumnIndex(Contacts.ContactMethodsColumns.DATA)); String type = addrCur.getString( addrCur.getColumnIndex(Contacts.ContactMethodsColumns.TYPE)); } addrCur.close();

Query Contacts.ContactMethods.CONTENT_URI with 2 conditionals, one limiting the contact ID and the second Contacts.ContactMethods.KIND matching Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE to only query postall addresses. Android can store multiple postal addresses so loop through the list of returned results. Android also stores a type record for the address. In Android 1.6 and before the address is stored as a free-form string containing the data. In Android 2.0 and later this has changed to being a series of fields containing parts of the address. 查询Contacts.ContactMethods.CONTENT_URI需要2个条件,一个是联系人的ID另一个是通过Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE来匹配Contacts.ContactMethods.KIND查询postal地址。Android可以通过循环来遍历返回的结果列表中的多个postal地址,他也存储了地址类型的记录。在Android1.6或更早的版本中,该地址是由一个自由格式的字符串存储,在2.0或更高的版本中,已经更改为多个小地址存储。

Instant Messenger (IM)  即时消息

The instant messenger query works just like the previous 2. The data is queried from Contacts.ContactMethods.CONTENT_URI and needs conditionals for the contact ID and Contacts.ContactMethods.KIND matching Contacts.ContactMethods.CONTENT_IM_ITEM_TYPE. 即时消息查询类似于postal地址查询,查询这些数据也需要联系人的Id和Contacts.ContactMethods.CONTENT_IM_ITEM_TYPE来匹配Contacts.ContactMethods.KIND为条件。

    
    
    
String imWhere = Contacts.ContactMethods.PERSON_ID + " = ? AND " + Contacts.ContactMethods.KIND + " = ? " ; String[] imWhereParams = new String[]{id, Contacts.ContactMethods.CONTENT_IM_ITEM_TYPE}; Cursor imCur = cr.query(Contacts.ContactMethods.CONTENT_URI, null , imWhere, imWhereParams, null ); if (imCur.moveToFirst()) { String imName = imCur.getString( imCur.getColumnIndex(Contacts.ContactMethodsColumns.DATA)); String imType = imCur.getString( imCur.getColumnIndex(Contacts.ContactMethodsColumns.TYPE)); } imCur.close();

Organizations 组织

The last part of the contact record to be covered is the Organizations data. The Android contact record can contain information about Employment, professional, and social memberships as well as roles and titles. These records are queried from the URI stored in Contacts.Organizations.CONTENT_URI. 联系人记录的最后一部分数据组织数据。在Android中联系记录可以包含有关就业、职业信息、社会成员以及角色和职称等信息。这些记录需从Contacts.Organizations.CONTENT_URI存储的URI查询。

    
    
    
String orgWhere = Contacts.ContactMethods.PERSON_ID + " = ? " ; String[] orgWhereParams = new String[]{id}; Cursor orgCur = cr.query(Contacts.Organizations.CONTENT_URI, null , orgWhere, orgWhereParams, null ); if (orgCur.moveToFirst()) { String orgName = orgCur.getString( orgCur.getColumnIndex(Contacts.Organizations.COMPANY)); String title = orgCur.getString( orgCur.getColumnIndex(Contacts.Organizations.TITLE)); } orgCur.close();

Working With Android Contacts

Gluing it together 整合

To put this together into an application there are a few glue pieces that need to be setup along with creating classes to manage accessing the data. First we need to create a set of classes to hold the data. Also we’ll create a class to handle 2.0 API calls and a class to handle 1.6 and earlier API calls. There’s also a wrapper class that determines and loads the proper class.

需把整合类一起放在程序里,同时创建类来管理访问这些数据。(该处实在是翻译不通)。首先,我们需要创建一个类用来保存数据。此外,我们将创建一个类用来处理2.0API 调用和一个类用来处理1.6 或更早的 API 调用。还有一个包装类,用来确定并加载适当的类。

Contact Data Classes 联系人数据类

The contact classes are a series of classes to hold a list of contacts. The list is stored in the class ContactList that maintains an ArrayList of Contacts. The Contact objects are represented in the Contact class. The contact class stores all the data from the Android contact record. In addition to the ContactList and Contact classes there are specialized classes to represent some of the record data.

Contact类是一系列保持名片夹列表的类。该类存储在ContactList类中,此类维护着联系人列表。Contact类表示联系人对象,该类保存着所有从Android查询出的联系人数据。另外,在ContactList和Contact类中,有特定的类来表示某些记录的信息。

We will create classes to represent the address, email, instant messenger, phone number, and organization(s). Most of these classes are mere data storage classes with variables and getter/setters.

我们将创建一个实体类来表示地址、email、即时通讯、电话号码、组织等信息。该类大部分数据是单纯的变量,和一些getter/setters方法。


ContactList

The ContactList class is a very basic class designed to hold an ArrayList of instances of the Contact class below. We’ve left this class very plain and ready to be expanded to suit your needs.

ContactList是一个基类,设计一个ArrayList数组用来存储Contact实例。这样降低了耦合度,可以随时扩大以满足需求。

    
    
    
package com.test; import java.util.ArrayList; public class ContactList { private ArrayList < Contact > contacts = new ArrayList < Contact > (); public ArrayList < Contact > getContacts() { return contacts; } public void setContacts(ArrayList < Contact > contacts) { this .contacts = contacts; } public void addContact(Contact contact) { this .contacts.add(contact); } public ContactList() { } }

Contact

The Contact class is used to store the details about each contact. There are a series of private class variables to hold this data. Singular data such as name and database ID are stored as strings. Complex data is stored either as an instance or ArrayList of data specific classes. This class is mainly getters and setters with a few methods to add to the internal ArrayLists.

Contact类用来存储每个联系人的详细信息。该类定义了一些私有变量来保存这些数据。如名称和id定义成String型,而复杂的数据侧定义成数组或实例。这个类主要是生成一个getters和setters方法以便添加到数组中。

    
    
    
package com.test; import java.util.ArrayList; public class Contact { private String id; private String displayName; private ArrayList < Phone > phone; private ArrayList < Email > email; private ArrayList < String > notes; private ArrayList < Address > addresses = new ArrayList < Address > (); private ArrayList < IM > imAddresses; private Organization organization; public String getId() { return id; } public void setId(String id) { this .id = id; } public String getDisplayName() { return displayName; } public void setDisplayName(String displayName) { this .displayName = displayName; } public ArrayList < Phone > getPhone() { return phone; } public void setPhone(ArrayList < Phone > phone) { this .phone = phone; } public void addPhone(Phone phone) { this .phone.add(phone); } public ArrayList < Email > getEmail() { return email; } public void setEmail(ArrayList < Email > email) { this .email = email; } public void addEmail(Email email) { this .email.add(email); } public ArrayList < String > getNotes() { return notes; } public void setNotes(ArrayList < String > notes) { this .notes = notes; } public void AddNotes(String notes){ this .notes.add(notes); } public ArrayList < Address > getAddresses() { return addresses; } public void setAddresses(ArrayList < Address > addresses) { this .addresses = addresses; } public void addAddress(Address address) { this .addresses.add(address); } public ArrayList < IM > getImAddresses() { return imAddresses; } public void setImAddresses(ArrayList < IM > imAddresses) { this .imAddresses = imAddresses; } public void addImAddresses(IM imAddr) { this .imAddresses.add(imAddr); } public Organization getOrganization() { return organization; } public void setOrganization(Organization organization) { this .organization = organization; } }


Address

The Address class is the only class in the ContactList framework that actually does any work. Due to differences in data storage between 1.x and 2.0 versions of Android the Address class has to determine if the input address is free-form from 1.x or was input from the formatted data structure from 2.0. Android 2.0 has individual data columns for PO-Box, street, city, region, postal code, country while 1.x was just a text field with the entire address. This is handled in the toString() method that returns a free-form address from either input type. Unfortunately when using Android 1.x all the individual address getters will return null.

地址类是联系人列表框架中负责所有操作的唯一类。鉴于1.x 和2.0 版本中数据存储不同,Android类必须确认输入的地址是1.x 以上的自由格式还是2.0的格式化数据结构。Android 2.0 有单独的数据列给PO-Box,street,city,region,postal code,country 而1.x整个地址中只是一个文本域,并在toString()方法中处理,该方法返回一个没有格式的地址或者输入类型。不幸的是使用Android1.x 所有的单独地址获取都将返回null 。

    
    
    
package com.test; public class Address { private String poBox; private String street; private String city; private String state; private String postalCode; private String country; private String type; private String asString = "" ; public String getType() { return type; } public void setType(String type) { this .type = type; } public String getPoBox() { return poBox; } public void setPoBox(String poBox) { this .poBox = poBox; } public String getStreet() { return street; } public void setStreet(String street) { this .street = street; } public String getCity() { return city; } public void setCity(String city) { this .city = city; } public String getState() { return state; } public void setState(String state) { this .state = state; } public String getPostalCode() { return postalCode; } public void setPostalCode(String postalCode) { this .postalCode = postalCode; } public String getCountry() { return country; } public void setCountry(String country) { this .country = country; } public String toString() { if ( this .asString.length() > 0 ) { return ( this .asString); } else { String addr = "" ; if ( this .getPoBox() != null ) { addr = addr + this .getPoBox() + " n " ; } if ( this .getStreet() != null ) { addr = addr + this .getStreet() + " n " ; } if ( this .getCity() != null ) { addr = addr + this .getCity() + " , " ; } if ( this .getState() != null ) { addr = addr + this .getState() + " " ; } if ( this .getPostalCode() != null ) { addr = addr + this .getPostalCode() + " " ; } if ( this .getCountry() != null ) { addr = addr + this .getCountry(); } return (addr); } } public Address(String asString, String type) { this .asString = asString; this .type = type; } public Address(String poBox, String street, String city, String state, String postal, String country, String type) { this .setPoBox(poBox); this .setStreet(street); this .setCity(city); this .setState(state); this .setPostalCode(postal); this .setCountry(country); this .setType(type); } }

Email

Another getter/setter and data storage class. The email class stores the email address and address type (work, home, etc).

另一个生成getter/setter的数据存储类。该类存储的是email的邮件地址和地址类型(工作、家庭等)。

    
    
    
package com.test; public class Email { private String address; private String type; public String getAddress() { return address; } public void setAddress(String address) { this .address = address; } public String getType() { return type; } public void setType(String t) { this .type = t; } public Email(String a, String t) { this .address = a; this .type = t; } }

IM

Class to hold instant messenger data. 该类用来保存即时消息数据。

    
    
    
package com.test; public class IM { private String name; private String type; public String getName() { return name; } public void setName(String name) { this .name = name; } public String getType() { return type; } public void setType(String type) { this .type = type; } public IM(String name, String type) { this .name = name; this .type = type; } }

Organization

Class to hold the contacts organizational data. 该类用来保存联系人组织数据。

    
    
    
package com.test; public class Organization { private String organization = "" ; private String title = "" ; public String getOrganization() { return organization; } public void setOrganization(String organization) { this .organization = organization; } public String getTitle() { return title; } public void setTitle(String title) { this .title = title; } public Organization() { } public Organization(String org, String title) { this .organization = org; this .title = title; } }


Phone

Class to hold the phone records.  该类用来保存电话记录。

    
    
    
package com.test; public class Phone { private String number; private String type; public String getNumber() { return number; } public void setNumber(String number) { this .number = number; } public String getType() { return type; } public void setType(String type) { this .type = type; } public Phone(String n, String t) { this .number = n; this .type = t; } }

Wrapper class 封装类

The wrapper class below is what will be invoked by applications. This class will determine the API level running on the device/emulator and load the correct class created on the next pages. To determine the correct Android API running the Build.VERSION.SDK variable is queried. This version code is then compared against the Eclair (2.0) version code stored in Build.VERSION_CODES.ECLAIR. Finally the proper API class is loaded.

下面的这个封装类是由应用程序所调用的。该类确认运行在设备/模拟器上API的版本。要确认正确的运行着的Android API需要访问VERSION.SDK变量。取得版本号然后跟存于Build.VERSION_CODES.ECLAIR的Eclair(2.0)版本号相比较。最好加载相应的API。

    
    
    
package com.test; import android.content.ContentResolver; import android.content.Intent; import android.database.Cursor; import android.os.Build; public abstract class ContactAPI { private static ContactAPI api; public static ContactAPI getAPI() { if (api == null ) { String apiClass; if (Integer.parseInt(Build.VERSION.SDK) >= Build.VERSION_CODES.ECLAIR) { apiClass = " com.highercollaboration.android.ContactAPI.ContactAPISdk5 " ; } else { apiClass = " com.highercollaboration.android.ContactAPI.ContactAPISdk3 " ; } try { Class <? extends ContactAPI > realClass = Class.forName(apiClass). asSubclass(ContactAPI. class ); api = realClass.newInstance(); } catch (Exception e) { throw new IllegalStateException(e); } } return api; } public abstract Intent getContactIntent(); public abstract ContactList newContactList(); public abstract Cursor getCur(); public abstract void setCur(Cursor cur); public abstract ContentResolver getCr(); public abstract void setCr(ContentResolver cr); }


2.0 Data access class 2.0数据访问类

This class takes what was covered on page 1 of the tutorial about the Android 2.0 Contact API and turns it into a class. This class extends and will be invoked by the wrapper class created previously.

这个类在前面的Android Contacts的使用(一)中介绍了关于Android 2.0 联系人API 的教程并把他封装成一个类。这个类扩展并由先前建立的封装类调用。

    
    
    
package com.test; import java.util.ArrayList; import android.content.ContentResolver; import android.content.Intent; import android.database.Cursor; import android.provider.ContactsContract; public class ContactAPISdk5 extends ContactAPI { private Cursor cur; private ContentResolver cr; public Cursor getCur() { return cur; } public void setCur(Cursor cur) { this .cur = cur; } public ContentResolver getCr() { return cr; } public void setCr(ContentResolver cr) { this .cr = cr; } public Intent getContactIntent() { return ( new Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI)); } public ContactList newContactList() { ContactList contacts = new ContactList(); String id; this .cur = this .cr.query(ContactsContract.Contacts.CONTENT_URI, null , null , null , null ); if ( this .cur.getCount() > 0 ) { while (cur.moveToNext()) { Contact c = new Contact(); id = cur.getString(cur.getColumnIndex(ContactsContract.Contacts._ID)); c.setId(id); c.setDisplayName(cur.getString(cur.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME))); if (Integer.parseInt(cur.getString(cur.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER))) > 0 ) { c.setPhone( this .getPhoneNumbers(id)); } c.setEmail( this .getEmailAddresses(id)); c.setNotes( this .getContactNotes(id)); c.setAddresses( this .getContactAddresses(id)); c.setImAddresses( this .getIM(id)); c.setOrganization( this .getContactOrg(id)); contacts.addContact(c); } } return (contacts); } public ArrayList < Phone > getPhoneNumbers(String id) { ArrayList < Phone > phones = new ArrayList < Phone > (); Cursor pCur = this .cr.query( ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null , ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ? " , new String[]{id}, null ); while (pCur.moveToNext()) { phones.add( new Phone( pCur.getString(pCur.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER)) , pCur.getString(pCur.getColumnIndex(ContactsContract.CommonDataKinds.Phone.TYPE)) )); } pCur.close(); return (phones); } public ArrayList < Email > getEmailAddresses(String id) { ArrayList < Email > emails = new ArrayList < Email > (); Cursor emailCur = this .cr.query( ContactsContract.CommonDataKinds.Email.CONTENT_URI, null , ContactsContract.CommonDataKinds.Email.CONTACT_ID + " = ? " , new String[]{id}, null ); while (emailCur.moveToNext()) { // This would allow you get several email addresses Email e = new Email(emailCur.getString(emailCur.getColumnIndex(ContactsContract.CommonDataKinds.Email.DATA)) ,emailCur.getString(emailCur.getColumnIndex(ContactsContract.CommonDataKinds.Email.TYPE)) ); emails.add(e); } emailCur.close(); return (emails); } public ArrayList < String > getContactNotes(String id) { ArrayList < String > notes = new ArrayList < String > (); String where = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ? " ; String[] whereParameters = new String[]{id, ContactsContract.CommonDataKinds.Note.CONTENT_ITEM_TYPE}; Cursor noteCur = this .cr.query(ContactsContract.Data.CONTENT_URI, null , where, whereParameters, null ); if (noteCur.moveToFirst()) { String note = noteCur.getString(noteCur.getColumnIndex(ContactsContract.CommonDataKinds.Note.NOTE)); if (note.length() > 0 ) { notes.add(note); } } noteCur.close(); return (notes); } public ArrayList < Address > getContactAddresses(String id) { ArrayList < Address > addrList = new ArrayList < Address > (); String where = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ? " ; String[] whereParameters = new String[]{id, ContactsContract.CommonDataKinds.StructuredPostal.CONTENT_ITEM_TYPE}; Cursor addrCur = this .cr.query(ContactsContract.Data.CONTENT_URI, null , where, whereParameters, null ); while (addrCur.moveToNext()) { String poBox = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.POBOX)); String street = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.STREET)); String city = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.CITY)); String state = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.REGION)); String postalCode = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.POSTCODE)); String country = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.COUNTRY)); String type = addrCur.getString(addrCur.getColumnIndex(ContactsContract.CommonDataKinds.StructuredPostal.TYPE)); Address a = new Address(poBox, street, city, state, postalCode, country, type); addrList.add(a); } addrCur.close(); return (addrList); } public ArrayList < IM > getIM(String id) { ArrayList < IM > imList = new ArrayList < IM > (); String where = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ? " ; String[] whereParameters = new String[]{id, ContactsContract.CommonDataKinds.Im.CONTENT_ITEM_TYPE}; Cursor imCur = this .cr.query(ContactsContract.Data.CONTENT_URI, null , where, whereParameters, null ); if (imCur.moveToFirst()) { String imName = imCur.getString(imCur.getColumnIndex(ContactsContract.CommonDataKinds.Im.DATA)); String imType; imType = imCur.getString(imCur.getColumnIndex(ContactsContract.CommonDataKinds.Im.TYPE)); if (imName.length() > 0 ) { IM im = new IM(imName, imType); imList.add(im); } } imCur.close(); return (imList); } public Organization getContactOrg(String id) { Organization org = new Organization(); String where = ContactsContract.Data.CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ? " ; String[] whereParameters = new String[]{id, ContactsContract.CommonDataKinds.Organization.CONTENT_ITEM_TYPE}; Cursor orgCur = this .cr.query(ContactsContract.Data.CONTENT_URI, null , where, whereParameters, null ); if (orgCur.moveToFirst()) { String orgName = orgCur.getString(orgCur.getColumnIndex(ContactsContract.CommonDataKinds.Organization.DATA)); String title = orgCur.getString(orgCur.getColumnIndex(ContactsContract.CommonDataKinds.Organization.TITLE)); if (orgName.length() > 0 ) { org.setOrganization(orgName); org.setTitle(title); } } orgCur.close(); return (org); } }


1.x Data access class 1.x数据访问类

This class is made up of the code explained covering version 1.x of the Android contact API. As with the 2.0 class this class extends the ContactAPI wrapper class and will be invoked by it if the 1.x version of Android is in use.

这个类由Android 1.0 联系人版本介绍的代码组成。正如2.0中,这个类扩展了ContactAPI封装类,并且封装类调用这个类,当然前提是使用1.x 版本。

    
    
    
package com.test; import java.util.ArrayList; import android.content.ContentResolver; import android.content.Intent; import android.database.Cursor; import android.provider.Contacts; import android.provider.Contacts.People; public class ContactAPISdk3 extends ContactAPI { private Cursor cur; private ContentResolver cr; public Cursor getCur() { return cur; } public void setCur(Cursor cur) { this .cur = cur; } public ContentResolver getCr() { return cr; } public void setCr(ContentResolver cr) { this .cr = cr; } public Intent getContactIntent() { return ( new Intent(Intent.ACTION_PICK, People.CONTENT_URI)); } public ContactList newContactList() { ContactList contacts = new ContactList(); String id = "" ; this .cur = this .cr.query(People.CONTENT_URI, null , null , null , null ); if ( this .cur.getCount() > 0 ) { while (cur.moveToNext()) { Contact c = new Contact(); id = cur.getString(cur.getColumnIndex(People._ID)); c.setId(id); c.setDisplayName(cur.getString(cur.getColumnIndex(People.DISPLAY_NAME))); if (Integer.parseInt(cur.getString(cur.getColumnIndex(People.PRIMARY_PHONE_ID))) > 0 ) { c.setPhone( this .getPhoneNumbers(id)); } c.setEmail( this .getEmailAddresses(id)); ArrayList < String > notes = new ArrayList < String > (); notes.add(cur.getString(cur.getColumnIndex(People.NOTES))); c.setNotes(notes); c.setAddresses( this .getContactAddresses(id)); c.setImAddresses( this .getIM(id)); c.setOrganization( this .getContactOrg(id)); contacts.addContact(c); } } return (contacts); } public ArrayList < Phone > getPhoneNumbers(String id) { ArrayList < Phone > phones = new ArrayList < Phone > (); Cursor pCur = this .cr.query( Contacts.Phones.CONTENT_URI, null , Contacts.Phones.PERSON_ID + " = ? " , new String[]{id}, null ); while (pCur.moveToNext()) { phones.add( new Phone( pCur.getString(pCur.getColumnIndex(Contacts.Phones.NUMBER)) , pCur.getString(pCur.getColumnIndex(Contacts.Phones.TYPE)) )); } pCur.close(); return (phones); } public ArrayList < Email > getEmailAddresses(String id) { ArrayList < Email > emails = new ArrayList < Email > (); Cursor emailCur = this .cr.query( Contacts.ContactMethods.CONTENT_EMAIL_URI, null , Contacts.ContactMethods.PERSON_ID + " = ? " , new String[]{id}, null ); while (emailCur.moveToNext()) { // This would allow you get several email addresses Email e = new Email(emailCur.getString(emailCur.getColumnIndex(Contacts.ContactMethods.DATA)) ,emailCur.getString(emailCur.getColumnIndex(Contacts.ContactMethods.CONTENT_EMAIL_TYPE)) ); emails.add(e); } emailCur.close(); return (emails); } public ArrayList < Address > getContactAddresses(String id) { ArrayList < Address > addrList = new ArrayList < Address > (); String where = Contacts.ContactMethods.PERSON_ID + " = ? AND " + Contacts.ContactMethods.KIND + " = ? " ; String[] whereParameters = new String[]{id, Contacts.ContactMethods.CONTENT_POSTAL_ITEM_TYPE}; Cursor addrCur = this .cr.query(Contacts.ContactMethods.CONTENT_URI, null , where, whereParameters, null ); while (addrCur.moveToNext()) { String addr = addrCur.getString(addrCur.getColumnIndex(Contacts.ContactMethodsColumns.DATA)); String type = addrCur.getString(addrCur.getColumnIndex(Contacts.ContactMethodsColumns.TYPE)); Address a = new Address(addr, type); addrList.add(a); } addrCur.close(); return (addrList); } public ArrayList < IM > getIM(String id) { ArrayList < IM > imList = new ArrayList < IM > (); String where = Contacts.ContactMethods.PERSON_ID + " = ? AND " + Contacts.ContactMethods.KIND + " = ? " ; String[] whereParameters = new String[]{id, Contacts.ContactMethods.CONTENT_IM_ITEM_TYPE}; Cursor imCur = this .cr.query(Contacts.ContactMethods.CONTENT_URI, null , where, whereParameters, null ); if (imCur.moveToFirst()) { String imName = imCur.getString(imCur.getColumnIndex(Contacts.ContactMethodsColumns.DATA)); String imType = imCur.getString(imCur.getColumnIndex(Contacts.ContactMethodsColumns.TYPE)); if (imName.length() > 0 ) { IM im = new IM(imName, imType); imList.add(im); } } imCur.close(); return (imList); } public Organization getContactOrg(String id) { Organization org = new Organization(); String where = Contacts.ContactMethods.PERSON_ID + " = ? " ; String[] whereParameters = new String[]{id}; Cursor orgCur = this .cr.query(Contacts.Organizations.CONTENT_URI, null , where, whereParameters, null ); if (orgCur.moveToFirst()) { String orgName = orgCur.getString(orgCur.getColumnIndex(Contacts.Organizations.COMPANY)); String title = orgCur.getString(orgCur.getColumnIndex(Contacts.Organizations.TITLE)); if (orgName.length() > 0 ) { org.setOrganization(orgName); org.setTitle(title); } } orgCur.close(); return (org); } }

至此,Android Contacts的使用三大部分已经介绍完成,不得不承认的是有很多地方翻译的很烂很勉强,自己看着都有点头晕。嘿嘿,建议大家多看代码。

作者博客:http://www.cnblogs.com/lycoris/archive/2011/05/13/2037716.html
 
技术交流群:179914858
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/126946.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • VLC搭建RTSP直播流,图文介绍

    VLC搭建RTSP直播流,图文介绍将一个视频转成rtsp流,通过vlc播放器,搭建一个rtsp服务器,让rtsp客户端去访问这个视频的rtsp流1需要有vlc播放器,我的版本如下2媒体–>流3添加视频文件,点击添加一个mp4文件4选择串流,然后点击”下一个”5选择新目标,RTSP,然后点击添加6端口默认,路径添加个自定义名……

    2022年10月20日
  • linux 常用命令

    linux 常用命令

    2021年10月10日
  • pstack命令_压缩命令 linux

    pstack命令_压缩命令 linuxpstack命令可显示每个进程的栈跟踪。pstack命令必须由相应进程的属主或root运行。可以使用pstack来确定进程挂起的位置。此命令允许使用的唯一选项是要检查的进程的PID。pstree以树结构显示进程pstree-proot|grepphp-fpmroot为工作用户,-p为显示进程识别码,ps-Lf父进程号pstackPID号 转载…

  • mycat实现读写分离_mycat主从复制

    mycat实现读写分离_mycat主从复制1,课程回顾2,本章重点mysql主从原理,好处mycat概念,读写分离好处,读写分离的实现3,具体内容3.1mysql主从3.1.1linux下mysql安装以mysql5.6为例1)修改IP地址,修改主机名称vim/etc/sysconfig/network-scripts/ifcfg-ens33vim/etc/hostname2)安装mysql查看已经安装mysql组件:(centos6.9需要卸载原来的mysql

    2022年10月13日
  • Java基准测试工具JMH使用

    Java基准测试工具JMH使用JMH,即JavaMicrobenchmarkHarness,这是专门用于进行代码的微基准测试的一套工具API。JMH由OpenJDK/Oracle里面那群开发了Java编译器的大牛们所开发。何谓MicroBenchmark呢?简单地说就是在方法层面上的benchmark,精度可以精确到微秒级。本文主要介绍了性能基准测试工具JMH,它可以通过一些功能来规避由JVM中的JIT或者其他优化对性能测试造成的影响。

  • Linux System Programming note 8 ——File and Directory Management

    Linux System Programming note 8 ——File and Directory Management

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号