Kotlin初探(一)基礎構建

Kotlin現在是google推薦的android開發語言,現在嘗試著通過寫一個Kotlin的應用來學習這門語言,學習的過程也在這里記錄。Github項目地址

因為是初次接觸Kotlin,如果在文章中有任何錯誤的地方歡迎指正

內容

Part 1:Kotlin初探(一)基礎構建

Part2:Kotlin初探(二)數據加載

APP構建


使用Android Studio3.0(后面簡稱為AS3)創建一個包含Kotlin以及C++的應用。


創建

后面根據需要配置一下即可。

Gradle配置


project build.gradle 配置用默認的就可以,如果需要使用lambda AS3會提示并自動添加
app build.gradle

//載入插件 使用的是apply函數,apply支持from、plugin、to
//apk 必須加載這個插件
apply plugin: 'com.android.application'
//kotlin插件
apply plugin: 'kotlin-android'
//kotlin綁定控件插件
apply plugin: 'kotlin-android-extensions'
//kotlin 注解處理器
apply plugin: 'kotlin-kapt'

控件綁定插件和注解處理插件不是必須的

dependencies {
    implementation project(':gearlibrary')
    implementation fileTree(include: ['*.jar'], dir: 'libs')
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jre7:$kotlin_version"
    implementation 'com.android.support:appcompat-v7:26.1.0'
    implementation 'com.android.support.constraint:constraint-layout:1.0.2'
    implementation 'com.android.support:cardview-v7:26.1.0'
    implementation 'com.android.support:design:26.1.0'
    ...
}

這里注意compile被標記過時,通過兩個新指令代替
api:完全等同于compile指令,沒區別
implementation:對于使用了該命令編譯的依賴,對該項目有依賴的項目將無法訪問到使用該命令編譯的依賴中的任何程序,
也就是將該依賴隱藏在內部,而不對外部公開,如果你的appModel中有使用依賴那么需要使用api指令對全局開放

框架


因為是剛開始使用kotlin編程,所以只使用了一些常用的框架,業務邏輯框架選擇使用MVP,網絡使用Retrofit+Okio+Rxjava,圖片處理Glide4,Rxjava生命周期Rxlifecycle

image.png

這里對activity的分類是按照功能來劃分的,主要也是想看一下這樣劃分與平常activity、adapter、fragment這種劃分平常開發中有什么不同。

類說明


后面除了對類說明以外,還會對相關的kotlin語言加以說明

GearApp

app全局初始化,以及自定義屬性

class GearApp : Application(){

    override fun onCreate() {
        super.onCreate()
        init()
    }

    private fun init(){

    }
}

1.java繼承使用extends關鍵字,kotlin的繼承使用 : 表示,并且父類后面需要加()
2.kotlin是強制使用override關鍵字,而不是以前java@override
3.fun 表示一個方法,前面可以用可見性修飾符修飾


可見性修飾符

BaseActivityBasePActivity

BasePActivity

RxLifecycleActivity 主要是為Rxjava生命周期服務,它會提供一個常量并綁定到activity的生命周期中,并提供可見的方法以供Rxjava使用。
BaseActivity 一般app都會有這個類集合共有的方法、參數、初始化等
BasePActivity 主要是為MVP的構建方式服務,提供一個延遲初始化的presenter

abstract class BasePActivity<T : BaseContract.BasePresenter<*>> : BaseActivity(){
    protected lateinit var mPresenter : T
}

1.var標識變量、val標識常量,如果你的常量需要被java方法調用還需要在val前面增加const關鍵字,因為java只認為被 const val 標識的參數為java的常量。
2.lateinit 標識為延遲初始化變量,當你的變量不為null但是又不想馬上初始化的話,就可以使用這個標識符,需要注意的是延遲初始化的變量需要注明參數類型,不然IDE會報錯。
3.既然變量可以延遲初始化,常量也是可以的,使用lazy來標識

val name: String by lazy { "abc" }
val name by lazy { 123 }

4.參數類型在kotlin中可以用: String標識,也可以不寫直接賦值kotlin會自動推導。
5.泛型的寫法<T : BaseContract.BasePresenter<*>>與java寫法類似。

MVP基礎類BaseContractAbstractPresenter

BaseContract、AbstractPresenter

主要想法是通過這樣的結構使得 PresenterA 和 ActivityA 之間通過接口來相互調用和解耦,并且ContractA中的接口后期還可以通過AnnotationProcessor方式直接生成。這樣MVP相對于之前的MVC只要多寫一個Presenter Class就可以并不會增加太多的影響,當然要注意的試因為這只是一個簡單的APP所以M、P的邏輯是合并到一起的。

BaseContract中主要是提供presenter、view接口的公共方法
AbstractPresenter 提供mView引用并在構造方法中初始化

abstract class AbstractPresenter<V : BaseContract.BaseView>(v : V) {
    var mView = v
    protected set
}

PresenterDataWrapper 會提供Presenter需要的數據模型

open class PresenterDataWrapper<R : Any,V : BaseContract.BaseView>(v : V) : AbstractPresenter<V>(v) {
    lateinit var mData : R
    protected set
    /**
     * 處理數據
     */
    protected open fun processData(d : R?) {}
    /**
     * 數據出錯
     */
    protected open fun errorData(error : String) {}
}

1.kotlin語言為變量自動裝箱也就是提供set()\get()方法,那么有時候你可能不需要set或者get開放,那么可以使用protected/private set方式來標識set方法的可見度,而get方法的可見度與該參數的可見度相同。
2.(v : V)相當于java中的構造方法,類型由泛型確定。
當你需要一個構造方法時就需要這樣寫Class(str: String, num: Int,...)
3.mData 會延遲加載并且類型由R : Any 泛型確定。
4.Any 類似于java的Object,但方法只有三個equals()、hashCode()、toString()。

網絡相關NetManagerApiManager

NetManager 提供全局的okClient和Retrofit引用

class NetManager {
    /**
     * 初始化Ok引用和Retrofit引用
     */
    companion object {
        var mClient : OkHttpClient = OkHttpManager.getInstance()
                .setHeader("apikey","beae89ef686795322d5a3c48579875d5")
                .build()
                .client
        var mRetrofit : Retrofit = GearHttpServiceManager.build(APIConfig.BASE_URL,mClient)
    }
}

1.在kotlin如果需要靜態參數需要使用 companion object {...} 的方式,這種方式類似于下面的java code

public final class NetManager {
   @NotNull
   private static OkHttpClient mClient;
   @NotNull
   private static Retrofit mRetrofit;
   public static final NetManager.Companion Companion = new NetManager.Companion((DefaultConstructorMarker)null);

   static {
      OkHttpClient var10000 = OkHttpManager.getInstance().setHeader("apikey", "beae89ef686795322d5a3c48579875d5").build().getClient();
      Intrinsics.checkExpressionValueIsNotNull(var10000, "OkHttpManager.getInstanc…)\n                .client");
      mClient = var10000;
      Retrofit var0 = GearHttpServiceManager.build("http://120.27.118.74/", Companion.getMClient());
      Intrinsics.checkExpressionValueIsNotNull(var0, "GearHttpServiceManager.b…IConfig.BASE_URL,mClient)");
      mRetrofit = var0;
   }
   public static final class Companion {
      @NotNull
      public final OkHttpClient getMClient() {
         return NetManager.mClient;
      }

      public final void setMClient(@NotNull OkHttpClient var1) {
         Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
         NetManager.mClient = var1;
      }

      @NotNull
      public final Retrofit getMRetrofit() {
         return NetManager.mRetrofit;
      }

      public final void setMRetrofit(@NotNull Retrofit var1) {
         Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
         NetManager.mRetrofit = var1;
      }

      private Companion() {
      }
   }
}

ApiManager提供接口調用方法

object ApiManager {
    val travelNotesApi : TravelNotesApi = NetManager.mRetrofit.create(TravelNotesApi::class.java)
}

1.object 標識該類為單例的

會生成java代碼:
public final class ApiManager {
   @NotNull
   private static final TravelNotesApi travelNotesApi;
   public static final ApiManager INSTANCE;

   @NotNull
   public final TravelNotesApi getTravelNotesApi() {
      return travelNotesApi;
   }

   private ApiManager() {
      INSTANCE = (ApiManager)this;
      Object var10000 = NetManager.Companion.getMRetrofit().create(TravelNotesApi.class);
      Intrinsics.checkExpressionValueIsNotNull(var10000, "NetManager.mRetrofit.cre…avelNotesApi::class.java)");
      travelNotesApi = (TravelNotesApi)var10000;
   }

   static {
      new ApiManager();
   }
}

Next Post:

Kotlin初探(二)數據加載

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,836評論 6 540
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,275評論 3 428
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,904評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,633評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,368評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,736評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,740評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,919評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,481評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,235評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,427評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,968評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,656評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,055評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,348評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,160評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,380評論 2 379

推薦閱讀更多精彩內容