使用Kotlin从零开始写一个现代Android 项目-Part1

<TextView

android:id=“@+id/repository_name”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:textSize=“20sp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintHorizontal_bias=“0.0”

app:layout_constraintLeft_toLeftOf=“parent”

app:layout_constraintRight_toRightOf=“parent”

app:layout_constraintTop_toTopOf=“parent”

app:layout_constraintVertical_bias=“0.083”

tools:text=“Modern Android app” />

<TextView

android:id=“@+id/repository_has_issues”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

android:text=“@string/has_issues”

android:textStyle=“bold”

app:layout_constraintBottom_toBottomOf=“@+id/repository_name”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintHorizontal_bias=“1.0”

app:layout_constraintStart_toEndOf=“@+id/repository_name”

app:layout_constraintTop_toTopOf=“@+id/repository_name”

app:layout_constraintVertical_bias=“1.0” />

<TextView

android:id=“@+id/repository_owner”

android:layout_width=“0dp”

android:layout_height=“wrap_content”

android:layout_marginBottom=“8dp”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toBottomOf=“@+id/repository_name”

app:layout_constraintVertical_bias=“0.0”

tools:text=“Mladen Rakonjac” />

<TextView

android:id=“@+id/number_of_starts”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginBottom=“8dp”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintHorizontal_bias=“1”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toBottomOf=“@+id/repository_owner”

app:layout_constraintVertical_bias=“0.0”

tools:text=“0 stars” />

</android.support.constraint.ConstraintLayout>

不要被tools:text搞迷惑了,它的作用仅仅是让我们可以预览我们的布局。

我们可以注意到,我们的布局是扁平的,没有任何嵌套,你应该尽量少的使用布局嵌套,因为它会影响我们的性能。ConstraintLayout也可以在不同的屏幕尺寸下正常工作。

我有种预感,很快就能达到我们想要的布局效果了。

上面只是一些关于ConstraintLayout的少部分介绍,你也可以看一下关于ConstraintLayout使用的google code lab: https://codelabs.developers.google.com/codelabs/constraint-layout/index.html?index=…%2F…%2Findex#0

4. Data binding library

当我听到Data binding 库的时候,我的第一反应是:Butterknife已经很好了,再加上,我现在使用一个插件来从xml中获取View,我为啥要改变,来使用Data binding呢?但当我对Data binding有了更多的了解之后,我的它的感觉就像我第一次见到Butterknife一样,无法自拔。

Butterknife能帮我们做啥?

ButterKnife帮助我们摆脱无聊的findViewById。因此,如果您有5个视图,而没有Butterknife,则你有5 + 5行代码来绑定您的视图。使用ButterKnife,您只有我行代码就搞定。就是这样。

Butterknife的缺点是什么?

Butterknife仍然没有解决代码可维护问题,使用ButterKnife时,我经常发现自己遇到运行时异常,这是因为我删除了xml中的视图,而没有删除Activity/Fragment类中的绑定代码。另外,如果要在xml中添加视图,则必须再次进行绑定。真的很不好维护。你将浪费大量时间来维护View绑定。

那与之相比,Data Binding 怎么样呢?

有很多好处,使用Data Binding,你可以只用一行代码就搞定View的绑定,让我们看看它是如何工作的,首先,先将Data Binding 添加到项目:

// at the top of file

apply plugin: ‘kotlin-kapt’

android {

//other things that we already used

dataBinding.enabled = true

}

dependencies {

//other dependencies that we used

kapt “com.android.databinding:compiler:3.0.0-beta1”

}

请注意,数据绑定编译器的版本与项目build.gradle文件中的gradle版本相同:

classpath ‘com.android.tools.build:gradle:3.0.0-beta1’

然后,点击Sync Now,打开activity_main.xml,将Constraint Layout用layout标签包裹

<?xml version="1.0" encoding="utf-8"?>

<layout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:app=“http://schemas.android.com/apk/res-auto”

xmlns:tools=“http://schemas.android.com/tools”>

<android.support.constraint.ConstraintLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

tools:context=“me.mladenrakonjac.modernandroidapp.MainActivity”>

<TextView

android:id=“@+id/repository_name”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:textSize=“20sp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintHorizontal_bias=“0.0”

app:layout_constraintLeft_toLeftOf=“parent”

app:layout_constraintRight_toRightOf=“parent”

app:layout_constraintTop_toTopOf=“parent”

app:layout_constraintVertical_bias=“0.083”

tools:text=“Modern Android app” />

<TextView

android:id=“@+id/repository_has_issues”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

android:text=“@string/has_issues”

android:textStyle=“bold”

app:layout_constraintBottom_toBottomOf=“@+id/repository_name”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintHorizontal_bias=“1.0”

app:layout_constraintStart_toEndOf=“@+id/repository_name”

app:layout_constraintTop_toTopOf=“@+id/repository_name”

app:layout_constraintVertical_bias=“1.0” />

<TextView

android:id=“@+id/repository_owner”

android:layout_width=“0dp”

android:layout_height=“wrap_content”

android:layout_marginBottom=“8dp”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toBottomOf=“@+id/repository_name”

app:layout_constraintVertical_bias=“0.0”

tools:text=“Mladen Rakonjac” />

<TextView

android:id=“@+id/number_of_starts”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginBottom=“8dp”

android:layout_marginEnd=“16dp”

android:layout_marginStart=“16dp”

android:layout_marginTop=“8dp”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintHorizontal_bias=“1”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toBottomOf=“@+id/repository_owner”

app:layout_constraintVertical_bias=“0.0”

tools:text=“0 stars” />

</android.support.constraint.ConstraintLayout>

注意,你需要将所有的xml移动到layout 标签下面,然后点击Build图标或者使用快捷键Cmd + F9,我们需要构建项目来使Data Binding库为我们生成ActivityMainBinding类,后面在MainActivity中将用到它。

如果没有重新编译项目,你是看不到ActivityMainBinding的,因为它在编译时生成。

我们还没有完成绑定,我们只是定义了一个非空的 ActivityMainBinding 类型的变量。你会注意到我没有把? 放在 ActivityMainBinding 的后面,而且也没有初始化它。这怎么可能呢?lateinit 关键字允许我们使用非空的延迟被初始化的变量。和 ButterKnife 类似,在我们的布局准备完成后,初始化绑定需要在 onCreate 方法中进行。此外,你不应该在 onCreate 方法中声明绑定,因为你很有可能在 onCreate 方法外使用它。我们的 binding 不能为空,所以这就是我们使用 lateinit 的原因。使用 lateinit 修饰,我们不需要在每次访问它的时候检查 binding 变量是否为空。

我们初始化binding变量,你需要替换:

setContentView(R.layout.activity_main)

为:

binding = DataBindingUtil.setContentView(this, R.layout.activity_main)

就是这样,你成功的绑定了所有View,现在你可以访问它并且做一些更改,例如,我们将仓库名字改为Modern Android Medium Article:

binding.repositoryName.text = “Modern Android Medium Article”

如你所见,现在我们可以通过bingding变量来访问main_activity.xml的所有View了(前提是它们有id),这就是Data Binding 比ButterKnife 好用的原因。

kotlin的 Getters 和 setters

大概,你已经注意到了,我们没有像Java那样使用.setText(),我想在这里暂停一下,以说明与Java相比,Kotlin中的getter和setter方法如何工作的。

首先,你需要知道,我们为什么要使用getters和setters,我们用它来隐藏类中的变量,仅允许使用方法来访问这些变量,这样我们就可以向用户隐藏类中的细节,并禁止用户直接修改我们的类。假设我们用 Java 写了一个 Square 类:

public class Square {

private int a;

Square(){

a = 1;

}

public void setA(int a){

this.a = Math.abs(a);

}

public int getA(){

return this.a;

}

}

使用setA()方法,我们禁止了用户向Square类的a变量设置一个负数,因为正方形的边长一定是正数,要使用这种方法,我们必须将其设为私有,因此不能直接设置它。这也意味着我们不能直接获得a,需要给它定一个get方法来返回a,如果有10个变量,那么我们就得定义10个相似的get方法,写这样无聊的样板代码,通常会影响我们的心情。

Kotling使我们的开发人员更轻松了。如果你调用下面的代码:

var side: Int = square.a

这并不意味着你是在直接访问a变量,它和Java中调用getA()是相同的

int side = square.getA();

因为Kotlin自动生成默认的getter和setter。在Kotlin中,只有当您有特殊的setter或getter时,才应指定它。否则,Kotlin会为您自动生成:

var a = 1

set(value) { field = Math.abs(value) }

field ? 这又是个什么东西?为了更清楚明白,请看下面代码:

var a = 1

set(value) { a = Math.abs(value) }

这表明你在调用set方法中的set(value){},因为Kotlin的世界中,没有直接访问属性,这就会造成无限递归,当你调用a = something,会自动调用set方法。使用filed就能避免无限递归,我希望这能让你明白为什么要用filed关键字,并且了解getters和setters是如何工作的。

回到代码中继续,我将向你介绍Kotlin语言的另一个重要功能:apply函数:

class MainActivity : AppCompatActivity() {

lateinit var binding: ActivityMainBinding

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

binding = DataBindingUtil.setContentView(this, R.layout.activity_main)

binding.apply {

repositoryName.text = “Medium Android Repository Article”

repositoryOwner.text = “Mladen Rakonjac”

numberOfStarts.text = “1000 stars”

}

}

}

apply 允许你在一个实例上调用多个方法,我们仍然还没有完成数据绑定,还有更棒的事儿,让我们为仓库定义一个UI模型(这个是github仓库的数据模型Repository,它持有要展示的数据,请不要和Repository模式的中的Repository搞混淆了哈),要创建一个Kotlin class,点击New -> Kotlin File/Class :

class Repository(var repositoryName: String?,var repositoryOwner: String?,var numberOfStars: Int? ,var hasIssues: Boolean = false)

在Kotlin中,主构造函数是类头的一部分,如果你不想定义次构造函数,那就是这样了,数据类到此就完成了,构造函数没有参数分配给字段,没有setters和getters,整个类就一行代码。

回到MainActivity.kt,为Repository创建一个实例。

var repository = Repository(“Medium Android Repository Article”,

“Mladen Rakonjac”, 1000, true)

你应该注意到了,创建类实例,没有用new

现在,我们在activity_main.xml中添加data标签。

<variable

name=“repository”

type=“me.mladenrakonjac.modernandroidapp.uimodels.Repository”

/>

我们可以在布局中访问存储的变量repository,例如,我们可以如下使用id是repository_name的TextView,如下:

`android:text=“@{repository.repositoryName}”`

repository_name文本视图将显示从repository变量的属性repositoryName获取的文本。剩下的唯一事情就是将repository变量从xml绑定到MainActivity.kt中的repository。

点击Build使DataBinding 为我们生成类,然后在MainActivity中添加两行代码:

binding.repository = repository

binding.executePendingBindings()

如果你运行APP,你会看到TextView上显示的是:“Medium Android Repository Article”,非常棒的功能,是吧?

但是,如果我们像下面这样改一下呢?

Handler().postDelayed({repository.repositoryName=“New Name”}, 2000)

新的文本将会在2000ms后显示吗?不会的,你必须重新设置一次repository,像这样:

Handler().postDelayed({repository.repositoryName=“New Name”

binding.repository = repository

binding.executePendingBindings()}, 2000)

最后

文章不易,如果大家喜欢这篇文章,或者对你有帮助希望大家多多点赞转发关注哦。文章会持续更新的。绝对干货!!!

  • Android进阶学习全套手册
    关于实战,我想每一个做开发的都有话要说,对于小白而言,缺乏实战经验是通病,那么除了在实际工作过程当中,我们如何去更了解实战方面的内容呢?实际上,我们很有必要去看一些实战相关的电子书。目前,我手头上整理到的电子书还算比较全面,HTTP、自定义view、c++、MVP、Android源码设计模式、Android开发艺术探索、Java并发编程的艺术、Android基于Glide的二次封装、Android内存优化——常见内存泄露及优化方案、.Java编程思想 (第4版)等高级技术都囊括其中。

  • Android高级架构师进阶知识体系图
    关于视频这块,我也是自己搜集了一些,都按照Android学习路线做了一个分类。按照Android学习路线一共有八个模块,其中视频都有对应,就是为了帮助大家系统的学习。接下来看一下导图和对应系统视频吧!!!

  • Android对标阿里P7学习视频

  • BATJ大厂Android高频面试题
    这个题库内容是比较多的,除了一些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

模式、Android开发艺术探索、Java并发编程的艺术、Android基于Glide的二次封装、Android内存优化——常见内存泄露及优化方案、.Java编程思想 (第4版)等高级技术都囊括其中。

[外链图片转存中…(img-DqKYRDo1-1714563952525)]

  • Android高级架构师进阶知识体系图
    关于视频这块,我也是自己搜集了一些,都按照Android学习路线做了一个分类。按照Android学习路线一共有八个模块,其中视频都有对应,就是为了帮助大家系统的学习。接下来看一下导图和对应系统视频吧!!!
    [外链图片转存中…(img-lomwmKVR-1714563952526)]

  • Android对标阿里P7学习视频

[外链图片转存中…(img-fJO0YJrX-1714563952526)]

  • BATJ大厂Android高频面试题
    这个题库内容是比较多的,除了一些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等
    [外链图片转存中…(img-Q3X7Vb8Z-1714563952526)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 13
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个简单的 Android 音乐播放器示例,使用 Kotlin 语言: 首先,需要在 AndroidManifest.xml 文件中添加以下权限: ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WAKE_LOCK" /> ``` 然后,在 MainActivity.kt 文件中添加以下代码: ```kotlin import android.media.MediaPlayer import android.os.Bundle import android.os.Handler import android.os.Message import android.util.Log import android.view.View import android.widget.SeekBar import androidx.appcompat.app.AppCompatActivity import kotlinx.android.synthetic.main.activity_main.* import java.io.IOException class MainActivity : AppCompatActivity() { private var mediaPlayer: MediaPlayer? = null private var isPlaying = false private var isPaused = false private var currentSongIndex = 0 private var songList = mutableListOf<String>() private lateinit var seekBarHandler: Handler private lateinit var runnable: Runnable override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) // Add songs to playlist songList.add("/sdcard/Music/Song1.mp3") songList.add("/sdcard/Music/Song2.mp3") songList.add("/sdcard/Music/Song3.mp3") // Initialize media player mediaPlayer = MediaPlayer() // Set up seek bar handler to update seek bar position seekBarHandler = object : Handler() { override fun handleMessage(msg: Message) { super.handleMessage(msg) if (mediaPlayer != null) { val currentPosition = mediaPlayer!!.currentPosition seek_bar.progress = currentPosition val duration = mediaPlayer!!.duration val progress = (currentPosition.toFloat() / duration.toFloat() * 100).toInt() seek_bar_percent.text = "$progress%" if (mediaPlayer!!.isPlaying) { runnable = Runnable { sendMessageDelayed(obtainMessage(0), 1000) } postDelayed(runnable, 1000) } } } } // Set up seek bar listener to seek to specific position seek_bar.setOnSeekBarChangeListener(object : SeekBar.OnSeekBarChangeListener { override fun onProgressChanged(seekBar: SeekBar, progress: Int, fromUser: Boolean) { if (fromUser) { mediaPlayer?.seekTo(progress) } } override fun onStartTrackingTouch(seekBar: SeekBar) { seekBarHandler.removeCallbacks(runnable) } override fun onStopTrackingTouch(seekBar: SeekBar) { seekBarHandler.removeCallbacks(runnable) mediaPlayer?.seekTo(seekBar.progress) runnable = Runnable { seekBarHandler.sendEmptyMessage(0) } seekBarHandler.postDelayed(runnable, 1000) } }) // Set up play button listener to play or pause current song btn_play.setOnClickListener { if (!isPlaying) { if (isPaused) { mediaPlayer?.start() isPaused = false } else { playSong(currentSongIndex) } isPlaying = true btn_play.setImageResource(R.drawable.ic_pause) } else { mediaPlayer?.pause() isPaused = true isPlaying = false btn_play.setImageResource(R.drawable.ic_play) } } // Set up next button listener to play next song in playlist btn_next.setOnClickListener { if (currentSongIndex < songList.size - 1) { currentSongIndex++ } else { currentSongIndex = 0 } playSong(currentSongIndex) } // Set up previous button listener to play previous song in playlist btn_prev.setOnClickListener { if (currentSongIndex > 0) { currentSongIndex-- } else { currentSongIndex = songList.size - 1 } playSong(currentSongIndex) } } override fun onDestroy() { super.onDestroy() mediaPlayer?.stop() mediaPlayer?.release() mediaPlayer = null seekBarHandler.removeCallbacks(runnable) } private fun playSong(index: Int) { try { mediaPlayer?.reset() mediaPlayer?.setDataSource(songList[index]) mediaPlayer?.prepare() mediaPlayer?.start() seek_bar.max = mediaPlayer!!.duration seekBarHandler.sendEmptyMessage(0) isPlaying = true btn_play.setImageResource(R.drawable.ic_pause) } catch (e: IOException) { Log.e("MainActivity", "Error playing song: ${e.message}") } } } ``` 这个示例实现了以下功能: - 播放/暂停当前歌曲; - 播放下一首歌曲; - 播放上一首歌曲; - 显示歌曲进度,并允许拖动进度条到指定位置。 请注意,这只是一个简单的示例,实际的音乐播放器可能需要更复杂的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值