Android音频管理

第一步

创建 MediaPlayer 对象 mMediaPlayer
AudioManager 对象 audioManager

第二步:

创建 OnAudioFocusChangeListener 对象 onAudioFocusChangeListener
重写 onAudioFocusChange 方法
第三步:
获取系统音频焦点:

1
audioManager=(AudioManager)getActivity().getSystemService(Context.AUDIO_SERVICE);

Android解析json数据

简介

JSONObject和JSONArray 是Android解析JSON数据的2个重要对象。

操作一:(从String中获取一个JSONObject):

1
JSONObjectjsonObject=newJSONObject(jsonString);

操作二:(从JSONObject中读取一个JSONArray):

1
JSONArrayjsonArray=jsonObject.getJSONArray("features");

操作三:(从JSONArray中遍历JSONObject):

1
2
3
4
5
6
7
8
9
10
jsonObject=jsonArray.getJSONObject(i);
while(jsonObject!=null){
JSONObject jsonObject2=jsonObject.getJSONObject("properties");
String level1=jsonObject2.getString("mag");
String region=jsonObject2.getString("place");
String date=jsonObject2.getString("time");
String url=jsonObject2.getString("url");
i++;
jsonObject=jsonArray.getJSONObject(i);
}

操作四:(从JSONObject中读取属性):

1
String date=jsonObject2.getString("time");

Androi线程的2中方式

方式1 : 通过 AsyncTask 实现:

第一步:

编写内部类 AsyncTask
该类型有3个泛型参数 :

  • a: 参数类型(传递进去的)
  • b: published 的步进单位
  • c: 结果的返回类型

需要重写2个方法:

  • doInBackground 后台所做的操作
  • onPostExecute 每次执行的操作后进度通告
第二步:

在 mainActivity中创建该类,执行 execute 操作:

1
2
EarthquakeAsyncTask task = new EarthquakeAsyncTask();
task.execute(USGS_REQUEST_URL);

代码示例地址:
https://github.com/udacity/ud843_DidYouFeelIt/blob/solution/app/src/main/java/com/example/android/didyoufeelit/MainActivity.java
缺陷:在旋转屏幕时 activity会重建和销毁.该类也会被重建.大量消耗系统资源

方式2 : AsyncTaskLoader

第一步:

编写 AsyncTaskLoader 类 AsyncTaskLoader

泛型a为返回的数据类型

重写 loadInBackground (返回结果)和 构造函数

第二步:

在Activity 实现 LoaderManager.LoaderCallbacks>

实现其 onCreateLoader onLoadFinished onLoaderReset方法

举例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public Loader<List<EarthData>> onCreateLoader(int id, Bundle args) {
try {
return new myEarthQuakerLoader(this,new URL("https://earthquake.usgs.gov/fdsnws/event/1/query?format=geojson&starttime=2010-01-01&endtime=2014-12-01&minmagnitude=7"));
} catch (MalformedURLException e) {
e.printStackTrace();
}
return null;
}
@Override
public void onLoadFinished(Loader<List<EarthData>> loader, List<EarthData> data) {
View view = findViewById(R.id.loading_indicator);
view.setVisibility(view.GONE);
TextView textview = (TextView)findViewById(R.id.empty_view);
textview.setText("No Earthqueke found");
earthquakes.clear();
if (data != null && !data.isEmpty()) {
earthquakes.addAll(data);
adapter.notifyDataSetChanged();
}
}
@Override
public void onLoaderReset(Loader<List<EarthData>> loader) {
earthquakes.clear();
}

第三步:

在activity中通过 LoaderManagerloaderManager 来初始化加载

1
2
LoaderManagerloaderManager=getLoaderManager();
loaderManager.initLoader(1,null,this);

示例代码:
点我跳转

Android布局解析

关于Android布局,我用到的有4种:

  1. 线性布局:LinearLayout
  2. 相对布局:RelativeLayout
  3. 列表布局:ListView
  4. 框架视图:Fragment

基础布局属性:

属性 介绍
Width 该属性配置布局的宽度
Height 该属性配置布局的高度
注: 当取值为 match_parent 随父布局
取值为 wrap_content 内容大小自适应
或者可一直设置 50dp这种(dp是安卓布局很常见的尺寸单位)
Gravity 该属性配置内部子视图的布局,例如居中为:center

Android 使用CursorLoader加载数据

第一步:

在Activity类中实现 android.app.LoaderManager.LoaderCallbacks

第二步:

重写 onLoadFinished onCreateLoader 方法

第三步:

在activity 的 oncreate 方法加载一下loader:
getLoaderManager().initLoader(id,null,this);

第四步:

在 provider 中通知数据更新:

getContext().getContentResolver().notifyChange(uri,null);

参考代码:https://github.com/udacity/ud845-Pets/tree/d942c19b45cdce4e3d9afba8e2165c7eebc389fb/app/src/main/java/com/example/android/pets

Android SQLite的使用

在android中,大量的数据存放在本地中,所以管理本地数据用轻量级数据库SQLite非常适合.

第一步:

在使用数据之前,Android推荐我们使用Contact来管理和使用数据库.

具体使用方法如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public final class PetContract {
private PetContract() {}
public static final class PetEntry implements BaseColumns {
public final static String TABLE_NAME = "pets";
public final static String _ID = BaseColumns._ID;
public final static String COLUMN_PET_NAME ="name";
public final static String COLUMN_PET_BREED = "breed";
public final static String COLUMN_PET_GENDER = "gender";
public final static String COLUMN_PET_WEIGHT = "weight";
public static final int GENDER_UNKNOWN = 0;
public static final int GENDER_MALE = 1;
public static final int GENDER_FEMALE = 2;
}
}

Android 内容提供者 ContentProvider

ContentProvider 是android的四大组件之一,用来为程序提供数据(可用于不同程序之间)

第一步:

在此之前需要先用UriMatcher对Uri进行匹配处理:

1
2
3
4
5
6
7
private static finalintPETS=100;
private static finalintPET_ID=101;
private static UriMatcheruriMatcher=newUriMatcher(UriMatcher.NO_MATCH);
static{
uriMatcher.addURI("com.example.android.pets","pets",PETS);
uriMatcher.addURI("com.example.android.pets","pets/#",PET_ID);
}

然后再 ContentProvider 增删查改中对应匹配的方法进行操作:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Int rCode=uriMatcher.match(uri);
SQLiteDatabasedb=mPetDbHelper.getReadableDatabase();
Cursor cursor;
switch(rCode){
casePETS:
cursor=db.query(PetsContract.PetsEntry.TABLE_NAME,strings,s,strings1,null,null,null);
break;
casePET_ID:
Stringselection=PetsContract.PetsEntry._ID+"=?";
String[]selectionArgs=newString[]{String.valueOf(ContentUris.parseId(uri))};
cursor=db.query(PetsContract.PetsEntry.TABLE_NAME,strings,selection,selectionArgs,null,null,null);
break;
default:
Throw newIllegalArgumentException("Uriisinvalid.");
}

第二步:

在方法中调用数据库操作方法来操作数据库.
ContentUris.withAppendedId(uri,id) 可以将 uri 与 id 组合起来
第三步:
在查询的尾部要通知更新数据:
cursor.setNotificationUri(getContext().getContentResolver(),uri);
其他的地方用下面的语句来通知
getContext().getContentResolver().notifyChange(uri,null);
第四步:
编写其 XML 文件
代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
public class PetProvider extends ContentProvider {
PetDbHelper mPetDbHelper;
private static final int PETS = 100;
private static final int PET_ID = 101;
private static UriMatcher uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
static{
uriMatcher.addURI("com.example.android.pets","pets",PETS);
uriMatcher.addURI("com.example.android.pets","pets/#",PET_ID);
}
@Override
public boolean onCreate() {
mPetDbHelper = new PetDbHelper(getContext());
return true;
}
@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] strings, @Nullable String s, @Nullable String[] strings1, @Nullable String s1) {
int rCode = uriMatcher.match(uri);
SQLiteDatabase db = mPetDbHelper.getReadableDatabase();
Cursor cursor;
switch (rCode){
case PETS:
cursor = db.query(PetsContract.PetsEntry.TABLE_NAME,strings ,s,strings1,null,null,null);
break;
case PET_ID:
String selection = PetsContract.PetsEntry._ID + " = ? ";
String[] selectionArgs = new String [] {String.valueOf(ContentUris.parseId(uri))};
cursor = db.query(PetsContract.PetsEntry.TABLE_NAME,strings ,selection,selectionArgs,null,null,null);
break;
default:
throw new IllegalArgumentException("Uri is invalid.");
}
cursor.setNotificationUri(getContext().getContentResolver(),uri);
return cursor;
}
@Nullable
@Override
public String getType(@NonNull Uri uri) {
return null;
}

@Nullable
@Override
public Uri insert(Uri uri, ContentValues contentValues) {
final int match = uriMatcher.match(uri);
switch (match) {
case PETS:
return insertPet(uri, contentValues);
default:
throw new IllegalArgumentException("Insertion is not supported for " + uri);
}
}

private Uri insertPet(Uri uri, ContentValues values) {
SQLiteDatabase db = mPetDbHelper.getWritableDatabase();
long id = db.insert(PetsContract.PetsEntry.TABLE_NAME,null,values);
getContext().getContentResolver().notifyChange(uri,null);
return ContentUris.withAppendedId(uri, id);
}

@Override
public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {
SQLiteDatabase database = mPetDbHelper.getWritableDatabase();

final int match = uriMatcher.match(uri);
switch (match) {
case PETS:
getContext().getContentResolver().notifyChange(uri,null);
return database.delete(PetsContract.PetsEntry.TABLE_NAME, selection, selectionArgs);
case PET_ID:
selection = PetsContract.PetsEntry._ID + "=?";
selectionArgs = new String[] { String.valueOf(ContentUris.parseId(uri)) };
getContext().getContentResolver().notifyChange(uri,null);
return database.delete(PetsContract.PetsEntry.TABLE_NAME, selection, selectionArgs);
default:
throw new IllegalArgumentException("Deletion is not supported for " + uri);
}
}
private int updatePet(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
SQLiteDatabase db=mPetDbHelper.getWritableDatabase();
return db.update(PetsContract.PetsEntry.TABLE_NAME,values,selection,selectionArgs);
}
@Override
public int update(@NonNull Uri uri, @Nullable ContentValues contentValues, @Nullable String s, @Nullable String[] strings) {
final int match = uriMatcher.match(uri);
switch (match) {
case PETS:
getContext().getContentResolver().notifyChange(uri,null);
return updatePet(uri, contentValues, s, strings);
case PET_ID:
s = PetsContract.PetsEntry._ID + "=?";
strings = new String[] { String.valueOf(ContentUris.parseId(uri)) };
int i=updatePet(uri, contentValues, s, strings);
getContext().getContentResolver().notifyChange(uri,null);
return i;
default:
throw new IllegalArgumentException("Update is not supported for " + uri);
}
}
}

参考地址:
https://github.com/udacity/ud845-Pets/blob/a244b5a4cbda9c5fb7b551032b40166542da9869/app/src/main/java/com/example/android/pets/data/PetProvider.java

Android HTTP通信

第一步:

在配置的XML文件中加入获取权限的代码:

1
2
<uses-permissionandroid:name="android.permission.INTERNET"/>
<uses-permissionandroid:name="android.permission.ACCESS_NETWORK_STATE"/>

第二步:

创建URL 对象 url
创建 HttpURLConnection 对象 httpURLConnection

1
2
HttpURLConnection httpURLConnection = null;
httpURLConnection=(HttpURLConnection)url.openConnection();(判断是否为空)

再设置3个参数:

1
2
3
httpURLConnection.setRequestMethod("GET");//请求方式
httpURLConnection.setReadTimeout(10000);//读取时间上限
httpURLConnection.setConnectTimeout(15000);//连接时间上限