一、引言
A modern programming language that makes developers happier.
正如官網的slogan所描述:kotlin,是一門讓程序員寫代碼時更有 幸福感 的 現代 語言。
同時,也正如維基百科裏介紹:
JetBrains公司希望Kotlin能夠推動IntelliJ IDEA的銷售
kotlin是一門奔着錢而生的語言,我相信他一定會成爲一門有 “錢途” 的語言。
JetBrains這家公司真的把程序員當 人,把程序員當用戶來對待。從kotlin的迭代和發版節奏,可以看出來,他們迭代kotlin就像我們程序員迭代我們公司的app一樣,很重視用戶體驗,很重視程序員的開發體驗,java哪個語法寫起來很痛苦,他們就改善那個語法。java程序員缺什麼,他們就給kotlin造什麼。作爲一個程序員,真心覺得,這種感覺真好。
大叔,已經有兩年多的kotlin開發經驗了,我能感受到,kotlin帶來的幸福感,甚至,大叔在寫kotlin的時候,有時會有驚喜感。讓我有一種感慨:哦…… 原來代碼還能這麼寫,還能有這種語法。
寫了近10年的java代碼,大叔幾乎已經習慣接受java的一切,包括一些不思進取的寫法,大叔想當然的以爲編程就應該是這樣的。kotlin正在顛覆大叔習慣了的這一切。【當然,這裏並不是貶低java,java不僅是優秀的也是偉大的,java迭代了24+年,java的生態是衆多語言無法比擬的,也正是java的優秀才有了kotlin、Clojure、groovy等一系列優秀的語言,java的偉大也是無法複製的】,我猜這裏肯定有人要跳出來說,批評大叔,php纔是最好的語言,讓大叔去學php~ 好吧,大叔有空再學學~
大叔身邊有很多小夥伴,在猶豫到底學kotlin還是flutter?
他們一方面覺得java已經夠用了,另一方面覺得google爸爸大力發展他的親兒子flutter,覺得乾兒子kotlin畢竟不是一個血脈。而且flutter誇多端的能力大於kotlin,於是得出結論,沒有必要學習kotlin。
然而,看到現在很多項目很多公司都在用kotlin,google官方文檔和demo都有kotlin的身影。又開始搖擺,糾結,迷茫,焦慮。大叔覺得小朋友才做選擇,公司需要什麼,大叔就學什麼。
我們簡單捋一捋flutter和kotlin的本質區別吧。
flutter和kotlin的定位完全不同,他們的原理完全不一樣。
flutter可以簡單理解爲UI引擎,跨平臺能力,是他與生俱來。就像遊戲引擎一樣。
kotlin的優勢在於他繼承了java的所有能力。而且還在不斷拓展自己的能力。他的能力不限於android開發,後端開發,js開發。甚至,大叔看到,kotlin有搶python地盤的意圖。
kotlin和flutter並不衝突的兩個技術,即便團隊選擇部分模塊使用 flutter開發app,android原生那部分代碼,kotlin依然可以發光發熱。
好像廢話講的有點多。我們進入正題吧。
二、kotlin如何解決java開發的痛點?
我們一個個語法特性來講吧,先從最基礎的字符串開始
2.1 kotlin字符串支持,三引號"""
我們直奔代碼,JavaHtml.java
public class JavaHtml {
private static final String HTML =
"<!DOCTYPE HTML>\n" +
"<HTML>\n" +
"<head>\n" +
"<meta charset=\"utf-8\">\n" +
"<title>IT互聯網大叔</title>\n" +
"</head>\n" +
"<body>\n" +
" <h1>大叔的標題</h1>\n" +
" <p>\"大叔\"的段落1</p>\n" +
"</body>\n" +
"</HTML>";
}
字符串中存在雙引號時,需要 反斜槓轉移;
想換行來提高可讀性,於是必須要字符串拼接;
原本易懂的html代碼變得晦澀難懂。
這代碼是人看的嗎?從java發佈到現在,我們忍了24年。都沒有改進。
我們再來看看,kotlin字符串的三引號。KotlinHtml.kt
val HTML = """
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>IT互聯網大叔</title>
</head>
<body>
<h1>大叔的標題</h1>
<p>"大叔"的段落</p>
</body>
</html>
"""
優雅嗎?happy嗎?
2.2 kotlin字符串模板
我們感受下,PeopleJava的toString方法。----- 這個toString方法是AndroidStudio自動生成的。
public class PeopleJava {
private String name;
private int age;
public PeopleJava(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "PeopleJava{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
我們再感受下,PeopleKt 的toString方法。
class PeopleKt (private var name:String, private var age:Int){
override fun toString(): String {
return "PeopleKt{name='$name', age=$age}"
}
}
來我們再用kotlin寫個main方法,把兩個對象日誌輸出看看。
fun main() {
print(PeopleJava("IT互聯網大叔", 18).toString())
println()
print(PeopleKt("IT互聯網大叔", 18).toString())
}
日誌輸出:
以前總聽人說簡潔美,簡潔美,簡潔怎麼能跟美關聯起來呢。
現在大叔明白了。簡潔不僅美,簡潔的東西更簡單易懂。
2.3 kotlin支持默認參數
默認參數我們並不陌生,大叔在大學時,學c/c++的時候就知道默認參數。概念也非常簡單。
如下代碼:
class A {
fun foo(i: Int = 10) {
/****/
}
}
A.foo()方法,如果調用時,不傳參數的話, 參數i 默認是10
大叔的自我拷問:
默認參數,有什麼用呢?能解決android開發中的什麼痛點呢?
默認參數,有什麼用呢?能解決android開發中的什麼痛點呢?
默認參數,有什麼用呢?能解決android開發中的什麼痛點呢?
在android開發中,我們寫一個自定義View的時候,我們的構造函數往往要寫好幾個。
public class JavaTextView extends AppCompatTextView {
public JavaTextView(Context context) {
super(context);
}
public JavaTextView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public JavaTextView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
}
有了默認參數,之後。。。
class KotlinTextView : AppCompatTextView {
constructor(
context: Context,
attrs: AttributeSet? = null,
defStyleAttr: Int = android.R.attr.textViewStyle
) : super(context, attrs, defStyleAttr){
}
}
2.4 kotlin 協程
我們直接上代碼吧,下面代碼來自於 : 一分鐘入門kotiln協程,線程切換
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
//上下文切換到IO主線程
GlobalScope.launch(Dispatchers.IO) {
Log.i(TG, "Dispatchers.IO isMainThread ${isMain()}")//輸出false
//上下文切換到主線程
GlobalScope.launch(Dispatchers.Main){
Log.i(TG, "Dispatchers.Main isMainThread ${isMain()}")//輸出true
}
}
}
以前我們習慣使用Handler、AsyncTask 等類來實現線程切換。
但是有了協程之後。我覺得比以前寫Handler的時候幸福多了。
切換線程,只是協程的冰山一角,協程很強大,值得深入學習。
2.5 kotlin的use函數,自動關閉closeable
我們分別用java和kotlin實現一個函數,讀取文件的第一行:
顯然kotlin的use比java要優雅的多。
我們來看看,use函數的實現吧。
2.6 findViewById()
java代碼:JavaFindViewActivity.java
最開始的時候,每次findViewById之後還要類型轉換,如下代碼
TextView tvTitle = (TextView) findViewById(R.id.tv_title);
後來,不記得哪個androidstudio版本 更新之後就不用做 類型強轉 了。代碼看這簡潔了很多。
再後來有人做了一個叫ButterKnife的插件,不用自己調用findViewById了。只需要寫點註解就可以了。個人感覺也不是很優雅。【大部分android老鳥都知道這個】
再後來,直到遇見了kotlin的 kotlin-android-extensions 插件,我覺得這才叫優雅。
無需調用findViewById()方法,也無需註解。
一切都是這麼的美好。
三、結尾
哇~~ 大叔居然一口氣寫了這……麼長一篇技術blog,你能看到這裏,我覺得你也真的是有耐心。
本來我想多寫幾個痛點案例,但我看着這篇幅長度,我覺得還是算了。絕大部分人是不會看到結尾的。我就拋個磚吧~
謝謝你在這個浮躁的時代,能靜下心來看一箇中年油膩大叔的絮絮叨叨。大叔祝您在kotlin的世界翱翔~ 希望kotlin能給您帶來幸福感~
讚美是一種美德,點個贊 再走啊,老鐵