git diff-tree -r --no-commit-id --name-only --diff-filter=ACMRT COMMIT1 COMMIT2
將其中的 commit1 和 2 換成想比較的兩個 commit 即可。
如果只想複製出有更新的檔案
cp $(git diff-tree -r --no-commit-id --name-only --diff-filter=ACMRT COMMIT1 COMMIT2) PATH_TO_TARGET
2013年7月21日 星期日
Android Scan File Cause Global Reference Table Overflow (max=51200)
Android 4.x (版本不確定,已知版本 4.1.1、4.2.2),如果極端大量且快速的執行寫檔後馬上 scanFile 的動作,會造成 JNI ERROR (app bug): global reference table overflow (max=51200),詳細原因見官方 Issue:Issue 53914
如果遇到這種問題,就在最後再執行一次完整 scanFile 避掉,而非每存一個檔案就進行一次 scanFile 。
如果遇到這種問題,就在最後再執行一次完整 scanFile 避掉,而非每存一個檔案就進行一次 scanFile 。
2013年7月17日 星期三
Android Context.startActivity From Uri Must Catch ActivityNotFoundException
如果 Intent 是用 Uri.parse 出的,在 startActivity 時請一定要 try catch ActivityNotFoundException。
例如:大部分的 Android 裝置都會有 Google Play,但是有些可能是白牌手機之類的,這時候如果想要去呼叫 Google Play 就會出現 ActivityNotFoundException。
所以如果不想管這麼多,就是只要用 Uri 或甚至是 package name 去 startActivity 的都要處理,除非有先用其他方式檢查過要開到的 Activity 是否存在。
例如:大部分的 Android 裝置都會有 Google Play,但是有些可能是白牌手機之類的,這時候如果想要去呼叫 Google Play 就會出現 ActivityNotFoundException。
所以如果不想管這麼多,就是只要用 Uri 或甚至是 package name 去 startActivity 的都要處理,除非有先用其他方式檢查過要開到的 Activity 是否存在。
2013年7月16日 星期二
Android Notification PendingIntent Received Old Intent
在 Android 中遇到點了 Notification 但是 Activity 卻的 onNewIntent 收到舊的 Intent 的狀況。
有兩種解法,各有各的用途和適合情況。
1. 在 PendingIntent.getActivity(context, requestCode, intent, flag) 的 flag 傳入 PendingIntent.FLAG_UPDATE_CURRENT
這種作法會在 requestCode 相同,但收到多個 Notification 時,只會保留最後的 PendingIntent。
官方文件說 requestCode 沒有用到是騙~人~的!
適合用在如只需要保留最後最新的資訊用到。
2. 在 PendingIntent.getActivity(context, requestCode, intent, flag) 的 requestCode 傳入 count
這個 count 是用來計算這是發第幾個通知。通常會和 NotificationManager.notify(count, pendingIntent) 用同一個。
這適合用在每則通知都要傳入不同的通知訊息內容時使用。
有兩種解法,各有各的用途和適合情況。
1. 在 PendingIntent.getActivity(context, requestCode, intent, flag) 的 flag 傳入 PendingIntent.FLAG_UPDATE_CURRENT
這種作法會在 requestCode 相同,但收到多個 Notification 時,只會保留最後的 PendingIntent。
官方文件說 requestCode 沒有用到是騙~人~的!
適合用在如只需要保留最後最新的資訊用到。
2. 在 PendingIntent.getActivity(context, requestCode, intent, flag) 的 requestCode 傳入 count
這個 count 是用來計算這是發第幾個通知。通常會和 NotificationManager.notify(count, pendingIntent) 用同一個。
這適合用在每則通知都要傳入不同的通知訊息內容時使用。
2013年7月4日 星期四
GenyMotion - The Faster Android Emulator
GenyMotion 是一個 Android 的模擬器,不像 Google 提供的慢吞吞模擬器,速度快很多。
裡面的映像檔有提供諸如:Nexus One、Nexus S、Galaxy Nexus、Nexus 7...等等各種尺寸的映像檔。也有 Google Play 可以讓你下載程式回來用,換句話說測試或是想拿來轉轉神魔之塔也可以用這個。
目前還沒開始收費。
安裝方式:
1. 先下載安裝 Virtual Box 的 MacOS 版,官網:https://www.virtualbox.org/wiki/Downloads
2. 下載安裝 GenyMotion,官網:https://cloud.genymotion.com/
註冊時,這個帳號之後用來下載映像檔也會用到。
Have a nice emulator :p
2013年6月20日 星期四
Detect Android App Not In Foreground
在 iOS 只要寫在 UIApplicationDelegate 的 applicationdidenterbackground: 就可以。
但是在 Android 要做到相同功能就沒這麼簡單了。
其中用 getRunningTasks 又只限定在 debug 時可以用,真正上到 Google Play 是不能用的。因此就必須要在 onStop() 和 onWindowFocusChange(boolean) 做一些狀態判斷,看看是不是有任何一個 Activity 在前景。
參考:
http://vardhan-justlikethat.blogspot.in/2013/05/android-solution-to-detect-when-android.html
但是在 Android 要做到相同功能就沒這麼簡單了。
其中用 getRunningTasks 又只限定在 debug 時可以用,真正上到 Google Play 是不能用的。因此就必須要在 onStop() 和 onWindowFocusChange(boolean) 做一些狀態判斷,看看是不是有任何一個 Activity 在前景。
參考:
http://vardhan-justlikethat.blogspot.in/2013/05/android-solution-to-detect-when-android.html
2013年4月30日 星期二
Android Start Activity With ACTION_VIEW Cause ActivityNotFoundException
因為並不能保證所產生的 Intent 一定能夠找到可以用來處理的程式。
所以只要有用到 ACTION_VIEW 的都要加上 catch ActivityNotFoundException
防止程式當掉。
常見的如 WebView
2013年4月10日 星期三
Separate Production And Development/Test/Beta Environments
在 Android 上不像 iOS 一樣可以很方便的用 Application ID 能夠在同一支手機上面裝好幾個一樣的 App,來測試正式和測試環境。
但是網路上大家的說法都是修改 AndroidManifast.xml 中的 package 去產生成兩個 App。
確實這樣是變成兩個 App 沒錯,但是要做這件事情可是大工程,連帶有很多東西也都必須要跟著改。加上 Google 提供的工具也兩光兩光的。
如:R 檔都要全部重新 import、有用到自訂 Attr 的 xmlns 也都要改、Google Maps v2 API Key因為有對應到 package 也要改…
但是又不太想寫 shell script 去做這件事情,因此目前是想到可以用 activity-alias 加上 meta-data 來完成切換正式和測試環境。
不過這個方法也是有缺點:因為其實這樣還是同一個 App,所以如果有資料是存在 DB、SharedPreferences 或是其他地方也都會共用到,因此有時候可能需要清除資料才行。
流程如下:
程式碼如下:
在 AndroidManifest.xml 內
在 Activity 內
但是網路上大家的說法都是修改 AndroidManifast.xml 中的 package 去產生成兩個 App。
確實這樣是變成兩個 App 沒錯,但是要做這件事情可是大工程,連帶有很多東西也都必須要跟著改。加上 Google 提供的工具也兩光兩光的。
如:R 檔都要全部重新 import、有用到自訂 Attr 的 xmlns 也都要改、Google Maps v2 API Key因為有對應到 package 也要改…
但是又不太想寫 shell script 去做這件事情,因此目前是想到可以用 activity-alias 加上 meta-data 來完成切換正式和測試環境。
不過這個方法也是有缺點:因為其實這樣還是同一個 App,所以如果有資料是存在 DB、SharedPreferences 或是其他地方也都會共用到,因此有時候可能需要清除資料才行。
流程如下:
- 程式每次按返回鍵完全關掉再去點另外一個環境,才會真的切換環境。因為我並不想讓人在程式執行中就能夠切換環境,怕搞混。
- activity-alias 如果沒有重新讀取桌面的話,暫時不會出現 / 消失,所以可能要整個移除或重開機才會看到他對應出現或消失。
程式碼如下:
在 AndroidManifest.xml 內
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example"
android:versionCode="1"
android:versionName="1.0.0">
...
<application
android:icon="@drawable/ic_launcher"
android:label="@string/application_name"
android:theme="@style/Theme" >
<activity android:name=".WelcomeActivity"
android:screenOrientation="portrait" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity-alias
android:enabled="true"
android:name=".Welcome-Beta"
android:label="Beta版"
android:icon="@drawable/ic_launcher"
android:targetActivity=".WelcomeActivity">
<meta-data android:name="production" android:value="false"/>
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity-alias>
...
</application>
</manifest>
在 Activity 內
public class WelcomeActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
boolean production = true;
try {
ActivityInfo info = getPackageManager().getActivityInfo(getComponentName(), PackageManager.GET_META_DATA);
Bundle meta = info.metaData;
if(null != meta) {
production = meta.getBoolean("production", true);
}
} catch (NameNotFoundException e) {
Log.i(TAG, "Use production as target");
}
ApplicationConfigs.setTarget(production ? Stage.PRODUCTION : Stage.BETA);
}
}
在 ApplicationConfigs.java 內
public class ApplicationConfigs {
public enum Stage {
BETA,
PRODUCTION
}
private static Stage stage = Stage.PRODUCTION;
public static void setTarget(Stage target) {
stage = target;
}
private static String getEndPoint() {
switch(stage) {
case BETA:
return "http://beta.example.com";
default:
case PRODUCTION:
return "http://www.example.com";
}
}
}
2012年11月23日 星期五
2012年11月22日 星期四
Java RFC3986 URL Encode
http://designdotworks.blogspot.tw/2009/05/java-urlencode.html
因為在Java這邊目前沒有找到比較好的 RFC3986 規格的 URLEncode 。
也就是除了 a-zA-Z0-9 和 -_.~ 這四個符號,其他都要被 encode 過。 不過跟網頁內不同的是,我是改成
因為在Java這邊目前沒有找到比較好的 RFC3986 規格的 URLEncode 。
也就是除了 a-zA-Z0-9 和 -_.~ 這四個符號,其他都要被 encode 過。 不過跟網頁內不同的是,我是改成
- encoded = encoded.replace("*", "%2A");
- encoded = encoded.replace("%7E", "~");
- encoded = encoded.replace("+", "%20");
訂閱:
文章 (Atom)