zoukankan      html  css  js  c++  java
  • Kotlin入门(33)运用扩展属性

    进行App开发的时候,使用震动器要在AndroidManifest.xml中加上如下权限:

        <!-- 震动 -->
        <uses-permission android:name="android.permission.VIBRATE" />
    

    让手机震动的功能用到了震动器Vibrator类,而震动器对象从系统服务VIBRATOR_SERVICE获得,实现该功能的代码很简单,即便用Java书写也只有以下两行代码:

        Vibrator vibrator = (Vibrator) getSystemService(Context.VIBRATOR_SERVICE);
        vibrator.vibrate(3000);
    

    两行代码看起来真没什么好简化的了,因为转换成Kotlin也要下面的两行代码:

        //常规做法:从系统服务中获取震动器对象
        val vibrator = getSystemService(Context.VIBRATOR_SERVICE) as Vibrator
        vibrator.vibrate(3000)
    

    虽然获取震动器的代码并不多,但是这真的真的很难记忆,首先开发者要调用getSystemService一把,接着绞尽脑汁才能想起该服务的名称是VIBRATOR_SERVICE,最后再强制将类型转换为Vibrator。其中又是大写子母又是小写字母还有大小写混合,对于英文不溜的朋友来说,这简直是个灾难。如果只要一个琅琅上口的单词就能代表震动器,那势必为开发者省去了背诵专业英语单词的麻烦。然而两行代码还能怎么优化?倘若改造成工具类获取震动器对象,也不见得一定省事。
    不过Kotlin可不会善罢甘休,相反是迎难而上,因为它坐拥扩展函数这个法宝,之前我们多次见识了扩展函数的威力,比如提示窗的toast、提醒对话框的alert等等。当然获取震动器对象也能按照扩展函数来改造,比如给Context添加一个扩展函数getVibrator,则该扩展函数的Kotlin代码示例如下:

    //获取震动器
    fun Context.getVibrator() : Vibrator {
        return getSystemService(Context.VIBRATOR_SERVICE) as Vibrator
    }
    

    接着回到Activity页面代码,实现震动功能只需下面的一行代码了:

        //利用扩展函数获得震动器对象
        getVibrator().vibrate(3000)
    

    以上代码固然简化了,却仍然不是最简单的写法,看看getVibrator()方法,前面有get后面有括号,都是碍手碍脚的家伙。可去掉括号就不是函数了,而变成了属性,难不成Kotlin啥时多了个扩展属性的用法?其实Kotlin还真的可以实现扩展属性的功能,关键是要利用扩展函数进行移花接木,只要在kt文件中声明一个Context类的新属性,同时定义该属性的get方法(get方法为扩展函数)。如此一来,外部访问该扩展属性之时,编译器会自动调用该属性的get方法,从而通过扩展函数间接实现了扩展属性。接下来依旧以震动器为例,看看如何使用Kotlin代码声明扩展属性vibrator:

    //获取震动器
    //利用扩展函数实现扩展属性,在Activity代码中即可直接使用vibrator
    val Context.vibrator : Vibrator
            get() = getSystemService(Context.VIBRATOR_SERVICE) as Vibrator
    

    现在回到Activity代码,如下所示只要通过vibrator就能访问震动器的方法了:

        //利用扩展函数实现扩展属性,直接使用vibrator即可指代震动器对象
        vibrator.vibrate(3000)
    

    当然要想正常访问自定义的扩展函数和扩展属性,需要在活动代码头部加上以下的导入语句:

    import com.example.custom.util.vibrator
    

      

    除了震动器之外,其它从系统服务获得对象的管理器也能照此办理,譬如通知管理器NotificationManager,按照之前的调用方式是下面的Kotlin代码:

        val notifyMgr = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
        notifyMgr.notify(R.string.app_name, notify)
    

    显然通知管理器对象的获取代码更冗长,接下来将其改造为扩展属性的方式,则相应的Context扩展代码如下所示:

    //获取通知管理器
    //试试在Activity代码中调用“notifier.notify(R.string.app_name, notify)”
    val Context.notifier: NotificationManager
        get() = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
    

    然后原来通知管理器的两行代码便缩减为下面的一行代码了:

        notifier.notify(R.string.app_name, notify)
    

      

    举一反三,剩下的来自系统服务的管理器统统运用扩展属性,能够更好地方便将来的开发工作。下面是几个常用管理器的扩展属性实现代码例子:

    //获取下载管理器
    val Context.downloader: DownloadManager
        get() = getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager
    //获取定位管理器
    val Context.locator: LocationManager
        get() = getSystemService(Context.LOCATION_SERVICE) as LocationManager
    //获取连接管理器
    val Context.connector: ConnectivityManager
        get() = getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager
    //获取电话管理器
    val Context.telephone: TelephonyManager
        get() = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
    //获取无线管理器
    val Context.wifi: WifiManager
        get() = getSystemService(Context.WIFI_SERVICE) as WifiManager
    //获取闹钟管理器
    val Context.alarm: AlarmManager
        get() = getSystemService(Context.ALARM_SERVICE) as AlarmManager
    //获取音频管理器
    val Context.audio: AudioManager
        get() = getSystemService(Context.AUDIO_SERVICE) as AudioManager
    

      

  • 相关阅读:
    全站仪定向距离差 方向不差 这样敢放线吗
    关于老王
    cad巧用插件自定义填充图形
    老王教你永不会错的测量坐标方位角计算方法
    jqgrid 点击列头的超链接或按钮时,不触发列排序事件
    jqgrid 将列头设置为超链接或按钮
    jqgrid 设置隔行换色
    jqgrid 设置行编辑为本地端编辑状态
    jqgrid 让隐藏的列在编辑状态时出现且可编辑
    jqgrid 设置编辑行中的某列为下拉选择项
  • 原文地址:https://www.cnblogs.com/aqi00/p/9939649.html
Copyright © 2011-2022 走看看