December
17th,
2014
前言: 最近開始仔細瞭解HealthKit,發現使用上並沒有那麼的直覺.(可能是我對於許多醫療單位沒有那麼清楚的關係 XD) 使用上有一些要注意的與需要瞭解的部分,稍微做一些紀錄. 筆記: [基本概念] 到目前為止,HealthKit只有iPhone可以用,並且要有iOS8.所以iPad執行程式會失敗. 每一個資料有所屬的資料格式數量(HKQuantity)與該數量的格式(HKQuantityType),而數量(Quantity)本身與單位的格式(HKUnit)有關. 必須要清楚地瞭解,存取的資料數量單位與單位的格式. 比如說,身高要確認是用cm還是inch,體重是使用pound還是kg. 每一個存取都與使用者權限有關,去拿使用者不開放的資料,就會引發錯誤. 所有的動作都是Async的方式: 這邊比較麻煩的是,如果需要抽象化的話可能要把async改成sync這邊可以參考這邊的內容. [基本資料欄位] 關於資料欄位舉例而言,要敘述關於血壓的舒張壓,你就得要撰寫以下的部分: //敘述你的血壓單位是 mmag mullumeter of mercyry unit HKUnit *BPUnit = [HKUnit millimeterOfMercuryUnit]; //透過這個HKUnit去建立單位 HKQuantity HKQuantity *BPSystolicQuantity = [HKQuantity quantityWithUnit:BPUnit doubleValue:SIS]; // 透過建立的單位與數值來建立這個單位類型(HKQuantityType) HKQuantityType *BPSystolicType = [HKQuantityType quantityTypeForIdentifier:HKQuantityTypeIdentifierBloodPressureSystolic]; [取得使用者權限] 上面的方式僅僅是拿來作為處理資料欄位而已,接下來要開始去跟HealthKit查詢資料.在查詢資料之前,由於HealthKit上的資料都是儲存在手機上面,屬於個人私密資料.需要取得使用者的權限. //取出體重單位格式 HKQuantityType *weightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierBodyMass]; //取出身高單位格式 HKQuantityType *heightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierHeight]; // 將身高體重加入要求的權限集合,這裡有分"讀取"與"寫入"的集合 NSSet *writeDataTypes = [NSSet setWithObjects:heightType, weightType, nil]; NSSet *readDataTypes = [NSSet setWithObjects:heightType, weightType, nil]; // 確認HealthKit 存不存在,這裡如果還是iPad 是不會有的,請注意. if ([HKHealthStore isHealthDataAvailable]) { // 要求使用者同意 [self.healthStore requestAuthorizationToShareTypes:writeDataTypes readTypes:readDataTypes completion:^(BOOL success, NSError *error) { if (!success) { NSLog(@"You didn't allow HealthKit to access these read/write data types. In your app, try to handle this error gracefully when a user decides not to provide access. The error was: %@. If you're using a simulator, try it on a device.", error); return; } //處理成功以後,可能需要的資料讀取部分. }]; } [讀取資料] 關於讀取資料這邊,需要注意的就是必須要取得正確的單位. 還是一樣,讀到資料以後是async,所以可以用另外的thread來更新UI 以下提供取得”心跳(Heart Rate)”的方式 //心跳屬於複合資料欄位,需要用字串來找到HKUnit.文件不好找到. HKUnit *dataUnit = [HKUnit unitFromString:@"count/min"];...
繼續閱讀
December
12th,
2014
前言:
本週都在Intel上課,大部份內容屬於NDA.只能找空閒時間看看其他的部分.
筆記:
[Mac OSX]一些常用的工具
想要尋找Mac上面類似pietty一樣好用的軟體,卻忘記terminal 本身就有一樣的功能.更何況我有裝iTerm2
[指令] ssh -l username -p portnumber host
參考:
25個SSH指令
除了想要尋找類似pietty一樣的軟體外,也想要尋找可以一次完成SSH跟SFTP的軟體
Fugu 就是我想找的.
[Android][iOS] PhoneGap Beta Release
這個東西可以幫助你開發Web App並且可以在手機(模擬器)上面瀏覽.
http://phonegap.com/blog/2014/12/11/phonegap-desktop-app-beta/
[Android] Android Studio 1.0 official release
剛剛痛苦的把Android Project轉換成Android 1.0RC2之後,official release出來了.找時間好好來看看如何使用.
https://developer.android.com/sdk/index.html
[iOS] Parse支援Crash Log,可以考慮要不要加進去自己的App.不過我App用的是Parse 1.2.19現在已經到1.6了.
相關介紹網址 http://goo.gl/70jIEb
這一篇有介紹一些toolkit可以使用
http://www.raywenderlich.com/33669/overview-of-ios-crash-reporting-tools-part-1
TestFlight也是很多人推薦 https://www.testflightapp.com/
當然挺多人推薦的HockeyApp http://hockeyapp.net/features/
[IaaS][PaaS] 在Azure上面架設VM(Virtual Machine)很方便.但是對外的port要透過設定不然預設都是關閉的.這件事情讓我很困擾.
問題:
建立了伺服器與服務,但是對外都無法找到相對應的服務
內部自己用curl或是其他client app可以找到相對應的服務
解決方式:
在Endpoint得加入你要開放給外面的port
甚至可以做一個簡單的port 修改,比如說把xmpp 從5222改到80,裡面的80 Apache改到8080
此外,現在Azure是有Preview的Manager Dashboard,東西有變多了.
繼續閱讀
December
5th,
2014
第一次打給我驚險在最後八秒過了。
可以關柵欄覺得比較容易。
極限很機車、雙劍刷腳很快倒。如果有極限強化可以更快
繼續閱讀
December
5th,
2014
前言:
本週工作依舊五花八門,下週要去Intel上課.要看看能寫多少出來.
筆記:
[MSFT] 關於Visual Studio 2013 upate 4更新過後,整個編輯器變慢.
新增的功能會預設把git當作他的source control plugin這樣會有一些問題.
問題:
更新VS2013 update4 之後,編輯器變慢.尤其是開啟檔案與關閉檔案的時候.
原因:
VS2013 source control plugin 會讀取該目錄git的資料,並且加入自已的source control plugin.
如果你已經有用到其他的source control system,可能會造成衝突而變慢.
解決方法:
也不算正式解決方法,只是先關閉這個plugin.可以幫助整體效能的提升.
Option -> Source Control -> Plug-in Selection -> Current Source control plugin -> None.
參考:
http://stackoverflow.com/questions/20101211/disable-git-in-visual-studio-2013
[iOS] 關於這週的一些小修改
有些App需要在背景的時候強制重啟,以確保連線狀態(尤其是xmpp).對於類似的App設計,可能可以考慮強制不要App進入背景模式.
原因:
這邊紀錄一下,主要是xmpp server通常會主動踢人(1 min),所以通常一般client 的實現都會有keepalive的機制.不過經過測試App方面的keepalive往往不盡人意.除非有傳輸的需求,不然建議不要進入背景模式比較好.
方法:
新增UIApplicationExitsOnSuspend = True在Plist裡面.
參考: http://stackoverflow.com/questions/5999638/quit-app-instead-of-entering-background-when-receiving-applicationdidenterbackgr
關於dispatch_async(dispatch_get_main_queue() 大家都會用它來把工作移到別的thread
但是其實他也是有照順序的,所以如果你依照順序跑三次,他就會依序跑那些你指定的指令.
用法上對於需要等待回應的程式,其實相當的方便.
關於NSUserDefaults的預設值總是會錯~如果第一次安裝App
錯誤狀況:
第一次安裝App後,讀取NSUserDefaults會無法讀到預設為True的值()或是其他預設值)
解決方式:
原因是因為你沒有把plist的數值去註冊,這邊有兩種方式可以選擇.
第一種: 讀取數值不要用boolValue 而用NSNumber接起來~然後判斷是不是nil
第二種: 執行NSUserDefaults registerDefaults把預設值先讀進App
參考資料:
http://oleb.net/blog/2014/02/nsuserdefaults-handling-default-values/
XCode更新了6.1.1:
很希望他有把IDE穩定性修好,不過有看到一個特別的詞 (SourceKit),去查了一下原來SourceKit是用來給swift作為Source code parser,Syntax highlighter 還有cross-language header generator…
這裏有詳細的解釋: http://www.jpsim.com/uncovering-sourcekit/
[Android] Android Studio 1.0RC版本出來,有些地方要注意:
Android安裝方式,可以選擇Standard來安裝.不過得要啟動時候沒有專案才會出現這個選項.
由於Gradle設定檔有問題,需要把runProguard改成minifyEnabled
參考這篇臉書: https://www.facebook.com/photo.php?fbid=1002023656481430&set=a.113952728621865.18504.100000214145898&type=1
參考這篇問答: http://stackoverflow.com/questions/27078075/gradle-dsl-method-not-found-runproguard
使用舊的Android Studio 含0.8.14版以前,所建立出來的專案,會自動產生res/values-v21/styles.xml造成在Android 5.0的裝置上會因沒有”android:Theme.Material.Light”而Crash掉.
參考這篇臉書: https://www.facebook.com/photo.php?fbid=1002023656481430&set=a.113952728621865.18504.100000214145898&type=1
[Linux][Ubuntu][Python] 手工打造process watcher and recovery
遇到的問題:
工作上遇到的問題主要是有管一些雲端服務,自然要監測這幾個service的狀況.如果遇到service出錯必須要有自動的回覆機制.
解決方式:
原本是透過Python寫一個透過grep來找特定的service的方式,這裡有程式碼.
不過研究過後,或許採取pidof的方式是不容易重複的方式.也改寫成shell script. 參考這裡.
這篇討論很值得思考,偵測process要考慮會不會產生額外的資源消耗.
繼續閱讀
November
28th,
2014
好像變成週記?不過~因為只是草草的紀錄一下,工作上有關的問題或是看到的網路介紹,並沒有打算寫成詳細的文章.
[綜合] 關於 Regular Expression
http://jex.im/regulex/ 是一個透過Java Script 可以視覺化呈現Regular Exoression 的網頁.
好用的Regular Expression範例(不論是取得信用卡號碼,電話號碼,生日,區碼甚至是Email 與 IP Address都有) http://www.virtuosimedia.com/dev/php/37-tested-php-perl-and-javascript-regular-expressions
個人認為兩個網站搭配起來,可以讓你的Regex的功力更上一層樓.
其他資源:
Eclipse 的Regex Tester介紹 http://www.ewdna.com/2009/12/eclipse-regular-expression-tester.html
Regex 精華整理 http://www.javaworld.com.tw/jute/post/view?bid=20&id=130126&sty=1&tpg=1&age=0
[iOS] 關於XCode6的Adaptive Layouts
到了Xcode6為了配合iPhone6(或是其他size iPhone),在XCode6裡面新增了Adaptive Layout的功能.仔細看了功能,發現真的是相當容易搞死人的設計.
相關資料:
One Storyboard for all sizes http://www.learnswift.io/blog/2014/6/12/size-classes-with-xcode-6-and-swift
Size Class tutorials http://www.ioscreator.com/tutorials/size-classes-ios8-xcode-6
Video tutorial https://www.youtube.com/watch?v=E3glNbNnokw
[Android] 關於InetAddress的 thread safe 問題
問題:
當手機在做Wifi 切換的時候,如果網路不穩定的狀況下,同時使用InetAddress.getByName()會hang住.
詳細狀況為第一次呼叫到會成功,第二次會hang在裡面.(同時呼叫)
解法:
沒有找到確切的證據,不過由於InetAddress本身是透過JNI來查詢網路,不適合作為網路狀況的確認.
網路上比較建議的方式必須要有UI來負責這件事情,因為比需要使用到permission “android.permission.ACCESS_NETWORK_STATE”
參考:
http://stackoverflow.com/questions/1560788/how-to-check-internet-access-on-android-inetaddress-never-timeouts
http://thiranjith.com/2011/03/31/how-to-monitor-network-connectivity-in-android/
// 當兩個thread同時呼叫以下的指令,在網路不穩的狀況下會出問題.
InetAddress.getByName("SOME_DOMAIN_NAME").isReachable(30);
// 比較建議的方式
public boolean isOnline() {
ConnectivityManager cm =
(ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo netInfo = cm.getActiveNetworkInfo();
return netInfo != null && netInfo.isConnectedOrConnecting();
}
// 必須要注意到加到 Manifest
< uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
繼續閱讀
November
23rd,
2014
前言: 最後一個作業是要做出一個自拍管理的App透過Content Provider來管理資料,還附有通知.從無到有其實很有趣的.我相信也能學到更多的東西,我希望能夠全部寫完,加油了. 筆記: 關於使用App Action Bar會強制請你加入Android-support-V7的project. 如果你是要讓Android V7~V10的App支援Action Bar,你必須要新增Android-support-V7的project並且新增相依專案.但是如果希望新增一個然後不要有這些的相依性問題的話,你需要利用以下設定建立.(同時符合大部分作業的需求在V18上) Minimal Requirement V17 Target V19 Compitible with V18 參考文章: http://oldgrayduck.blogspot.tw/2013/10/android-support-library-actionbar.html 如果要找default camera 在action menu的icon 其實Android SDK裡面就有,在ANDROID_SDK/platforms/android-XX/data/res/drawable-XXX 參考: http://stackoverflow.com/questions/8179653/where-can-i-find-standards-icons-for-actionbar 裡面主要透過Content Provider 的方式來達到資料的存取,並且增加了Notification的功能. Alarm Manager 透過他呼叫起一個Intent然後來設定Notification.透過這個方式來達到,離開App之後,可以在固定時間後收到通知. 關於讀取camera所拍照回來的照片檔案位址: Android無法直接取得相簿裡面的位址,比較容易的方法是在透過intent設定camera 的時候先設定好相關的參數. 光是那個參數設定了好久才能正常把相片另存地方.getBaseContext().getFilesDir()這個位址會讓Intent回不來. 參考http://stackoverflow.com/questions/15009581/camera-intent-picture-file-path //指定時間作為檔名來另存新檔 PackageManager pm = getPackageManager(); if (pm.hasSystemFeature(PackageManager.FEATURE_CAMERA)) { Intent intent_camera = new Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMdd_HHmmss"); String currentTimeStamp = dateFormat.format(new Date()); File mFile = new File(Environment.getExternalStorageDirectory(), currentTimeStamp+".jpg"); Uri outputFileUri = Uri.fromFile(mFile); intent_camera.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, outputFileUri); Toast.makeText(getBaseContext(), "File URI is "+outputFileUri, Toast.LENGTH_LONG).show(); startActivityForResult(intent_camera, 0); } else { Toast.makeText(getBaseContext(), "Camera is not available", Toast.LENGTH_LONG).show(); } 讀取縮圖並且直接設定到ListView裡面: 關於這部分的我把它想得太複雜了,網路上不少作法推薦直接把檔案放進content provider裡面,但是這樣沒有辦法存取多個資料. 最後其實就直接記住檔名(這裡檔名使用當下的時間,最後把圖片放進去BitmapDrawable就可以. // 從content provide拿取檔案名稱 String photoContentUri = cursor.getString(cursor.getColumnIndex(DataContract.PIC_FILE_NAME)); // 與系統位置配對成完整檔案位置 File mFile = new File(Environment.getExternalStorageDirectory(), photoContentUri); if (null != photoContentUri) { InputStream input = null; try { // 讀取縮圖 input = context.getContentResolver().openInputStream(Uri.fromFile(mFile)); if (input != null) { photoBitmap = new BitmapDrawable(mApplicationContext.getResources(), input); photoBitmap.setBounds(0, 0, mBitmapSize, mBitmapSize); } } catch (FileNotFoundException e) { Log.e(TAG, "FileNotFoundException"); }...
繼續閱讀