鍍金池/ 教程/ Android/ 創(chuàng)建和監(jiān)視地理圍欄
檢測常用的手勢
優(yōu)化layout的層級
用戶輸入
管理應(yīng)用的內(nèi)存
聯(lián)系人信息
開發(fā)輔助程序
Android多媒體
添加語音功能
顯示位置地址
提供向下與橫向?qū)Ш?/span>
支持游戲控制器
訪問可穿戴數(shù)據(jù)層
處理多點觸控手勢
全屏沉浸式應(yīng)用
為多線程創(chuàng)建管理器
數(shù)據(jù)保存
Intent的發(fā)送
更新Notification
優(yōu)化下載以高效地訪問網(wǎng)絡(luò)
打印
打包可穿戴應(yīng)用
接收從其他App傳送來的數(shù)據(jù)
發(fā)送與接收消息
建立靈活動態(tài)的UI
處理鍵盤輸入
Building a Work Policy Controller
建立測試環(huán)境
創(chuàng)建表盤
分享文件
顯示Notification進度
實現(xiàn)自適應(yīng)UI流(Flows)
使用設(shè)備管理策略增強安全性
使用能感知版本的組件
執(zhí)行網(wǎng)絡(luò)操作
建立文件分享
添加移動
更新你的Security Provider來對抗SSL漏洞利用
支持鍵盤導(dǎo)航
創(chuàng)建和監(jiān)視地理圍欄
發(fā)送并同步數(shù)據(jù)
使用BigView樣式
無線連接設(shè)備
提供向上導(dǎo)航與歷史導(dǎo)航
最小化定期更新造成的影響
實現(xiàn)向下的導(dǎo)航
支持不同的屏幕大小
Android 可穿戴應(yīng)用
添加動畫
顯示聯(lián)系人頭像
使用OpenGL ES顯示圖像
處理輸入法可見性
分享文件
保持設(shè)備喚醒
淡化系統(tǒng)Bar
使用NFC分享文件
保存到Preference
Android聯(lián)系人信息與位置信息
創(chuàng)建標準的網(wǎng)絡(luò)請求
使用Drawables
管理Bitmap的內(nèi)存使用
管理Activity的生命周期
按需加載視圖
傳輸資源
為可穿戴設(shè)備創(chuàng)建自定義UI
在一個線程中執(zhí)行一段特定的代碼
性能優(yōu)化
隱藏導(dǎo)航欄
創(chuàng)建目錄瀏覽器
為多種大小的屏幕進行規(guī)劃
View間漸變
使用觸摸手勢
高效加載大圖
使用CursorLoader在后臺加載數(shù)據(jù)
創(chuàng)建抽屜式導(dǎo)航(navigation drawer)
管理音頻焦點
創(chuàng)建后臺服務(wù)
創(chuàng)建功能測試
創(chuàng)建使用Material Design的應(yīng)用
停止與重啟Activity
添加一個簡便的分享功能
啟動Activity時保留導(dǎo)航
TV應(yīng)用清單
創(chuàng)建向后兼容的UI
?# 優(yōu)化自定義View
創(chuàng)建單元測試
在UI上顯示Bitmap
建立OpenGL ES的環(huán)境
構(gòu)建表盤服務(wù)
JNI Tips
建立搜索界面
實現(xiàn)自定義View的繪制
使用HTTPS與SSL
按需操控BroadcastReceiver
分享簡單的數(shù)據(jù)
繪制形狀
Android位置信息
創(chuàng)建并運行可穿戴應(yīng)用
執(zhí)行 Sync Adpater
獲取最后可知位置
創(chuàng)建 Android 項目
實現(xiàn)高效的導(dǎo)航
退出全屏的Activity
創(chuàng)建Card
兼容音頻輸出設(shè)備
同步數(shù)據(jù)單元
傳輸數(shù)據(jù)時避免消耗大量電量
保存到文件
緩存Bitmap
提供配置 Activity
調(diào)度重復(fù)的鬧鐘
實現(xiàn)輔助功能
重復(fù)的下載是冗余的
隱藏狀態(tài)欄
實現(xiàn)自定義的網(wǎng)絡(luò)請求
規(guī)劃界面和他們之間的關(guān)系
使用Sync Adapter傳輸數(shù)據(jù)
TV應(yīng)用內(nèi)搜索
響應(yīng)觸摸事件
使用Google Cloud Messaging(已廢棄)
控制相機
Android網(wǎng)絡(luò)連接與云服務(wù)
請求分享一個文件
處理TV硬件
響應(yīng)UI可見性的變化
使用網(wǎng)絡(luò)服務(wù)發(fā)現(xiàn)
指定輸入法類型
優(yōu)化電池壽命
創(chuàng)建TV應(yīng)用
獲取聯(lián)系人列表
拖拽與縮放
啟動與停止線程池中的線程
創(chuàng)建 Sync Adpater
使用 WiFi P2P 服務(wù)發(fā)現(xiàn)
開始使用Material Design
代理至新的APIs
使用include標簽重用layouts
使得View可交互
高效顯示Bitmap
創(chuàng)建企業(yè)級應(yīng)用
Fragments之間的交互
創(chuàng)建與執(zhí)行測試用例
綜合:設(shè)計我們的樣例 App
繪制表盤
建立簡單的用戶界面
自定義動畫
開發(fā)輔助服務(wù)
避免出現(xiàn)程序無響應(yīng)ANR(Keeping Your App Responsive)
使用ViewPager實現(xiàn)屏幕滑動
設(shè)計高效的導(dǎo)航
Android分享操作(Building Apps with Content Sharing)
提供向后的導(dǎo)航
保持向下兼容
創(chuàng)建TV播放應(yīng)用
縮放View
使用 WiFi 建立 P2P 連接
Android后臺任務(wù)
連接到網(wǎng)絡(luò)
為 Notification 添加頁面
使TV應(yīng)用是可被搜索的
添加Action Bar
使用Material的主題
啟動另一個Activity
顯示正在播放卡片
適配不同的系統(tǒng)版本
輕松錄制視頻
創(chuàng)建可穿戴的應(yīng)用
創(chuàng)建自定義的布局
重新創(chuàng)建Activity
使用CursorLoader執(zhí)行查詢?nèi)蝿?wù)
使用舊的APIs實現(xiàn)新API的效果
使用備份API
安全要點
Android入門基礎(chǔ):從這里開始
保存并搜索數(shù)據(jù)
根據(jù)網(wǎng)絡(luò)連接類型來調(diào)整下載模式
使用Tabs創(chuàng)建Swipe視圖
SMP(Symmetric Multi-Processor) Primer for Android
解析 XML 數(shù)據(jù)
使用 Volley 傳輸網(wǎng)絡(luò)數(shù)據(jù)
建立ActionBar
Android交互設(shè)計
使用Intent修改聯(lián)系人信息
增加搜索功能
輕松拍攝照片
定義形狀
測試你的Activity
在 Notifcation 中接收語音輸入
與其他應(yīng)用的交互
管理系統(tǒng)UI
追蹤手勢移動
Android界面設(shè)計
執(zhí)行 Android 程序
顯示確認界面
創(chuàng)建Lists與Cards
打印HTML文檔
創(chuàng)建TV應(yīng)用
為多屏幕設(shè)計
定義Shadows與Clipping視圖
使用Fragment建立動態(tài)UI
接收Activity返回的結(jié)果
布局變更動畫
定位常見的問題
自定義ActionBar的風格
定義Layouts
發(fā)送簡單的網(wǎng)絡(luò)請求
啟動與銷毀Activity
與UI線程通信
非UI線程處理Bitmap
創(chuàng)建TV布局
提升Layout的性能
報告任務(wù)執(zhí)行狀態(tài)
判斷并監(jiān)測網(wǎng)絡(luò)連接狀態(tài)
兼容不同的設(shè)備
處理按鍵動作
優(yōu)化性能和電池使用時間
給其他App發(fā)送簡單的數(shù)據(jù)
Implementing App Restrictions
向后臺服務(wù)發(fā)送任務(wù)請求
展示Card翻轉(zhuǎn)動畫
管理ViewGroup中的觸摸事件
兼容不同的屏幕密度
通過藍牙進行調(diào)試
為可穿戴設(shè)備創(chuàng)建Notification
控制音量與音頻播放
獲取聯(lián)系人詳情
在表盤上顯示信息
提供向上的導(dǎo)航
滾動手勢動畫
幫助用戶在TV上找到內(nèi)容
創(chuàng)建TV導(dǎo)航
為索引指定App內(nèi)容
ActionBar的覆蓋疊加
Android Wear 上的位置檢測
保護安全與隱私的最佳策略
Ensuring Compatibility with Managed Profiles
解決云同步的保存沖突
獲取位置更新
創(chuàng)建List
測試程序
管理網(wǎng)絡(luò)的使用情況
為App內(nèi)容開啟深度鏈接
推薦TV內(nèi)容
建立一個Notification
管理音頻播放
設(shè)計表盤
拍照
處理控制器輸入動作
判斷并監(jiān)測設(shè)備的底座狀態(tài)與類型
處理查詢的結(jié)果
保存到數(shù)據(jù)庫
支持多個游戲控制器
創(chuàng)建 Stub Content Provider
使得ListView滑動順暢
處理數(shù)據(jù)層的事件
創(chuàng)建TV應(yīng)用的第一步
使得你的App內(nèi)容可被Google搜索
將 Notification 放成一疊
創(chuàng)建 Stub 授權(quán)器
暫停與恢復(fù)Activity
管理設(shè)備的喚醒狀態(tài)
Android圖像與動畫
打印照片
云同步
創(chuàng)建TV直播應(yīng)用
為Notification賦加可穿戴特性
提供一個Card視圖
建立請求隊列(RequestQueue)
適配不同的語言
創(chuàng)建詳情頁
測試UI組件
接收其他設(shè)備的文件
創(chuàng)建自定義View
建立第一個App
創(chuàng)建2D Picker
監(jiān)測電池的電量與充電狀態(tài)
打印自定義文檔
抽象出新的APIs
通知提示用戶
獲取文件信息
運用投影與相機視角
在IntentService中執(zhí)行后臺任務(wù)
多線程操作
創(chuàng)建一個Fragment
添加Action按鈕
在不同的 Android 系統(tǒng)版本支持控制器
維護兼容性
發(fā)送文件給其他設(shè)備
創(chuàng)建TV游戲應(yīng)用
創(chuàng)建自定義的View類
代碼性能優(yōu)化建議
Intent過濾
適配不同的屏幕

創(chuàng)建和監(jiān)視地理圍欄

編寫:penkzhou - 原文:http://developer.android.com/training/location/geofencing.html

地理圍欄將用戶當前位置感知和附件地點特征感知相結(jié)合。為了標示一個感興趣的位置,我們需要指定這個位置的經(jīng)緯度。為了調(diào)整位置的鄰近度,需要添加一個半徑。經(jīng)緯度和半徑定義一個地理圍欄,即在感興趣的位置創(chuàng)建一個圓形區(qū)域或者圍欄。

我們可以有多個活動的地理圍欄(限制是一個設(shè)備用戶100個)。對于每個地理圍欄,我們可以讓 Location Services 發(fā)出進入和離開事件,或者我們可以在觸發(fā)一個事件之前,指定在某個地理圍欄區(qū)域等待一段時間或者停留。通過指定一個以毫秒為單位的截止時間,我們可以限制任何一個地理圍欄的持續(xù)時間。當?shù)乩韲鷻谑Ш螅琇ocation Services 會自動刪除這個地理圍欄。

http://wiki.jikexueyuan.com/project/android-training-geek/images/geofence.png" alt="geofence" />

這節(jié)課介紹如何添加和刪除地理圍欄,和用 IntentService 監(jiān)聽地理位置變化。

設(shè)置地理圍欄監(jiān)視

請求地理圍欄監(jiān)視的第一步就是設(shè)置必要的權(quán)限。在使用地理圍欄時,我們必須設(shè)置 ACCESS_FINE_LOCATION 權(quán)限。在應(yīng)用的 manifest 文件中添加如下子節(jié)點即可:

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

如果想要用 IntentService 監(jiān)聽地理位置變化,那么還需要添加一個節(jié)點來指定服務(wù)名字。這個節(jié)點必須是 的子節(jié)點:

<application
   android:allowBackup="true">
   ...
   <service android:name=".GeofenceTransitionsIntentService"/>
<application/>

為了訪問位置 API,我們需要創(chuàng)建一個 Google Play services API client 的實例。想要學習如何連接 client,請見連接Google Play Services。

創(chuàng)建和添加地理圍欄

我們的應(yīng)用需要用位置 API 的 builder 類來創(chuàng)建地理圍欄,用 convenience 類來添加地理圍欄。另外,我們可以定義一個 PendingIntent(將在這節(jié)課介紹)來處理當?shù)乩砦恢冒l(fā)生遷移時,Location Services 發(fā)出的 intent。

創(chuàng)建地理圍欄對象

首先,用 Geofence.Builder 創(chuàng)建一個地理圍欄,設(shè)置想要的半徑,持續(xù)時間,和地理圍欄遷移的類型。例如,填充一個叫做 mGeofenceList 的 list 對象:

mGeofenceList.add(new Geofence.Builder()
    // Set the request ID of the geofence. This is a string to identify this
    // geofence.
    .setRequestId(entry.getKey())

    .setCircularRegion(
            entry.getValue().latitude,
            entry.getValue().longitude,
            Constants.GEOFENCE_RADIUS_IN_METERS
    )
    .setExpirationDuration(Constants.GEOFENCE_EXPIRATION_IN_MILLISECONDS)
    .setTransitionTypes(Geofence.GEOFENCE_TRANSITION_ENTER |
            Geofence.GEOFENCE_TRANSITION_EXIT)
    .build());

這個例子從一個固定的文件中獲取數(shù)據(jù)。在實際情況下,應(yīng)用可能會根據(jù)用戶的位置動態(tài)地創(chuàng)建地理圍欄。

指定地理圍欄和初始化觸發(fā)器

下面的代碼用到 GeofencingRequest 類。該類嵌套了 GeofencingRequestBuilder 類來需要監(jiān)視的地理圍欄和設(shè)置如何觸發(fā)地理圍欄事件:

private GeofencingRequest getGeofencingRequest() {
    GeofencingRequest.Builder builder = new GeofencingRequest.Builder();
    builder.setInitialTrigger(GeofencingRequest.INITIAL_TRIGGER_ENTER);
    builder.addGeofences(mGeofenceList);
    return builder.build();
}

這個例子介紹了兩個地理圍欄觸發(fā)器。當設(shè)備進入一個地理圍欄時, GEOFENCE_TRANSITION_ENTER 轉(zhuǎn)移會觸發(fā)。當設(shè)備離開一個地理圍欄時, GEOFENCE_TRANSITION_EXIT 轉(zhuǎn)移會觸發(fā)。如果設(shè)備已經(jīng)在地理圍欄里面,那么指定 INITIAL_TRIGGER_ENTER 來通知位置服務(wù)觸發(fā) GEOFENCE_TRANSITION_ENTER。

在很多情況下,使用 INITIAL_TRIGGER_DWELL 可能會更好。僅僅當由于到達地理圍欄中已定義好的持續(xù)時間,而導(dǎo)致用戶停止時,INITIAL_TRIGGER_DWELL 才會觸發(fā)事件。這個方法可以減少當設(shè)備短暫地進入和離開地理圍欄時,由大量的通知造成的“垃圾警告信息”。另一種獲取最好的地理圍欄結(jié)果的策略是設(shè)置最小半徑為100米。這有助于估計典型的 Wifi 網(wǎng)絡(luò)的位置精確度,也有利于降低設(shè)備的功耗。

為地理圍欄轉(zhuǎn)移定義Intent

從 Location Services 發(fā)送來的Intent能夠觸發(fā)各種應(yīng)用內(nèi)的動作,但是不能用它來打開一個 Activity 或者 Fragment,這是因為應(yīng)用內(nèi)的組件只能在響應(yīng)用戶動作時才可見。大多數(shù)情況下,處理這一類 Intent 最好使用 IntentService。一個 IntentService 可以推送一個通知,可以進行長時間的后臺作業(yè),可以將 intent 發(fā)送給其他的 services ,還可以發(fā)送一個廣播 intent。下面的代碼展示了如何定義一個 PendingIntent 來啟動一個 IntentService:

public class MainActivity extends FragmentActivity {
    ...
    private PendingIntent getGeofencePendingIntent() {
        // Reuse the PendingIntent if we already have it.
        if (mGeofencePendingIntent != null) {
            return mGeofencePendingIntent;
        }
        Intent intent = new Intent(this, GeofenceTransitionsIntentService.class);
        // We use FLAG_UPDATE_CURRENT so that we get the same pending intent back when
        // calling addGeofences() and removeGeofences().
        return PendingIntent.getService(this, 0, intent, PendingIntent.
                FLAG_UPDATE_CURRENT);
    }

添加地理圍欄

使用 GeoencingApi.addGeofences() 方法來添加地理圍欄。為該方法提供 Google API client,GeofencingRequest 對象和 PendingIntent。下面的代碼,在 onResult() 中處理結(jié)果,假設(shè)主 activity 實現(xiàn) ResultCallback。

public class MainActivity extends FragmentActivity {
    ...
    LocationServices.GeofencingApi.addGeofences(
                mGoogleApiClient,
                getGeofencingRequest(),
                getGeofencePendingIntent()
        ).setResultCallback(this);

處理地理圍欄轉(zhuǎn)移

當 Location Services 探測到用戶進入或者離開一個地理圍欄,它會發(fā)送一個包含在 PendingIntent 的 Intent,這個 PendingIntent 就是在添加地理圍欄時被我們包括在請求當中。這個 Intent 被一個類似 GeofenceTransitionsIntentService 的 service 接收,這個 service 從 intent 得到地理圍欄事件,決定地理圍欄轉(zhuǎn)移的類型,和決定觸發(fā)哪個已定義的地理圍欄。然后它會發(fā)出一個通知。

下面的代碼介紹了如何定義一個 IntentService。這個 IntentService 在地理圍欄轉(zhuǎn)移出現(xiàn)時,會推送一個通知。當用戶點擊這個通知,那么應(yīng)用的主 activity 會出現(xiàn):

public class GeofenceTransitionsIntentService extends IntentService {
   ...
    protected void onHandleIntent(Intent intent) {
        GeofencingEvent geofencingEvent = GeofencingEvent.fromIntent(intent);
        if (geofencingEvent.hasError()) {
            String errorMessage = GeofenceErrorMessages.getErrorString(this,
                    geofencingEvent.getErrorCode());
            Log.e(TAG, errorMessage);
            return;
        }

        // Get the transition type.
        int geofenceTransition = geofencingEvent.getGeofenceTransition();

        // Test that the reported transition was of interest.
        if (geofenceTransition == Geofence.GEOFENCE_TRANSITION_ENTER ||
                geofenceTransition == Geofence.GEOFENCE_TRANSITION_EXIT) {

            // Get the geofences that were triggered. A single event can trigger
            // multiple geofences.
            List triggeringGeofences = geofencingEvent.getTriggeringGeofences();

            // Get the transition details as a String.
            String geofenceTransitionDetails = getGeofenceTransitionDetails(
                    this,
                    geofenceTransition,
                    triggeringGeofences
            );

            // Send notification and log the transition details.
            sendNotification(geofenceTransitionDetails);
            Log.i(TAG, geofenceTransitionDetails);
        } else {
            // Log the error.
            Log.e(TAG, getString(R.string.geofence_transition_invalid_type,
                    geofenceTransition));
        }
    }

在通過 PendingIntent 檢測轉(zhuǎn)移事件之后,這個 IntentService 獲取地理圍欄轉(zhuǎn)移類型和測試一個事件是不是應(yīng)用用來觸發(fā)通知的 —— 要么是 GEOFENCE_TRANSITION_ENTER,要么是 GEOFENCE_TRANSITION_EXIT。然后,這個 service 會發(fā)出一個通知并且記錄轉(zhuǎn)移的詳細信息。

停止地理圍欄監(jiān)視

當不再需要監(jiān)視地理圍欄或者想要節(jié)省設(shè)備的電池電量和 CPU 周期時,需要停止地理圍欄監(jiān)視。我們可以在用于添加和刪除地理圍欄的主 activity 里停止地理圍欄監(jiān)視;刪除地理圍欄會導(dǎo)致它馬上停止。API 要么通過 request IDs,要么通過刪除與指定 PendingIntent 相關(guān)的地理圍欄來刪除地理圍欄。

下面的代碼通過 PendingIntent 刪除地理圍欄,當設(shè)備進入或者離開之前已經(jīng)添加的地理圍欄時,停止所有通知:

LocationServices.GeofencingApi.removeGeofences(
            mGoogleApiClient,
            // This is the same pending intent that was used in addGeofences().
            getGeofencePendingIntent()
    ).setResultCallback(this); // Result processed in onResult().
}

你可以將地理圍欄同其他位置感知的特性結(jié)合起來,比如周期性的位置更新。像要了解更多的信息,請看本章的其它課程。

上一篇:繪制形狀下一篇:打印