class SimpleClass2(x: Int, y: String): SimpleClass(x, y){
override fun zzz(string: String){
}
}
如果类要被子类继承,则要被继承的类名和方法名前面都要加open
关键字,否则不能被继承和覆写。
- 属性引用:
class Person(age: Int, name: String) {
var age: Int = age
var name: String = name
}
fun main() {
val ageRef = Person::age
val person = Person(18, “Bennyhuo”)
val nameRef = person::name //绑定接受者的属性引用,调用set的时候可以不用传接受者
ageRef.set(person, 20)
nameRef.set(“Andyhuo”)
}
扩展方法和扩展属性:
class PoorGuy{
var pocket: Double = 0.0
}
//定义类的扩展方法
fun PoorGuy.noMoney() {
println(“noMoney”)
}
//定义类的扩展属性 property = backing field + getter + setter
var PoorGuy.moneyLeft: Double
get() {
return this.pocket
}
set(value) {
pocket = value
}
就是在类定义大括号之外,再后续给类定义方法和属性,有点像java静态方法的调用形式,但这样定义的是成员方法和属性,并不是java中那样的静态方法。
fun main() {
val poorGuy = PoorGuy()
poorGuy.noMoney()
println(poorGuy.moneyLeft)
poorGuy.moneyLeft = 10000.0
println(poorGuy.moneyLeft)
}
类和扩展方法不一定在同一个kt文件中,但必须在方法的外层定义,即不能在某个函数方法中定义,必须是顶层的,如不能在main方法中定义。
fun Person.eat(s : String) {
println(“eat$s”)
}
var Person.howOld: Int
get() {
return this.age
}
set(value) {
age = value
}
fun main() {
val eat = Person::eat
val person = Person(10, “张三”)
person.eat(“aaa”)
eat(person, “222”)
println(person.howOld)
}
上面文件中只要导入Person这个类就可以给它定义扩展方法。
- 给系统类添加扩展方法:
//给String类定义扩展方法 给String前后加count个空格
fun String.padding(count: Int, char: Char = ’ '): String {
//生成重复count次的空格连续串
val padding = (1 … count).joinToString(“”){ char.toString() }
return “ p a d d i n g {padding} padding{this}${padding}”
}
//给String类定义扩展方法 判断字符串是否是邮箱
fun String.isEmail(): Boolean {
return matches(Regex(“(?:[a-z0-9!# %&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!# %&'+/=?^_`{|}~-]+)|”(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])")@(?😦?:a-z0-9?\.)+a-z0-9?|\[(?😦?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?|[a-z0-9-][a-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])"))
}
//给String类定义扩展方法 字符串重复count次
fun String.times(count: Int): String {
return (1 … count).joinToString(“”) { this }
}
上面是给String类定义扩展方法,使用起来简单:
fun main() {
println(“admin@bennyhuo.com”.isEmail())
println(“Hello”.padding(10))
println(“*”.times(10))
//定义对扩展方法的引用
val stringTimes = String::times
val stringTimesBound = “*”::times
}
只要字符串点方法名即可, 看到这个顿时感觉便捷性这块kotlin确实比java强了太多!
空类型安全:
- kotlin中,指定类型的变量不能赋值为null,即空类型安全
var nonNull: String = “Hello”
nonNull = null //赋值为空,这一行编译器会报错
val length = nonNull.length //这样使用是安全的
就是说kotlin中明确的类型是不能赋值为一个null值的,这与java明显不同。
- 定义可接受null类型的变量,在类型后面加一个 ?
var nullable: String? = “Hello”
val length = nullable?.length //安全访问可能为空的变量
这时变量可能为null, 所以要判空,判空方式 ?.
比java简洁,有点类似js语法。
但是用 ?.
操作符之后,如果变量为null,则最终等号左边的变量结果 也可能为null, 因此在使用length之前还要再判断length是否为null.
var nullable: String? = “Hello”
val length = nullable?.length ?: 0 //确保length不为空的写法 等价三目运算 boolean? a : b
其中操作符 ?:
等价于三目运算符,length 为null 返回右边的,不为null返回左边的。
说明: String 类型是 String? 类型的子类, Int 类型是 Number 类型的子类,使用遵循里氏替换原则:所有使用父类的地方都可以使用子类替换,反之则不行
var x: String = “Hello”
var y: String? = “World”
// x = y // Type mismatch
y = x // OK
var a: Int = 2
var b: Number = 10.0
// a = b // Type mismatch
b = a // OK
- 引用其他平台语言的对象要使用
?.
判空:
java类
public class Person {
@Nullable
public String getTitle(){
return null;
}
}
kotlin类
val person = Person() // 创建一个java的Person类实例
val title = person.title // 此时的title类型是java平台的String类型,不是kotlin的String类型
//kotlin中无法判断title的实际类型是来自哪个平台的,所以主动添加?.可确保安全访问
val titleLength = title?.length
由于kotlin 支持java、javascript、native三个平台,并且在android项目中支持java和kotlin类混用,因此当在kotlin中导入java类的时候,kotlin无法判断导入类创建的对象属性的实际类型是来自哪个平台的,所以主动添加?.可确保安全访问。
下面的例子也是一样:
val file = File(“abc”) //这个File类是java的
val files = file.listFiles()
//因此这里的files使用可能为空,应该使用files?.size
println(files.size)
所以要注意导入的类是不是java的。当然有时出现可能为空时编译器也会给出提示,可以点击快速修复即可。但是java方法上没有加@Nullable
注解的话,编译器也无法识别,这时要人工判断了。
- 类型强转:
val kotliner: Kotliner = Person(“benny”, 20)
if(kotliner is Person){
println((kotliner as Person).name)
}
使用 is
判断是否是实例,对比java的instanceof
使用 as
转换类型,对比java的小括号强制转换
那如果类型转换失败呢,在java中不try-catch会直接抛异常,在kotlin中提供?
的形式安全访问:
val kotliner: Kotliner = Person(“benny”, 20)
if(kotliner is Person){
println((kotliner as? Person)?.name)
}
(kotliner as? Person)?.name
其中 as?
表示转换失败返回null, 所以name前面也同时使用?.
保证安全访问。
- 智能类型转换:
var value: String? = null
value = “benny” //如果显示的经过了类型赋值,在下面的使用是安全的
// if(value != null){
println(value.length);//这里kotlin会隐式的转成kotlin的String类型,不需要大括号
//value = null
// }
我的理解是显示的经过了类型赋值以后,那么再去.引用属性的时候就是安全的,因为kotlin转成了它的内部确定类型,不可能为null。
慕课网上课程介绍说这里的value在大括号里面会转成Sting类型,出了大括号,value又会变成String?类型。我认为他这里说法是错误的,只是在调用.length时才会自动转换,如果调用完了会即刻恢复,而不是出了大括号。你可以把上面注释的代码恢复,你会发现在大括号中是可以给它赋值null的,这说明它还是String? 而不是String, 因为String是不能接受null的。
- 外部变量不支持智能类型转换:
var tag: String? = null
fun main() {
if(tag != null){
//对顶级变量的使用会有风险,即便判断了null, 因为可能有其他线程修改了它的类型
println(tag.length)
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
最后
题外话,我在一线互联网企业工作十余年里,指导过不少同行后辈。帮助很多人得到了学习和成长。
我意识到有很多经验和知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料。
【Android思维脑图(技能树)】
知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。
【Android进阶学习视频】、【全套Android面试秘籍】
希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
长。
我意识到有很多经验和知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料。
【Android思维脑图(技能树)】
知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。
[外链图片转存中…(img-LxdkBggg-1712797341119)]
【Android进阶学习视频】、【全套Android面试秘籍】
希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-tNNACw3a-1712797341119)]