2012年11月9日 星期五

[C語言考題-1]strcpy()實作

Question:已知strcpy函數的原型是 char* strcpy(char* strDest,const char* strSrc);
1.不使用原始函式庫,請實現strcpy函數
2.解釋為什麼要返回char*  -->其實我還是不知道為何要return? 沒return也可以run說...

以下收集幾種"寫"法,目前我的程度還沒辦法明確告訴您哪一個最"好",但都可以執行,請自行參考:

寫法一(網路上最常看到的)
char* strcpy(char* strDest,const char* strSrc)
{
 if((strDest == NULL) || (strSrc == NULL)) //[1]
  throw "Invalid Arguments"; //[2]
 char* strDestCopy = strDest; //[3]
 while((*strDest++ = *strSrc++) != '\0')//[4]
  ;
 return strDestCopy;
}
這裡要注意[4]前面的 \0,大部分轉貼的都是寫/0,會有warning並沒辦法執行。(害我try了一陣子)

寫法二(Wiki)
http://zh.wikipedia.org/wiki/Strcpy
char *strcpy(char *dest, const char *src)
{
  const char *p;
  char *q; 

  for(p = src, q = dest; *p != '\0'; p++, q++)
    *q = *p;
   
  *q = '\0';

  return dest;
}

寫法三(某部落客)
http://vijayinterviewquestions.blogspot.tw/2007/07/implement-strcpy-function.html

char *mystrcpy(char *dst, const char *src)
{
char *ptr;
ptr = dst;
while(*dst++=*src++);
return(ptr);
}
or
char *my_strcpy(char dest[], const char source[])
{
int i = 0;
while (source[i] != '\0')
{
dest[i] = source[i];
i++;
}
dest[i] = '\0';
return(dest);
}

大同小異囉~ 大家參考參考~

2012年6月26日 星期二

簽章自動輸入密碼(Auto Sing Key)

這篇是:Creating Release Keys and Signing Builds 的延伸。

 當您作到
./build/tools/releasetools/sign_target_files_apks -d vendor/<vendor_name>/security/<product_name> <product_name>-target_files.zip signed-target-files.zip
這一步後,都會詢問密碼,而且一次都是問四次,對於重複使用者或自動化來說,都是不太方便的。
我找到一個地方是在寫這段流程的Python程式,建議大家可以從這個地方來修改:
位置在 build\tools\releasetools\common.py
line:562
 result[k] = getpass.getpass("Enter password for %s key> "
                                      % (k,)).strip()
getpass 這個方法就是用來提示使用者輸入,並回傳值。
因此很簡單的,你可以將你的密碼直接指定給result[k],他就不會一直問了。
result[k]='xxxx'
當然你可以用各種方式來取得密碼,我這裡就不贅述了~
給大家參考!







2012年6月7日 星期四

Android 4.0 CTS 設備設定


還是需以官方文件為準,只是有些部份我覺得寫得不是很清楚或是過時,可以參考看看!

1.確認裝置手機或平台是user-build

2.settings-->Backup & reset-->Factorydata reset

3.平台插入一張空的sd卡(沒有空的他也會把你刪掉)

4.平台安裝APK:CtsDelegatingAccessibilityService.apk,CtsDeviceAdmin.apk

5.settings打開Wi-Fi,並確定ap可連外

6.settings-->Security-->Screenlock-->None

7.settings-->Security-->Deviceadministrators-->Enable

8.settings-->Display-->Sleep-->30minutes或None

9.settings-->Accessibility-->EnableDelegating Accessibility Service

10.settings-->Developeroptions-->Enable USB debugging、Stay awake、Allowmock locations

11.開始測試前畫面需停留在Homescreen,並且不要有其他的task在跑,也不要任何操作。

2012年6月5日 星期二

system/app權限開啟

有時候會需要push apk 到system/app下,才有足夠的權限。

remount 不行的話,可以用以下指令:

mount -o rw,remount /dev/mmcblkp0 /system

2012年4月17日 星期二

螢幕鎖(Screen Lock)開啟/關閉

預設是開啟的,要關閉有兩個地方要改:

1. frameworks\policies\base\phone\com\android\internal\policy\impl\KeyguardViewMediator.java

  private boolean mExternallyEnabled = true;
  //set to false if you want disable Screen Lock

2. frameworks/base/packages/SettingsProvider/res/values/defaults.xml

 60000
 // set to -1 if you want disable Screen Lock

兩個都要設定才ok~

2012年3月8日 星期四

本地和設備間傳遞檔案

從設備取得檔案

adb pull /data/data/com.example.sample   ./sample/

從本機上傳檔案
adb push ./sample/  /sdcard/sample/

當然也可以用DDMS,但是有時候指令是方便許多~


參考

2012年3月4日 星期日

安裝程式按安裝時間排序

結合「取得安裝程式列表」和「取得安裝程式時間」這兩個功能就可以完成。

取得安裝程式列表:
private void bindAllApps() {
        
        Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);
        mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
        mAllApps = mPackageManager.queryIntentActivities(mainIntent, 0);    
        sortappInstalledTime(mAllApps);
    }

根據安裝時間排序:
public void sortappInstalledTime(List<ResolveInfo> mAllApps2) {
        Collections.sort(mAllApps2, new Comparator<Object>() {
            public int compare(Object o1, Object o2) {
                ResolveInfo p1 = (ResolveInfo) o1;
                ResolveInfo p2 = (ResolveInfo) o2;
                Date d1 = getInstallTime(p1.activityInfo.packageName);
                Date d2 = getInstallTime(p2.activityInfo.packageName);

                if (d1 != null && d2 != null) {
                    if (d1.after(d2)) {
                        return -1;
                    } else if (d1 == d2) {
                        return 0;
                    } else {
                        return 1;
                    }
                } else {
                    return 1;
                }
            }
        });
    }

取得安裝時間:
    public Date getInstallTime(String packageName) {
        PackageManager pkgMgt = this.getPackageManager();
        return firstNonNull(installTimeFromPackageManager(pkgMgt, packageName),
                apkUpdateTime(pkgMgt, packageName));
    }

    private Date apkUpdateTime(PackageManager packageManager, String packageName) {

        try {
            android.content.pm.ApplicationInfo info = packageManager
                    .getApplicationInfo(packageName, 0);
            File apkFile = new File(info.sourceDir);
            return apkFile.exists() ? new Date(apkFile.lastModified()) : null;
        } catch (NameNotFoundException e) {
            return null; // package not found
        }
    }

    private Date installTimeFromPackageManager(PackageManager packageManager,
            String packageName) {
        try {
            PackageInfo info = packageManager.getPackageInfo(packageName, 0);
            Field field = PackageInfo.class.getField("firstInstallTime");
            long timestamp = field.getLong(info);
            return new Date(timestamp);
        } catch (NameNotFoundException e) {
            return null; // package not found
        } catch (IllegalAccessException e) {
        } catch (NoSuchFieldException e) {
        } catch (IllegalArgumentException e) {
        } catch (SecurityException e) {
        }
        // field wasn't found
        return null;
    }

            private Date firstNonNull(Date... dates) {
                for (Date date : dates)
                    if (date != null)
                        return date;
                return null;
            }

20120702補充:
後來發現,如果你像我一樣是要迴圈取得多個apk來排序的話,這樣的方式效能很x。可改用uid來排序,可以達到一樣的效果。
根據uid大小排序:
public void sortappByUid(List<ResolveInfo> mAllApps2) {
        Collections.sort(mAllApps2, new Comparator<ResolveInfo>() {
            public int compare(ResolveInfo o1, ResolveInfo o2) {
                ResolveInfo r1 = (ResolveInfo) o1;
                ResolveInfo r2 = (ResolveInfo) o2;
                int uid1 = r1.activityInfo.applicationInfo.uid;
                int uid2 = r2.activityInfo.applicationInfo.uid;

                
                    if (uid1>=uid2) {
                        return -1;
                    } else if (r1 == r2) {
                        return 0;
                    } else {
                        return 1;
                    }
                
            }
        });
    }

2012年2月16日 星期四

GridView Null Pointer Exception

error log:
 W/dalvikvm( 727): threadid=1: thread exiting with uncaught exception (group=0x2aac47e8)  
 E/AndroidRuntime( 727): FATAL EXCEPTION: main  
 E/AndroidRuntime( 727): java.lang.NullPointerException  
 E/AndroidRuntime( 727):    at android.widget.GridView.setupChild(GridView.java:1246)  
 E/AndroidRuntime( 727):    at android.widget.GridView.makeAndAddView(GridView.java:1222)  
 E/AndroidRuntime( 727):    at android.widget.GridView.makeRow(GridView.java:265)  
 E/AndroidRuntime( 727):    at android.widget.GridView.fillDown(GridView.java:218)  
 E/AndroidRuntime( 727):    at android.widget.GridView.fillSpecific(GridView.java:482)  
 E/AndroidRuntime( 727):    at android.widget.GridView.layoutChildren(GridView.java:1122)  
 E/AndroidRuntime( 727):    at android.widget.AbsListView.onLayout(AbsListView.java:1147)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.widget.RelativeLayout.onLayout(RelativeLayout.java:909)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.widget.AbsoluteLayout.onLayout(AbsoluteLayout.java:120)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.widget.FrameLayout.onLayout(FrameLayout.java:333)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.widget.FrameLayout.onLayout(FrameLayout.java:333)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.widget.FrameLayout.onLayout(FrameLayout.java:333)  
 E/AndroidRuntime( 727):    at android.view.View.layout(View.java:7085)  
 E/AndroidRuntime( 727):    at android.view.ViewRoot.performTraversals(ViewRoot.java:1045)  
 E/AndroidRuntime( 727):    at android.view.ViewRoot.handleMessage(ViewRoot.java:1727)  
 E/AndroidRuntime( 727):    at android.os.Handler.dispatchMessage(Handler.java:99)  
 E/AndroidRuntime( 727):    at android.os.Looper.loop(Looper.java:123)  
 E/AndroidRuntime( 727):    at android.app.ActivityThread.main(ActivityThread.java:4627)  
 E/AndroidRuntime( 727):    at java.lang.reflect.Method.invokeNative(Native Method)  
 E/AndroidRuntime( 727):    at java.lang.reflect.Method.invoke(Method.java:521)  
 E/AndroidRuntime( 727):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:868)  
 E/AndroidRuntime( 727):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:626)  
 E/AndroidRuntime( 727):    at dalvik.system.NativeStart.main(Native Method)  
log有點難看出是哪段程式碼出了問題,只知道GridView出了問題,通常這種問題都是出現在getView的時候回傳null值產生,所以一定要避免漏接了null的View
 if (xxxView = = null)  
  {  
    ...  
 }   
參考網址

2012年2月15日 星期三

提高背景圖的設定

在一般手機螢幕的圖檔,其實解析度稍微低一些,並不會太明顯。
但是在電視的環境就會要更高的要求。
因此要求出圖的設計師都要提供32bit圖檔。
但是android預設的解析度是 RGB_565,因此要另外更改設定成ARGB_8888才行,否則會有color loss的狀況。


我的狀況是在背景圖,可以用下面的code來設定。
Code:

        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inPreferredConfig = Bitmap.Config.ARGB_8888;   
        Bitmap myBitmap =BitmapFactory.decodeResource(mContext.getResources(),R.drawable.m_background,options);  
        BitmapDrawable bd = new BitmapDrawable(this.getResources(), myBitmap);
        youview.setBackgroundDrawable(bd);


當然,為大幅增加記憶體用量,要同時使用gc的機制會比較安全~

2012年2月9日 星期四

如何移除與置換預設Launcher


如果想要把預設的Launcher給換掉,可以利用以下方法

使用adb工具,下以下指令:

adb remount

adb shell rm system/app/Launcher2.apk

adb uninstall com.android.launcher

adb install Launcher_xxx.apk (or use eclipse Run ..)

這樣就可以了喔~

Android如何找到最適合的資源檔

參考官方的說明:How Android Finds the Best-matching Resource


1. 消除和設備設定不符合的資源資料夾(qualifier),例如語言設定為en,就會把 fr-rCA消除掉
    drawable/
  drawable-en/
  drawable-fr-rCA/
  drawable-en-port/
  drawable-en-notouch-12key/
  drawable-port-ldpi/
  drawable-port-notouch-12key/

  例外:螢幕解析度在這個階段是唯一衝突但是不會刪的資源資料夾,例如device是hdpi但是ldpi還是會留著 。(想瞭解更多請參考Supporting Multiple Screens)

2.選擇下一個最高順位的qualifier
(最高是 MCC,MNC,Language...) 請參考這個table的順位 
3.有沒有符合的資料夾,沒有就回到上一步直到找到。
4.消除掉沒有最高順位qualifier的資源資料夾,例如:把沒有語言qualifier的資料夾都消掉。
  drawable/
  drawable-en/
  drawable-en-port/
  drawable-en-notouch-12key/
  drawable-port-ldpi/
  drawable-port-notouch-12key/
  例外:如果qualifier和螢幕解析度有不符合的狀況,Android會自動選擇最符合設備螢幕密度的qualifier,一般來說andorid會選擇把高解析度的圖檔給縮小,而不是把低解析度的讀檔給放大。
5.再回到2,直到刪到只剩下一個資源資料夾。





2012年2月8日 星期三

Android View ScrollBar 設定


WebView 和 GridView的ScrollBar 設定

.setVerticalScrollBarEnabled(false);  //取消Vertical ScrollBar顯示

.setHorizontalScrollBarEnabled(false); //取消Horizontal ScrollBar顯示

.setScrollBarStyle(View.SCROLLBARS_XXX) ;// 設定ScrollBar樣式

樣式請參考這裡

.setHorizontalScrollbarOverlay(boolean overlay);
//設定Horizontal ScrollBar是否覆蓋樣式

.setVerticalScrollbarOverlay(boolean overlay);
//設定 Vertical ScrollBar  ScrollBar是否覆蓋樣式