├── .idea
├── .name
├── copyright
│ └── profiles_settings.xml
├── encodings.xml
├── modules.xml
├── runConfigurations.xml
├── compiler.xml
├── gradle.xml
└── misc.xml
├── app
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ ├── values
│ │ │ │ ├── strings.xml
│ │ │ │ ├── colors.xml
│ │ │ │ ├── dimens.xml
│ │ │ │ └── styles.xml
│ │ │ ├── mipmap-hdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-mdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xxhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xxxhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── values-w820dp
│ │ │ │ └── dimens.xml
│ │ │ └── layout
│ │ │ │ └── activity_main.xml
│ │ ├── java
│ │ │ └── com
│ │ │ │ └── fashare
│ │ │ │ └── hellokotlin
│ │ │ │ └── MainActivity.java
│ │ ├── kotlin
│ │ │ └── com
│ │ │ │ └── fashare
│ │ │ │ └── hellokotlin
│ │ │ │ ├── feature
│ │ │ │ ├── delegate
│ │ │ │ │ └── Delegate.kt
│ │ │ │ └── extension
│ │ │ │ │ └── Extension.kt
│ │ │ │ └── sugar
│ │ │ │ ├── classes
│ │ │ │ └── SomeClasses.kt
│ │ │ │ ├── stream
│ │ │ │ └── StreamLikeJava8.kt
│ │ │ │ ├── interfaces
│ │ │ │ └── DefaultInterface.kt
│ │ │ │ ├── safe
│ │ │ │ └── NullPointerSafe.kt
│ │ │ │ └── lambda
│ │ │ │ └── LambdaTest.kt
│ │ └── AndroidManifest.xml
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── fashare
│ │ │ └── hellokotlin
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── fashare
│ │ └── hellokotlin
│ │ └── ApplicationTest.java
├── proguard-rules.pro
└── build.gradle
├── mydemo
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ ├── values
│ │ │ │ ├── strings.xml
│ │ │ │ ├── dimens.xml
│ │ │ │ ├── colors.xml
│ │ │ │ └── styles.xml
│ │ │ ├── drawable
│ │ │ │ └── ic_launcher.png
│ │ │ ├── values-w820dp
│ │ │ │ └── dimens.xml
│ │ │ └── layout
│ │ │ │ ├── activity_main.xml
│ │ │ │ └── item_story.xml
│ │ ├── kotlin
│ │ │ └── com
│ │ │ │ └── fashare
│ │ │ │ └── mydemo
│ │ │ │ ├── Api.kt
│ │ │ │ ├── data
│ │ │ │ └── HomeInfo.kt
│ │ │ │ └── MainActivity.kt
│ │ └── AndroidManifest.xml
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── fashare
│ │ │ └── mydemo
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── fashare
│ │ └── mydemo
│ │ └── ApplicationTest.java
├── proguard-rules.pro
└── build.gradle
├── settings.gradle
├── screen-record
├── mydemo.jpg
└── project.png
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── .gitignore
├── gradle.properties
├── gradlew.bat
├── gradlew
└── README.md
/.idea/.name:
--------------------------------------------------------------------------------
1 | HelloKotlin
--------------------------------------------------------------------------------
/app/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/mydemo/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':app', ':mydemo'
2 |
--------------------------------------------------------------------------------
/screen-record/mydemo.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/screen-record/mydemo.jpg
--------------------------------------------------------------------------------
/screen-record/project.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/screen-record/project.png
--------------------------------------------------------------------------------
/.idea/copyright/profiles_settings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | HelloKotlin
3 |
4 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | MyDemo
3 |
4 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/mydemo/src/main/res/drawable/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/mydemo/src/main/res/drawable/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/fashare2015/HelloKotlin/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | *.iml
2 | .gradle
3 | /local.properties
4 | /.idea/workspace.xml
5 | /.idea/libraries
6 | /.idea/*
7 | .DS_Store
8 | /build
9 | /captures
10 |
--------------------------------------------------------------------------------
/.idea/encodings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/app/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 |
6 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 |
6 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Mon Dec 28 10:00:20 PST 2015
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.10-all.zip
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/mydemo/src/test/java/com/fashare/mydemo/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.fashare.mydemo;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/app/src/test/java/com/fashare/hellokotlin/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/mydemo/src/main/kotlin/com/fashare/mydemo/Api.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.mydemo
2 |
3 | import com.fashare.mydemo.data.HomeInfo
4 | import retrofit2.http.GET
5 | import rx.Observable
6 |
7 | /**
8 | * Created by apple on 17-5-31.
9 | */
10 | interface Api {
11 | companion object{
12 | val BASE_URL: String = "http://news-at.zhihu.com/"
13 | }
14 |
15 | @GET("api/4/news/latest")
16 | fun getHomeInfo(): Observable
17 | }
--------------------------------------------------------------------------------
/app/src/main/java/com/fashare/hellokotlin/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin;
2 |
3 | import android.support.v7.app.AppCompatActivity;
4 | import android.os.Bundle;
5 |
6 | public class MainActivity extends AppCompatActivity {
7 |
8 | @Override
9 | protected void onCreate(Bundle savedInstanceState) {
10 | super.onCreate(savedInstanceState);
11 | setContentView(R.layout.activity_main);
12 |
13 |
14 | }
15 | }
16 |
--------------------------------------------------------------------------------
/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/mydemo/src/androidTest/java/com/fashare/mydemo/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package com.fashare.mydemo;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/mydemo/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/app/src/androidTest/java/com/fashare/hellokotlin/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/.idea/modules.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/feature/delegate/Delegate.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.feature.delegate
2 |
3 |
4 | /**
5 | * Created by apple on 17-5-31.
6 | *
7 | * 属性代理
8 | */
9 |
10 | val lazySum: Int by lazy {
11 | println("begin compute lazySum ...")
12 | var sum = 0
13 | for (i in 0..100)
14 | sum += i
15 | println("lazySum computed!\n")
16 | sum // 返回计算结果
17 | }
18 |
19 | fun main(args: Array) {
20 | println(lazySum)
21 | println(lazySum)
22 | }
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/sugar/classes/SomeClasses.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.sugar.classes
2 |
3 | /**
4 | * Created by apple on 17-5-31.
5 | * 类与继承
6 | */
7 | open class People(val name: String){}
8 |
9 | class Coder(name: String, val language: String = "Kotlin"): People(name){
10 |
11 | override fun toString(): String {
12 | return "name: $name, language: $language"
13 | }
14 | }
15 |
16 | fun main(args: Array) {
17 | val coder = Coder("Tom")
18 |
19 | println(coder)
20 | }
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/sugar/stream/StreamLikeJava8.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.sugar.stream
2 |
3 | /**
4 | * Created by apple on 17-5-30.
5 | *
6 | * 集合操作, 类似于 Java8 新增的 Stream
7 | */
8 | fun main(args: Array){
9 | val list = listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
10 | list.filter { it%2==0 } // 取偶数
11 | .map{ it*it } // 平方
12 | .sortedDescending() // 降序排序
13 | .take(3) // 取前 3 个
14 | .forEach { println(it) } // 遍历, 打印
15 | }
--------------------------------------------------------------------------------
/.idea/runConfigurations.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
11 |
12 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/app/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in /Users/apple/Library/Android/sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/mydemo/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in /Users/apple/Library/Android/sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/feature/extension/Extension.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.feature.extension
2 |
3 | import com.fashare.hellokotlin.sugar.lambda.myForEach
4 |
5 | /**
6 | * Created by apple on 17-5-31.
7 | *
8 | * 函数拓展, 属性拓展
9 | */
10 | fun main(args: Array) {
11 | val list = listOf("1", "2", "3", "4")
12 |
13 | // 函数拓展
14 | list.myForEach { println(it) }
15 |
16 | // 属性拓展
17 | println("last: ${list.lastItem}")
18 | }
19 |
20 | /**
21 | * 拓展 List 类, 加一个自定义的遍历方法
22 | */
23 | fun List.myForEach(doTask: (T) -> Unit){
24 | for(item in this)
25 | doTask(item)
26 | }
27 |
28 | /**
29 | * 拓展 List 类, 加一个自定义的长度属性
30 | */
31 | val List.lastItem: T
32 | get() = get(size - 1)
33 |
--------------------------------------------------------------------------------
/.idea/compiler.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
12 |
13 |
17 |
18 |
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/sugar/interfaces/DefaultInterface.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.sugar.interfaces
2 |
3 | /**
4 | * Created by apple on 17-5-30.
5 | * 接口可以有默认实现, 类似 Java8 引入的 默认接口和 default 关键字
6 | */
7 | interface A {
8 | fun foo() { println("A") } // 默认实现, 打印"A"
9 | fun bar()
10 | }
11 |
12 | interface B {
13 | fun foo() { println("B") }
14 | fun bar() { println("bar") }
15 | }
16 |
17 | class C : A {
18 | override fun bar() { println("bar") }
19 | }
20 |
21 | class D : A, B {
22 | override fun foo() {
23 | super.foo()
24 | super.foo()
25 | }
26 |
27 | override fun bar() {
28 | super.bar()
29 | }
30 | }
31 |
32 | fun main(args: Array){
33 | val d: D = D()
34 | d.foo()
35 | d.bar()
36 | }
37 |
38 |
39 |
--------------------------------------------------------------------------------
/mydemo/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
6 |
7 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
--------------------------------------------------------------------------------
/mydemo/src/main/res/layout/item_story.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
13 |
14 |
22 |
23 |
24 |
--------------------------------------------------------------------------------
/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/.idea/gradle.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
--------------------------------------------------------------------------------
/mydemo/src/main/kotlin/com/fashare/mydemo/data/HomeInfo.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.mydemo.data
2 |
3 | import com.fashare.mydemo.R
4 | import com.fashare.no_view_holder.annotation.BindImageView
5 | import com.fashare.no_view_holder.annotation.BindImageViews
6 | import com.fashare.no_view_holder.annotation.BindRecyclerView
7 | import com.fashare.no_view_holder.annotation.BindTextView
8 | import java.util.*
9 |
10 | /**
11 | * Created by apple on 17-5-31.
12 | */
13 | class HomeInfo(){
14 | @BindRecyclerView(id = R.id.rv_story, layout = R.layout.item_story)
15 | val stories: List = Collections.emptyList()
16 |
17 | inner class Story{
18 | @BindTextView(id = R.id.tv_title)
19 | val title: String = ""
20 |
21 | @BindImageViews(
22 | BindImageView(id = R.id.iv_image, placeHolder = R.drawable.ic_launcher)
23 | )
24 | val images: List = Collections.emptyList()
25 |
26 | override fun toString(): String {
27 | return title
28 | }
29 | }
30 | }
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/sugar/safe/NullPointerSafe.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.sugar.safe
2 |
3 | /**
4 | * Created by apple on 17-5-30.
5 | *
6 | * 空安全, 编译时期的空检查, 类似 @Nullable, @NonNull
7 | */
8 | class NullPointerSafe{
9 | var mNullable: String? = null
10 | var mNonNull: String = "mNonNull"
11 |
12 | fun testNull(){
13 | println("testNull: ")
14 | println(mNullable?.length)
15 | println(mNonNull.length)
16 | println()
17 | }
18 |
19 | /**
20 | * 入参严格非空, 编译时检查
21 | */
22 | fun printList(list: List){
23 | println("printList: ")
24 | list.forEach{ println(it) }
25 | println()
26 | }
27 | }
28 |
29 | fun main(args: Array){
30 | val nullPointerSafe = NullPointerSafe()
31 |
32 | nullPointerSafe.testNull()
33 |
34 | var listNonNull: List = listOf("1", "2", "3", "4")
35 | nullPointerSafe.printList(listNonNull) // 入参为非空List, 正常运行
36 |
37 | // 编译错误
38 | var listNullable: List? = listOf("1", "2", "3", "4")
39 | // nullPointerSafe.printList(listNullable) // 入参为可空List, 编译错误
40 | }
--------------------------------------------------------------------------------
/app/src/main/kotlin/com/fashare/hellokotlin/sugar/lambda/LambdaTest.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.hellokotlin.sugar.lambda
2 |
3 | /**
4 | * Created by apple on 17-5-30.
5 | *
6 | * lambda 表达式, 类似 java8 中的lambda
7 | */
8 | fun main(args: Array) {
9 | // new 一个线程
10 | // 匿名类写法
11 | val runnable1 = object : Runnable{
12 | override fun run() {
13 | println("I'm an anonymous class")
14 | }
15 | }
16 |
17 | // 函数写法, 略像js
18 | val runnable2 = fun (){
19 | println("I'm a function")
20 | }
21 |
22 | // lambda写法1
23 | val runnable3 = Runnable { ->
24 | println("I'm a Lambda")
25 | }
26 |
27 | // lambda写法2
28 | val runnable4 = { println("I'm a Lambda") }
29 | Thread(runnable4).start()
30 |
31 | // 自定义高阶函数, lambda 表达式 作为入参
32 | listOf("1", "2", "3", "4").myForEach { println(it) }
33 |
34 | // 自定义高阶函数, lambda 表达式 作为返回值
35 | // getLogger()("I'm a Closure")
36 | var logger = getLogger()
37 | logger("I'm a Closure")
38 | }
39 |
40 | /**
41 | * 接受一个 lambda 表达式, 作为遍历任务
42 | */
43 | fun List.myForEach(doTask: (T) -> Unit){
44 | for(item in this)
45 | doTask(item)
46 | }
47 |
48 | /**
49 | * 返回一个 lambda 表达式(闭包), 如: 日志输出工具 logger
50 | */
51 | fun getLogger(): (String) -> Unit{
52 | // return { println(it) }
53 | return fun (it: String){
54 | println(it)
55 | }
56 | }
--------------------------------------------------------------------------------
/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion rootProject.ext.compileSdkVersion as Integer
5 | buildToolsVersion rootProject.ext.buildToolsVersion as String
6 |
7 | defaultConfig {
8 | applicationId "com.fashare.hellokotlin"
9 | minSdkVersion rootProject.ext.minSdkVersion as Integer
10 | targetSdkVersion rootProject.ext.targetSdkVersion as Integer
11 | versionCode rootProject.ext.versionCode as Integer
12 | versionName rootProject.ext.versionName as String
13 | }
14 |
15 | buildTypes {
16 | release {
17 | minifyEnabled false
18 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
19 | }
20 | }
21 | sourceSets {
22 | main.java.srcDirs += 'src/main/kotlin'
23 | }
24 | }
25 |
26 | dependencies {
27 | compile fileTree(dir: 'libs', include: ['*.jar'])
28 | testCompile 'junit:junit:4.12'
29 | compile "com.android.support:appcompat-v7:$supportLibsVersion"
30 | }
31 |
32 | // kotlin
33 | buildscript {
34 | repositories {
35 | jcenter()
36 | }
37 | dependencies {
38 | classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion"
39 | }
40 | }
41 |
42 | apply plugin: 'kotlin-android'
43 |
44 | android {
45 | sourceSets {
46 | main.java.srcDirs += 'src/main/kotlin'
47 | }
48 | }
49 |
50 | dependencies {
51 | compile "org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion"
52 | }
53 | repositories {
54 | mavenCentral()
55 | }
56 |
57 |
58 |
--------------------------------------------------------------------------------
/mydemo/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 | apply plugin: 'android-apt'
3 |
4 | android {
5 | compileSdkVersion rootProject.ext.compileSdkVersion as Integer
6 | buildToolsVersion rootProject.ext.buildToolsVersion as String
7 |
8 | defaultConfig {
9 | minSdkVersion rootProject.ext.minSdkVersion as Integer
10 | targetSdkVersion rootProject.ext.targetSdkVersion as Integer
11 | versionCode rootProject.ext.versionCode as Integer
12 | versionName rootProject.ext.versionName as String
13 | }
14 |
15 | buildTypes {
16 | release {
17 | minifyEnabled false
18 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
19 | }
20 | }
21 | sourceSets {
22 | main.java.srcDirs += 'src/main/kotlin'
23 | }
24 | }
25 |
26 | dependencies {
27 | compile fileTree(dir: 'libs', include: ['*.jar'])
28 | testCompile 'junit:junit:4.12'
29 | compile "com.android.support:appcompat-v7:$supportLibsVersion"
30 | compile "com.android.support:recyclerview-v7:$supportLibsVersion"
31 |
32 | // Butter Knife
33 | compile 'com.jakewharton:butterknife:8.2.1'
34 | apt 'com.jakewharton:butterknife-compiler:8.2.1'
35 |
36 | // glide
37 | compile 'com.github.bumptech.glide:glide:3.7.0'
38 | compile 'jp.wasabeef:glide-transformations:2.0.1'
39 |
40 | // Rxjava
41 | compile 'io.reactivex:rxjava:1.1.0'
42 | compile 'io.reactivex:rxandroid:1.1.0'
43 |
44 | // Retrofit
45 | compile 'com.squareup.retrofit2:retrofit:2.0.0-beta4'
46 | compile 'com.squareup.retrofit2:adapter-rxjava:2.0.0-beta4'
47 | compile 'com.squareup.retrofit2:converter-gson:2.0.0-beta4'
48 |
49 | // Gson
50 | compile 'com.google.code.gson:gson:2.6.2'
51 |
52 | compile 'com.github.fashare2015:NoViewHolder:1.0.1'
53 | }
54 |
55 | // kotlin
56 | buildscript {
57 | repositories {
58 | jcenter()
59 | }
60 | dependencies {
61 | classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion"
62 | }
63 | }
64 |
65 | apply plugin: 'kotlin-android'
66 |
67 | android {
68 | sourceSets {
69 | main.java.srcDirs += 'src/main/kotlin'
70 | }
71 | }
72 |
73 | dependencies {
74 | compile "org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion"
75 | }
76 |
--------------------------------------------------------------------------------
/mydemo/src/main/kotlin/com/fashare/mydemo/MainActivity.kt:
--------------------------------------------------------------------------------
1 | package com.fashare.mydemo
2 |
3 | import android.app.Activity
4 | import android.os.Bundle
5 | import android.support.v4.widget.SwipeRefreshLayout
6 | import android.support.v7.app.AppCompatActivity
7 | import android.widget.Toast
8 | import butterknife.BindView
9 | import butterknife.ButterKnife
10 | import com.fashare.mydemo.data.HomeInfo
11 | import com.fashare.no_view_holder.NoViewHolder
12 | import com.fashare.no_view_holder.annotation.click.BindItemClick
13 | import com.fashare.no_view_holder.widget.NoOnItemClickListener
14 | import retrofit2.Retrofit
15 | import retrofit2.adapter.rxjava.RxJavaCallAdapterFactory
16 | import retrofit2.converter.gson.GsonConverterFactory
17 | import rx.android.schedulers.AndroidSchedulers
18 | import rx.schedulers.Schedulers
19 |
20 | class MainActivity : AppCompatActivity() {
21 |
22 | // retrofit 网络请求
23 | val retrofit = Retrofit.Builder().baseUrl(Api.BASE_URL)
24 | .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
25 | .addConverterFactory(GsonConverterFactory.create())
26 | .build()
27 |
28 | // NoViewHolder! 所有 view 的容器
29 | var mNoViewHolder: NoViewHolder? = null
30 |
31 | // 下拉刷新
32 | @BindView(R.id.srl_refresh)
33 | var mSrlRefresh: SwipeRefreshLayout? = null
34 |
35 | // 点击事件
36 | @BindItemClick(id = R.id.rv_story)
37 | internal var clickMeiZhi = NoOnItemClickListener { view, data, pos -> toast("click: $pos, $data") }
38 |
39 | override fun onCreate(savedInstanceState: Bundle?) {
40 | super.onCreate(savedInstanceState)
41 | setContentView(R.layout.activity_main)
42 |
43 | ButterKnife.bind(this)
44 |
45 | mNoViewHolder = NoViewHolder.Builder(this)
46 | .initView(HomeInfo())
47 | .build()
48 |
49 | initView()
50 | loadData()
51 | }
52 |
53 | private fun initView() {
54 | mSrlRefresh?.setOnRefreshListener { loadData() }
55 | }
56 |
57 | private fun loadData() {
58 | mSrlRefresh?.setRefreshing(true)
59 |
60 | retrofit.create(Api::class.java)
61 | .getHomeInfo()
62 | .subscribeOn(Schedulers.io())
63 | .observeOn(AndroidSchedulers.mainThread())
64 | .subscribe({
65 | mNoViewHolder?.notifyDataSetChanged(it) // mHomeInfo 发生变化, 通知 UI 及时刷新
66 | }, {
67 | toast("服务器跑路啦~")
68 | }, {
69 | mSrlRefresh?.setRefreshing(false)
70 | })
71 | }
72 | }
73 |
74 | fun Activity.toast(msg: String) {
75 | Toast.makeText(this, msg, Toast.LENGTH_SHORT).show()
76 | }
77 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/.idea/misc.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 | 1.8
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # Attempt to set APP_HOME
46 | # Resolve links: $0 may be a link
47 | PRG="$0"
48 | # Need this for relative symlinks.
49 | while [ -h "$PRG" ] ; do
50 | ls=`ls -ld "$PRG"`
51 | link=`expr "$ls" : '.*-> \(.*\)$'`
52 | if expr "$link" : '/.*' > /dev/null; then
53 | PRG="$link"
54 | else
55 | PRG=`dirname "$PRG"`"/$link"
56 | fi
57 | done
58 | SAVED="`pwd`"
59 | cd "`dirname \"$PRG\"`/" >/dev/null
60 | APP_HOME="`pwd -P`"
61 | cd "$SAVED" >/dev/null
62 |
63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
64 |
65 | # Determine the Java command to use to start the JVM.
66 | if [ -n "$JAVA_HOME" ] ; then
67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
68 | # IBM's JDK on AIX uses strange locations for the executables
69 | JAVACMD="$JAVA_HOME/jre/sh/java"
70 | else
71 | JAVACMD="$JAVA_HOME/bin/java"
72 | fi
73 | if [ ! -x "$JAVACMD" ] ; then
74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
75 |
76 | Please set the JAVA_HOME variable in your environment to match the
77 | location of your Java installation."
78 | fi
79 | else
80 | JAVACMD="java"
81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
82 |
83 | Please set the JAVA_HOME variable in your environment to match the
84 | location of your Java installation."
85 | fi
86 |
87 | # Increase the maximum file descriptors if we can.
88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
89 | MAX_FD_LIMIT=`ulimit -H -n`
90 | if [ $? -eq 0 ] ; then
91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
92 | MAX_FD="$MAX_FD_LIMIT"
93 | fi
94 | ulimit -n $MAX_FD
95 | if [ $? -ne 0 ] ; then
96 | warn "Could not set maximum file descriptor limit: $MAX_FD"
97 | fi
98 | else
99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
100 | fi
101 | fi
102 |
103 | # For Darwin, add options to specify how the application appears in the dock
104 | if $darwin; then
105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
106 | fi
107 |
108 | # For Cygwin, switch paths to Windows format before running java
109 | if $cygwin ; then
110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
112 | JAVACMD=`cygpath --unix "$JAVACMD"`
113 |
114 | # We build the pattern for arguments to be converted via cygpath
115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
116 | SEP=""
117 | for dir in $ROOTDIRSRAW ; do
118 | ROOTDIRS="$ROOTDIRS$SEP$dir"
119 | SEP="|"
120 | done
121 | OURCYGPATTERN="(^($ROOTDIRS))"
122 | # Add a user-defined pattern to the cygpath arguments
123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
125 | fi
126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
127 | i=0
128 | for arg in "$@" ; do
129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
131 |
132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
134 | else
135 | eval `echo args$i`="\"$arg\""
136 | fi
137 | i=$((i+1))
138 | done
139 | case $i in
140 | (0) set -- ;;
141 | (1) set -- "$args0" ;;
142 | (2) set -- "$args0" "$args1" ;;
143 | (3) set -- "$args0" "$args1" "$args2" ;;
144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
150 | esac
151 | fi
152 |
153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
154 | function splitJvmOpts() {
155 | JVM_OPTS=("$@")
156 | }
157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
159 |
160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
161 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Kotlin的来历
2 | Kotlin的作者是大名鼎鼎的Jetbrains公司。它有一系列耳熟能详的产品,诸如Android兄弟天天用的Android Studio, IntelliJ IDEA, 还有前端的WebStorm, PhpStorm。
3 | - 2011年7月,JetBrains推出Kotlin项目。
4 | - 2012年2月,JetBrains以Apache 2许可证开源此项目。
5 | - 2016年2月15日,Kotlin v1.0(第一个官方稳定版本)发布。
6 | - 2017 Google I/O 大会,Kotlin 转正。
7 |
8 | 我个人大概是在16年年底接触到这门语言,当时由于语法上的诸多相似性,它被誉为“Android中的Swift”。说来也巧,当时正在研究ButterKnife这个框架,偶然发现它还有Kotlin版本——[kotterknife](https://github.com/JakeWharton/kotterknife),于是就打开了新世界的大门。
9 |
10 | # 亮点
11 | 个人感觉,Kotlin最大的亮点便是可以和Java**无缝衔接**(虽然它Jvm上的兄弟Groovy,Scala也能做到)。这意味着,我们不但拥有原来java中所有的资源,还能体验到js般的编程体验(雾)。
12 |
13 | 另外,它的很多特性都分别对应java中的一个坑, 本文会从**语法糖**和**新特性**两个方面来介绍:
14 | - 语法糖
15 | - 类的简化,隐式getter()、setter()
16 | - 接口的默认实现
17 | - lambda与高阶函数
18 | - 空指针安全,编译时期的空指针检查
19 | - 流式集合操作 map(), forEach()
20 | - 新特性
21 | - 函数拓展、属性拓展
22 | - 属性代理
23 |
24 | # 简要的环境配置
25 | 根目录/build.gradle:
26 | ```gradle
27 | // external 全局变量
28 | ext{
29 | kotlinVersion = "1.0.0-rc-1036"
30 | }
31 | ```
32 | app/build.gradle
33 | ```gradle
34 | // kotlin
35 | buildscript {
36 | repositories {
37 | jcenter()
38 | }
39 | dependencies {
40 | classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion"
41 | }
42 | }
43 |
44 | apply plugin: 'kotlin-android'
45 |
46 | android {
47 | // 建立一个与'src/main/java'同级的kotlin工作目录
48 | sourceSets {
49 | main.java.srcDirs += 'src/main/kotlin'
50 | }
51 | }
52 |
53 | dependencies {
54 | compile "org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion"
55 | }
56 | ```
57 | 
58 |
59 | # 官方资料
60 | [Kotlin 官方参考文档 中文版](https://hltj.gitbooks.io/kotlin-reference-chinese/content/)
61 |
62 | # 详细介绍
63 | >以下每个例子附带一份源码,在(根目录/app)下
64 |
65 | ## 1. 语法糖
66 | ### 1.1 类的简化
67 | #### 1.1.1 类的定义
68 | 一切从一个简单的JavaBean说起吧。
69 | ```kotlin
70 | // java bean
71 | class People{
72 | private String name;
73 |
74 | public People(String name){
75 | this.name = name;
76 | }
77 |
78 | public void setName(String name){
79 | this.name = name;
80 | }
81 |
82 | public String getName(){
83 | return this.name;
84 | }
85 | }
86 |
87 | // kotlin bean
88 | class People(val name: String){}
89 | ```
90 | 从我学java以来就接触了javabean这个概念,我也不懂为啥非要getter()、setter()这种东西,难道为了kpi? 可以看到同样的数据实体,在kotlin中简化至1行。变量定义和构造函数合并,同时提供隐式的getter()、setter()。
91 |
92 | #### 1.1.2 继承
93 | 下面是一段简单的继承关系:
94 | ```kotlin
95 | // 所有类默认final,要显式指定为open才可被继承
96 | open class People(val name: String){}
97 |
98 | // Code: People 表示继承关系
99 | class Coder(name: String, val language: String = "Kotlin"): People(name){
100 |
101 | override fun toString(): String {
102 | return "name: $name, language: $language"
103 | }
104 | }
105 |
106 | // 可执行的main()方法
107 | fun main(args: Array) {
108 | val coder = Coder("Tom")
109 |
110 | println(coder)
111 | }
112 |
113 | // 输出: name: Tom, language: Kotlin
114 | ```
115 |
116 | #### 1.1.3 数据对象 data class
117 | 在看Effective Java的时候,总是对如何正确重写hashCode(),equals(),clone()感到棘手。然而他们又是正确使用Collection的关键所在(无论是HashMap、ArrayList)。
118 | Kotlin专门提供了一个`data class`,来自动生成hashCode(),equals(),clone()。
119 | ```kotlin
120 | data class People(val name: String){}
121 | ```
122 |
123 |
124 | ### 1.2 接口的默认实现
125 | 顾名思义,它便是指接口可以和抽象类一样,有方法体的默认实现。
126 | 我把它归结在语法糖里,是因为java8中早已有了一模一样的东西,对应的关键字叫`default`。
127 |
128 | 看起来抽象类好像可以退休了,实则不然。接口依然只是接口,不能拥有属性,只是在方法定义上更加灵活了。
129 |
130 | ```kotlin
131 | interface A {
132 | fun foo() { println("A") } // 默认实现, 打印"A"
133 | fun bar()
134 | }
135 |
136 | interface B {
137 | fun foo() { println("B") }
138 | fun bar() { println("bar") }
139 | }
140 |
141 | // 多继承时,显式指定 super.foo() 以去冲突
142 | class D : A, B {
143 | override fun foo() {
144 | super.foo()
145 | super.foo()
146 | }
147 |
148 | override fun bar() {
149 | super.bar()
150 | }
151 | }
152 | ```
153 |
154 |
155 | ### 1.3 lambda与高阶函数
156 | #### 1.3.1 lambda
157 | lambda也不是什么新鲜玩意,在gradle、js之类的其他语言中早就玩烂了。java8中也有,不过官方迟迟不上咱也没办法。
158 | lambda本身是一个函数片段,作为一等公民,它可以作为高阶函数的参数或返回值。个人而言,我喜欢以匿名类来理解它。
159 | ```kotlin
160 | // new 一个线程
161 | // 匿名类写法
162 | val runnable1 = object : Runnable{
163 | override fun run() {
164 | println("I'm an anonymous class")
165 | }
166 | }
167 |
168 | // 函数写法, 略像js
169 | val runnable2 = fun (){
170 | println("I'm a function")
171 | }
172 |
173 | // lambda写法1
174 | val runnable3 = Runnable { ->
175 | println("I'm a Lambda")
176 | }
177 |
178 | // lambda写法2
179 | val runnable4 = { println("I'm a Lambda") }
180 | Thread(runnable4).start()
181 | ```
182 | 平常习惯使用`lambda写法2`,省略匿名类的名字Runnable,以及无参数时省略箭头->。不过js那种函数写法也挺不错的。
183 |
184 | #### 1.3.2 高阶函数
185 | lambda本身作为一等公民,它是有类型的。如上例的runnable4的类型为`()->Unit`。再比如下面这个加法表达式sum的类型为`(Int, Int) -> Int`。
186 | ```kotlin
187 | val sum: (Int, Int) -> Int = { x, y -> x+y }
188 | ```
189 | 一个变量有类型是再自然不过的事。而高阶函数的入参与返回值既然是lambda,那其类型奇怪一点也很正常。
190 | ```kotlin
191 | fun main(args: Array) {
192 | // 自定义高阶函数, lambda 表达式 作为入参
193 | listOf("1", "2", "3", "4").myForEach { println(it) }
194 |
195 | // 自定义高阶函数, lambda 表达式 作为返回值
196 | // getLogger()("I'm a Closure")
197 | var logger = getLogger()
198 | logger("I'm a Closure")
199 | }
200 |
201 | /**
202 | * 接受一个 lambda 表达式, 作为遍历任务
203 | */
204 | fun List.myForEach(doTask: (T) -> Unit){
205 | for(item in this)
206 | doTask(item)
207 | }
208 |
209 | /**
210 | * 返回一个 lambda 表达式(闭包), 如: 日志输出工具 logger
211 | */
212 | fun getLogger(): (String) -> Unit{
213 | // return { println(it) }
214 | return fun (it: String){
215 | println(it)
216 | }
217 | }
218 | ```
219 | PS: 看到getLogger()这种用法,你大概意识到可以像js那样写闭包了。
220 |
221 | ### 1.4 空指针安全
222 | 你也许会想空指针不是很简单的东西吗,加个if(xxx != null)判断就好了。事实上,空指针绝对是出现频率最高,最讨厌的问题。而**空安全**则是kotlin主打的一个特性之一。在java8中,我们可以借助Optional勉强做到这一点。
223 |
224 | 我们来看看是咋回事。
225 | ```kotlin
226 | var mNullable: String? = null
227 | var mNonNull: String = "mNonNull"
228 |
229 | fun testNull(){
230 | println("testNull: ")
231 | println(mNullable?.length)
232 | println(mNonNull.length)
233 | println()
234 | }
235 |
236 | // 输出:
237 | testNull:
238 | null
239 | 8
240 | ```
241 | kotlin定义变量时区分两种类型:
242 | - var mNullable: Any? = null 可空
243 | - var mNonNull: Any = XXX 非空
244 |
245 | 1.对一个 mNullable 你可以像普通的java类一样使用:
246 |
247 | ```kotlin
248 | // java 风格,判空
249 | if(mNullable != null)
250 | mNullable.length
251 |
252 | // kotlin 语法糖,判空(推荐)
253 | mNullable?.length
254 | ```
255 | 2.对一个 mNonNull 则有严格的限制:
256 |
257 | ```kotlin
258 | // 不必判空,因为必然非空
259 | mNonNull.length
260 |
261 | // 编译错误(试图给非空值赋予null)
262 | mNonNull = null
263 | // 编译错误(试图给非空值赋予可空值)
264 | mNonNull = mNullable
265 | ```
266 | 我们可以体会到mNonNull天然的好处,永不会空指针。而mNullable加上?判断,也可以达到同样的效果。
267 |
268 |
269 | ### 1.5 流式集合操作 map(), forEach()
270 | 这个我不知道怎么称呼,姑且叫流式集合操作符把。算是很普遍了,任何语言里都有,然而不支持函数式的话,写起来比较臃肿。
271 | 如下例子,一些操作符的衔接,使得操作逻辑十分清晰,之后需求变动,比如降序改为升序,也只需改动`.sortedDescending()`一行,十分灵活。
272 | ```kotlin
273 | fun main(args: Array){
274 | val list = listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
275 | list.filter { it%2==0 } // 取偶数
276 | .map{ it*it } // 平方
277 | .sortedDescending() // 降序排序
278 | .take(3) // 取前 3 个
279 | .forEach { println(it) } // 遍历, 打印
280 | }
281 |
282 | // 输出:
283 | 100
284 | 64
285 | 36
286 | ```
287 |
288 |
289 | ## 2. 新特性
290 | ### 2.1 拓展
291 | 拓展这个东西,貌似是以装饰者模式来做的。它的效果是在不改源码的基础上,添加功能。比如我们要在Activity上加一个toast(),完全不用卸载基类里。这样简化了很多工作,尤其是对一些已打成jar包的类。
292 | ```kotlin
293 | fun Activity.toast(msg: String) {
294 | Toast.makeText(this, msg, Toast.LENGTH_SHORT).show()
295 | }
296 | ```
297 | 别的例子:
298 | ```kotlin
299 | /**
300 | * Created by apple on 17-5-31.
301 | *
302 | * 函数拓展, 属性拓展
303 | */
304 | fun main(args: Array) {
305 | val list = listOf("1", "2", "3", "4")
306 |
307 | // 函数拓展
308 | list.myForEach { println(it) }
309 |
310 | // 属性拓展
311 | println("last: ${list.lastItem}")
312 | }
313 |
314 | /**
315 | * 拓展 List 类, 加一个自定义的遍历方法
316 | */
317 | fun List.myForEach(doTask: (T) -> Unit){
318 | for(item in this)
319 | doTask(item)
320 | }
321 |
322 | /**
323 | * 拓展 List 类, 加一个自定义的长度属性
324 | */
325 | val List.lastItem: T
326 | get() = get(size - 1)
327 |
328 | // 输出:
329 | 1
330 | 2
331 | 3
332 | 4
333 | last: 4
334 | ```
335 |
336 |
337 | ### 2.2 属性代理
338 | 这个东西干嘛用呢?它把属性的get()、set()代理给了一个类,以便可以在get()和set()时做一些额外的操作。如:
339 | - 懒加载
340 | - 观察者(属性变化时,自动发出通知)
341 | - 属性非空判断
342 | - ...
343 |
344 | 以懒加载为例,lazySum可能需要复杂的运算,我们把它代理给`lazy`。
345 | 可以看到,只有第一次加载进行了计算,之后都是直接取值,提高了效率。
346 | ```kotlin
347 | val lazySum: Int by lazy {
348 | println("begin compute lazySum ...")
349 | var sum = 0
350 | for (i in 0..100)
351 | sum += i
352 | println("lazySum computed!\n")
353 | sum // 返回计算结果
354 | }
355 |
356 | fun main(args: Array) {
357 | println(lazySum)
358 | println(lazySum)
359 | }
360 |
361 | // 输出:
362 | begin compute lazySum ...
363 | lazySum computed!
364 |
365 | 5050
366 | 5050
367 | ```
368 | 另外,我们可以自定义属性代理,之前提到的[kotterknife](https://github.com/JakeWharton/kotterknife),便是一个很好的例子(实现了bindview):
369 | ```kotlin
370 | public class PersonView(context: Context, attrs: AttributeSet?) : LinearLayout(context, attrs) {
371 | val firstName: TextView by bindView(R.id.first_name)
372 | val lastName: TextView by bindView(R.id.last_name)
373 |
374 | // Optional binding.
375 | val details: TextView? by bindOptionalView(R.id.details)
376 | }
377 | ```
378 |
379 |
380 | # Demo实战
381 | 用 kotlin 结合一些流行第三方库 + 知乎日报api 做了一个简易的demo,在模块(根目录/mydemo)中:
382 | - Butter Knife
383 | - Retrofit
384 | - Gson
385 | - Rxjava
386 | - Glide
387 | - 个人库 NoViewHolder
388 |
389 | 
390 |
391 |
392 |
393 | # 感谢
394 | [Kotlin 官方参考文档 中文版](https://hltj.gitbooks.io/kotlin-reference-chinese/content/)
395 |
396 | [为什么说Kotlin值得一试](http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2016/0226/4000.html)
397 |
398 |
--------------------------------------------------------------------------------