動(dòng)態(tài)Android編程

原文轉(zhuǎn)載地址:http://tech.glowing.com/cn/dynamic-android-programming/

作者:劉聰

MENU

Home

SUBSCRIBE

MENU

動(dòng)態(tài)Android編程

11 DECEMBER 2015

注意:本文章有些例子需要對(duì)Java或Android有一定編程基礎(chǔ)。

與Python相比慕蔚,Java是一門比較嚴(yán)肅的語(yǔ)言。作為一個(gè)先學(xué)Python的程序員,做起Android難免會(huì)覺得不舒服,有些死板允坚,非常懷念decorator等方便的方法呵晨。為了實(shí)現(xiàn)一個(gè)簡(jiǎn)單的邏輯吐限,你可能需要寫很多額外的代碼。

舉個(gè)例子天揖,做數(shù)據(jù)庫(kù)查詢時(shí)夺欲,怎么從一個(gè)Cursor里取出類型為User的實(shí)例到List?

假設(shè)User是這樣的

classUser{

intid;

String name;

}

1. 找出User對(duì)應(yīng)所有的列和每列在Cursor對(duì)應(yīng)的索引今膊。

2. 如果索引存在些阅,根據(jù)類型取出正確的值。

3. 對(duì)于每個(gè)屬性斑唬,不斷重復(fù)上述步驟取出對(duì)應(yīng)的值市埋。

{

intcolumnIndex = cursor.getColumnIndex("id");

if(columnIndex >=0) {

instance.id = cursor.getInt(columnIndex);

}

}

{

intcolumnIndex = cursor.getColumnIndex("name");

if(columnIndex >=0) {

instance.name = cursor.getString(columnIndex);

}

}

這么做的問題在哪?

* 重復(fù)代碼

* 重復(fù)代碼

* 無(wú)聊

* 容易出錯(cuò)黎泣,不好維護(hù)

反射

我就是不想寫那么多無(wú)聊的代碼,怎么辦缤谎?要不試試范型/反射抒倚。

1. 取出所有的屬性。

2. 循環(huán)屬性隊(duì)列坷澡。

3. 把屬性設(shè)置成accessible托呕。

4. 找到索引。

5. 取出屬性的類型频敛,根據(jù)類型從Cursor里取出正確的值项郊。

publicstaticTfromCursor(Cursor cursor, Class cls){

T instance = cls.newInstance();

List fields = Arrays.asList(cls.getDeclaredFields())

for(Field field : fields) {

field.setAccessible(true);

String fieldName = field.getName();

intcolumnIndex = cursor.getColumnIndex(fieldName);

if(columnIndex <0) {

continue;

}

Class fieldType = field.getType();

if(fieldType.equals(int.class)) {

field.setInt(instance, cursor.getInt(columnIndex));

}else{

// 處理更多類型 String/float...

}

returninstance;

}

這樣我們就不用很無(wú)聊的把同樣的邏輯對(duì)于每種類型寫一遍又一遍。

Processor

用了反射后斟赚,也會(huì)有一些其他問題呆抑,這樣的代碼可讀性不是太好,不是很容易調(diào)試汁展。

既然我們可以通過(guò)反射實(shí)現(xiàn)這些邏輯鹊碍,為什么不干脆通過(guò)反射把這部分代碼直接生成出來(lái)呢?

1. 定義你要處理的annotation食绿。

@Retention(RetentionPolicy.CLASS)

@Target(ElementType.TYPE)

public@interface MyAnnotation {

Stringvalue();

}

2. 定義你的Processor類侈咕,繼承AbstractProcessor。

// Helper to define the Processor

@AutoService(Processor.class)

// Define the supported Java source code version

@SupportedSourceVersion(SourceVersion.RELEASE_7)

// Define which annotation you want to process

@SupportedAnnotationTypes("com.glow.android.MyAnnotation")

publicclassMyProcessorextendsAbstractProcessor{

3. 重載process方法器紧,實(shí)現(xiàn)生成代碼的邏輯耀销。

@Override

publicboolean process(Set annotations, RoundEnvironment roundEnv) {

Set elements = roundEnv.getElementsAnnotatedWith(MyAnnotation.class);

Set typeElements = ElementFilter.typesIn(elements);

for(TypeElement element : typeElements) {

ClassName currentType = ClassName.get(element);

MethodSpec.Builder builder = MethodSpec.methodBuilder("fromCursor")

.returns(currentType)

.addModifiers(Modifier.STATIC)

.addModifiers(Modifier.PUBLIC)

.addParameter(ClassName.get("android.database","Cursor"),"cursor");

...// 像在反射那節(jié)中,取出所有的fields铲汪,并循環(huán)取出每個(gè)元素熊尉,即每列

CodeBlock.Builder blockBuilder = CodeBlock.builder();

blockBuilder.beginControlFlow("");

blockBuilder.addStatement("int columnIndex = cursor.getColumnIndex($S)", column);

blockBuilder.beginControlFlow("if (columnIndex >= 0)");

ColumnType columnType = columnTypeMap.get(column);

String cursorType =null;

if(columnType == ColumnType.INT) {

cursorType ="Int";

}elseif(columnType == ColumnType.LONG) {

cursorType ="Long";

}elseif(columnType == ColumnType.FLOAT) {

cursorType ="Float";

}elseif(columnType == ColumnType.STRING) {

cursorType ="String";

}else{

abort("Unsupported type", element);

}

blockBuilder.addStatement("instance.$L = cursor.get$L(columnIndex)",

fieldName,

cursorType);

blockBuilder.endControlFlow();

blockBuilder.endControlFlow();

builder.addCode(blockBuilder.build());

// 結(jié)束循環(huán)

// 生成代碼到文件里

String className = ...// 設(shè)置你要生成的代碼class名字

JavaFileObject sourceFile =processingEnv.getFiler().createSourceFile(className, element);

Writer writer = sourceFile.openWriter();

javaFile.writeTo(writer);

writer.close();

}

returnfalse;

}

4. 修改Userclass加上annotation

@MyAnnotation

classUser{

intid;

String name;

}

5. 用apt工具把我們上面寫的庫(kù)加到編譯過(guò)程去。

Tips:

* 用AutoService可以方便的生成Processor方法

* 強(qiáng)推Javapoet掌腰,用來(lái)生成漂亮的代碼

AOP

AOP的做法和Processor類似狰住,這里就不詳述。你可能用AspectJ齿梁。

Gradle plugin

最后我還是沒有完全采用上面的方法催植,因?yàn)?

* 在編譯時(shí)生成的代碼在打開編譯器時(shí)找不到

* 有時(shí)候有些特殊需求,比如很多屬性要在多個(gè)地方共享使用勺择,能配置化會(huì)更好些

于是我們就用了Gradle Plugin來(lái)通過(guò)可配置文件生成代碼

以下是簡(jiǎn)單的例子:

1. 定義配置文件创南,這里選用比較簡(jiǎn)單的toml文件

srcDir ="src/main/java"

pkg ="com.glow.android.storage.db"

[[tables]]

name ="user"

[[tables.columns]]

name ="id"

type="int"

isKey =true

2. 在buildSrc項(xiàng)目里創(chuàng)建Plugin

publicclassDbPluginimplementsPlugin {

@Override

voidapply(Project project){

project.task('initDb') << {

def dir = project.getProjectDir()

def file =newFile(dir,"table.toml")

generateCode(dir,newToml().parse(file).to(DB.class))

}

}

staticvoidgenerateCode(File dir, DB db){

def outputDir =newFile(dir, db.srcDir)

outputDir.mkdirs()

for(Table table : db.tables) {

//Process it

}

}

}

3. 像在上節(jié)講的那樣生成代碼,把數(shù)據(jù)源從annotation換成toml里的定義

4. 在項(xiàng)目里把Plugin引用進(jìn)去省核,并執(zhí)行

5. 這樣就可以得到漂亮的已經(jīng)生成好的代碼

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末稿辙,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子气忠,更是在濱河造成了極大的恐慌邻储,老刑警劉巖未桥,帶你破解...
    沈念sama閱讀 219,110評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異芥备,居然都是意外死亡冬耿,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,443評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門萌壳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)亦镶,“玉大人,你說(shuō)我怎么就攤上這事袱瓮$凸牵” “怎么了?”我有些...
    開封第一講書人閱讀 165,474評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵尺借,是天一觀的道長(zhǎng)绊起。 經(jīng)常有香客問我,道長(zhǎng)燎斩,這世上最難降的妖魔是什么虱歪? 我笑而不...
    開封第一講書人閱讀 58,881評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮栅表,結(jié)果婚禮上笋鄙,老公的妹妹穿的比我還像新娘。我一直安慰自己怪瓶,他們只是感情好萧落,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,902評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著洗贰,像睡著了一般找岖。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上敛滋,一...
    開封第一講書人閱讀 51,698評(píng)論 1 305
  • 那天许布,我揣著相機(jī)與錄音,去河邊找鬼矛缨。 笑死爹脾,一個(gè)胖子當(dāng)著我的面吹牛帖旨,可吹牛的內(nèi)容都是我干的箕昭。 我是一名探鬼主播,決...
    沈念sama閱讀 40,418評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼解阅,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼落竹!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起货抄,我...
    開封第一講書人閱讀 39,332評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤述召,失蹤者是張志新(化名)和其女友劉穎朱转,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體积暖,經(jīng)...
    沈念sama閱讀 45,796評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡藤为,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,968評(píng)論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了夺刑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片缅疟。...
    茶點(diǎn)故事閱讀 40,110評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖遍愿,靈堂內(nèi)的尸體忽然破棺而出存淫,到底是詐尸還是另有隱情,我是刑警寧澤沼填,帶...
    沈念sama閱讀 35,792評(píng)論 5 346
  • 正文 年R本政府宣布桅咆,位于F島的核電站,受9級(jí)特大地震影響坞笙,放射性物質(zhì)發(fā)生泄漏岩饼。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,455評(píng)論 3 331
  • 文/蒙蒙 一薛夜、第九天 我趴在偏房一處隱蔽的房頂上張望忌愚。 院中可真熱鬧,春花似錦却邓、人聲如沸硕糊。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,003評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)简十。三九已至,卻和暖如春撬腾,著一層夾襖步出監(jiān)牢的瞬間螟蝙,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,130評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工民傻, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留胰默,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,348評(píng)論 3 373
  • 正文 我出身青樓漓踢,卻偏偏與公主長(zhǎng)得像牵署,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子喧半,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,047評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理奴迅,服務(wù)發(fā)現(xiàn),斷路器挺据,智...
    卡卡羅2017閱讀 134,664評(píng)論 18 139
  • 很實(shí)用的編程英語(yǔ)詞庫(kù)取具,共收錄一千五百余條詞匯脖隶。 第一部分: application 應(yīng)用程式 應(yīng)用、應(yīng)用程序app...
    春天的蜜蜂閱讀 1,362評(píng)論 0 22
  • WHY 如果說(shuō)OOP(面向?qū)ο蟮某绦蛟O(shè)計(jì))的主要思想是將問題歸分到相應(yīng)的對(duì)象(類)中去實(shí)現(xiàn),再把相關(guān)類模塊化使得模...
    野生大P閱讀 957評(píng)論 0 8
  • ¥開啟¥ 【iAPP實(shí)現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開一個(gè)線程暇检,因...
    小菜c閱讀 6,426評(píng)論 0 17
  • 站在一顆星球上 看滿天無(wú)數(shù)繁星 你不會(huì)孤單 因?yàn)槲乙苍诹硪活w星球上 一樣的看著同一片星空
    辛安小閱讀 572評(píng)論 41 51