开发中避免延时操作技巧详解

目录
  • 前言
  • 使用延时的场景
  • 获取view的宽高
  • 定时查询服务器结果
  • 广播顺序
  • 延时初始化
  • 使用延时的场景
  • 小心使用延时

前言

开发中我们或多或少会涉及到一些场景需要使用延时操作,而延时操作其实并不是一个很好的选择,并不是一个很好的方案,因为它不可控,也可能产生时序的逻辑问题。这次就来盘点一些使用延时操作的场景和如何去避免,本次内容比较基础。

使用延时的场景

在刚接触开发的时候,我们无脑解决问题的方案基本只有两种,异常捕获和延时。异常捕获容易理解,当碰到崩溃又分析不出原因时,往往加个try-catch就能绕过,但往往这样做只会导致出现更难排查的BUG,当慢慢有经验的时候,也不会再这么做了,毕竟crash提示出来的BUG也都比较好解决。

其实延时操作也是这样,找不到问题出在哪里,然后胡乱尝试,发现在某个地方加个延时几秒就能让流程正常。其实这个和乱加异常捕获一样,往往会引发更难排查的问题,所以有问题就平下心解决问题,不要试图使用这种方式。这也是延时操作最不应该出现的方式。

获取view的宽高

刚开始接触的时候,往往不会正确的获取view的宽高,直接就view.getHeight()然后发现获取到的高度是0,然后不知道为什么,开始瞎尝试,最终尝试出加个延时1秒就能获取到高度。

但是这并不能解决办法,这时正确的做法应该是去了解view的绘制流程,去探究为什么一开始获取不到值,去看源码(当然一开始自己瞎看源码有点难),去看看这个东西是怎样的一个机制,然后结合这些甚至结合别人的分析,再去看源码,就很容易能看得。

就能知道为什么应该用view.post

定时查询服务器结果

假如你有一个应用,你怎么知道你何时被人抢登,你何时能收到别人的消息,等等。往往很多人的做法是写一个定时器,每隔多长时间去向服务器发送http请求查询一下状态,那么这样的做法没问题吗?

如果你把细节都处理好,这样的做法当然没问题,但有没有了解过有一个协议叫websocket,你总有见过有些链接不是http开头,而是ws开头的吧。有没有了解过一个协议叫MQTT,没了解过也没关系,可以看看我这篇基础的文章:https://www.jb51.net/article/276761.htm,甚至上升到智能硬件层面,有没有了解过什么是IOT。

当然不是说轮询请求有问题,只是需要处理一些细节,中断什么的,甚至如果有更好的方法能实现你想要的效果,那为何不用更好的呢?

但是如果你是要定时执行某些本地的任务,那用定时器倒是没有什么问题,关键要处理好一些细节,生命周期、中断操作、暂停操作等等。

广播顺序

复杂的多应用情况下,往往或多或少会使用到广播,那其实广播的注册和广播的发送,就是有个顺序问题。有可能你的某些逻辑导致广播先发送,另外一边再注册,那就会出现接收不到广播的情况。有些人为了简单处理这个问题,往往会加个延时,让广播延时发送。

那这其实是个很危险的操作,正确的做法是,应该去对广播这个东西有一定的了解。你就会知道有种广播类型叫粘性广播,哪怕你熟悉了这个领域的知识,还是不了解它或者说它对你来说在这个场景不适用,那你也会有更好的办法去解决这个问题,而不是通过延时这种不安全的操作。

延时初始化

我们都知道在Application或者在onCreate中做过多的初始化操作是不好的,大家都知道优化,优化启动速度,所以不会在这些地方做初始化。那有些人就会想出一些骚办法,我在这些地方加个延时,延时个一两秒再做初始化,这样又能优化启动速度,又能在对应的功能使用前进行初始化,岂不美哉?你觉得呢

你的延时的原理是什么,handler机制,那有没有了解有个东西叫IdleHandler。

其实单独用IdleHandler也不是一个安全的操作,那为什么不放在第一次使用的时候再给功能做初始化呢?有的人可能会说,如果初始化是耗时操作,那在第一次使用的时候再进行初始化,就会影响使用的体验。对于这个问题,我一般情况是一起使用,又在IdleHandler进行初始化,也在第一次使用的时候判断没有初始化的话再进行初始化。特殊的情况可以在某些地方进行,这个要看具体的需求。

使用延时的场景

那既然延时是一个这么危险的操作,并且一般都有更好的方式去替代,那我们是不是就打死都不使用延时操作。并不是,有时候不用还真不行,那是什么时候需要使用?当然是没办法通过其它方法去处理这个问题的时候。用,但是要小心用。

比如我调用别人的一个库,那个库里做了操作,然后给我回调,这是一个基本的流程嘛。但是,这个库不是你的,可能他写了BUG或者什么原因,导致你调用他的方法,但他不给你回调。那这种情况下如果你不做什么操作,你就会一直卡在这。

所以针对这种情况,一般会做个超时机制去让这个流程更安全一些,比如说你20秒不给我回调,我就返回失败。当然我觉得首先应该和库的作者去沟通这个问题,实在没办法了,才用这种下下策。

但是如果这么做了,那要注意状态,比如说你已经超时了,要是他这时候再给你回调怎么办?所以用这种方法,还需要写很多东西去保证它的安全。

还有一种情况是Loading,这个可能我从以前开始就这样操作比较多,Loading的时候我不会马上显示菊花转,会延时0.5秒再显示,这样能有比较好的体验。

当然还有你想先写个延时,然后过几个版本你和你老板说,我要做个优化,然后你把延时给去掉,看到没有,速度明显快了,如果你想这样玩,那就当我没说。

还有,你这种延时也是有讲究的,比如我做重绘更新页面后页面显示后才做某些操作,那我怎么做,系统有方法实现,如果你说我就要做延时(我这里为了举个例子),那你就要知道屏幕刷新是16ms,但如果没画完,会放到下次刷新,为了安全,你可以设置高一些,你可以设80ms的延时,但没必要设到一两秒。

小心使用延时

为什么说要尽量避免使用延时操作,因为这个操作确实坑多,在Android中大部分的延时操作都会用postDelayed去实现。

首先你要考虑一个问题,中断问题,需要有个中断机制,比如你在Activity做了延时操作,但是Activity销毁了,这时候你延时时间到了难道还要继续执行操作?所以会在Activity的onDestroy里面去移除Handler的消息。

假设你加了中断操作,但是只这样做安全吗?有没有考虑过你中断的时候其实消息已经开始处理了。所以这时候还需要用一个状态去做判断,根据这个状态判断Activity是否被销毁,被消耗了就不执行后面的操作。

这里也只是列举其中一个场景,其实在使用延时的时候往往会很危险,所以使用需谨慎,能不用就不用,如果一定要用,也需要考虑周全。

以上就是开发中避免延时操作技巧详解的详细内容,更多关于避免延时操作技巧的资料请关注我们其它相关文章!

(0)

相关推荐

  • Redisson 分布式延时队列 RedissonDelayedQueue 运行流程

    目录 前言 基本使用 内部数据结构介绍 基本流程 发送延时消息 获取延时消息 初始化延时队列 总结 前言 因为工作中需要用到分布式的延时队列,调研了一段时间,选择使用 RedissonDelayedQueue,为了搞清楚内部运行流程,特记录下来. 总体流程大概是图中的这个样子,初看一眼有点不知从何下手,接下来我会通过以下几点来分析流程,相信看完本文你能了解整个运行流程. 基本使用 内部数据结构介绍 基本流程 发送延时消息 获取延时消息 初始化延时队列 基本使用 发送延迟消息代码如下,发送了一条延

  • Bat脚本-timeout 命令(延时执行)

    概述 做一个延时执行的批处理命令,选用timeout命令来实现,但是碰到一个情况是写完后直接双击运行时,timeout 命令运行不了,每次运行到timeout的时候,命令就会跳到.bta文件的开头重新执行,后来选择管理员权限运行就可以了. 命令-timeout 在命令行中运行 timeout  /? 可以得到该命令的说明 C:\Users\L>timeout /? TIMEOUT [/T] timeout [/NOBREAK] 描述:    这个工具接受超时参数,等候一段指定的时间(秒)或等按任

  • 盘点Java中延时任务的多种实现方式

    目录 场景描述 实现方式 一.挂起线程 二.ScheduledExecutorService 延迟任务线程池 三.DelayQueue(延时队列) 四.Redis-为key指定超时时长,并监听失效key 五.时间轮 六.消息队列-延迟队列 场景描述 ①需要实现一个定时发布系统通告的功能,如何实现? ②支付超时,订单自动取消,如何实现? 实现方式 一.挂起线程 推荐指数:★★☆ 优点: JDK原生(JUC包下)支持,无需引入新的依赖: 缺点: (1)基于内存,应用重启(或宕机)会导致任务丢失 (2

  • shell 创建子进程及并行延时执行命令方法

    目录 shell 创建子进程方法 1. 什么是shell子进程 2. shell什么情况下会产生子进程 3.使用括号来创建子进程 shell 创建子进程方法 1. 什么是shell子进程 子进程,是从父子进程的概念出发的,unix操作系统的进程从init进程开始(init进程为1,而进程号0为系统原始进程,以下讨论的进程原则上不包括进程0)均有其对应的子进程,就算是由于父进程先行结束导致的孤儿进程,也会被init领养,使其父进程ID为1.也因为所有的进程均有父进程,事实上,所有进程的创建,都可视

  • 生产redisson延时队列不消费问题排查解决

    目录 问题描述 初步排查 排查过程 解决方案 redisson 延时队列原理 流程总结 问题描述 项目使用redisson延时队列功能,实现直播的开播提醒,突然有一天业务爆出问题,未触发开播提醒. 初步排查 首先通过查询生产日志,发送端日志存在,没有消费日志,猜测消费端没有消费到延时消息,,在dba的协助下查询redis队列,消息也确实存在,但已经过了过期时间,由此证明redisson消费者出现问题.通过服务日志发现在最后一次设置自定义推送任务是在一次服务发布之前,服务发布后,之前设置的自定义推

  • 开发中避免延时操作技巧详解

    目录 前言 使用延时的场景 获取view的宽高 定时查询服务器结果 广播顺序 延时初始化 使用延时的场景 小心使用延时 前言 开发中我们或多或少会涉及到一些场景需要使用延时操作,而延时操作其实并不是一个很好的选择,并不是一个很好的方案,因为它不可控,也可能产生时序的逻辑问题.这次就来盘点一些使用延时操作的场景和如何去避免,本次内容比较基础. 使用延时的场景 在刚接触开发的时候,我们无脑解决问题的方案基本只有两种,异常捕获和延时.异常捕获容易理解,当碰到崩溃又分析不出原因时,往往加个try-cat

  • Android 开发中使用Linux Shell实例详解

    Android 开发中使用Linux Shell实例详解 引言 Android系统是基于Linux内核运行的,而做为一名Linux粉,不在Android上面运行一下Linux Shell怎么行呢? 最近发现了一个很好的Android Shell工具代码,在这里分享一下. Shell核心代码 import java.io.BufferedReader; import java.io.DataOutputStream; import java.io.IOException; import java.

  • Java开发中为什么要使用单例模式详解

    一.什么是单例模式? 单例设计模式(Singleton Design Pattern)理解起来非常简单.一个类只允许创建一个对象(或者实例),那这个类就是一个单例类,这种设计模式就叫作单例设计模式,简称单例模式. 二.实战案例一:处理资源访问冲突 我们先来看第一个例子.在这个例子中,我们自定义实现了一个往文件中打印日志的 Logger 类.具体的代码实现如下所示: public class Logger { private FileWriter writer; public Logger() {

  • Java Web开发中过滤器和监听器使用详解

    目录 1 Filter 1.1 Filter简介 1.2 Filter的快速入门 1.2.1 创建Filter类 1.2.2 访问index.jsp 1.3 Filter的拦截路径的配置 1.4 过滤器链 1.4.1 过滤器链简介 1.4.2 过滤器链的例子 2 Listener 2.1 概念 2.2 监听器的使用 1 Filter 1.1 Filter简介 Filter表示过滤器,是JavaWeb三大组件(Servlet.Filter.Listener)之一. 过滤器可以把资源的请求拦截下来,

  • Hutool开发利器MapProxy类使用技巧详解

    目录 概述 场景引入 MapProxy使用 MapProxy源码解析 总结 概述 Hutool是一个小而全的Java工具类库,通过静态方法封装,降低相关API的学习成本,提高工作效率,使Java拥有函数式语言般的优雅,让Java语言也可以“甜甜的”. 目前公司项目中主要采用Hutool作为项目的工具包,相对于google的guava, hutool的工具类采用中文注释,更加符合国人使用.所谓知己知彼,我们需要了解Hutool都具有什么样的功能,才能够最大化发挥它的价值. 本文主要就hutool

  • JDBC中resutset接口操作实例详解

    本文主要向大家展示JDBC接口中resutset接口的用法实例,下面我们看看具体内容. 1. ResultSet细节1 功能:封锁结果集数据 操作:如何获得(取出)结果 package com.sjx.a; import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.Statement; import org.junit.Test; //1. next方

  • IOS 开发中画扇形图实例详解

    IOS 开发中画扇形图实例详解 昨天在做项目中,遇到一个需要显示扇形图的功能,网上搜了一下,发现code4app里面也没有找到我想要的那种类似的效果,没办法了,只能自己学习一下如何画了. 首先我们需要了解一个uiview的方法 -(void)drawRect:(CGRect)rect 我们知道了这个方法,就可以在自定义UIView的子类的- (void)drawRect:(CGRect)rect里面绘图了,关于drawrect的调用周期,网上也是一找一大堆,等下我会整理一下,转载一篇供你们参考.

  • Android程序开发中单选按钮(RadioGroup)的使用详解

    在还没给大家介绍单选按钮(RadioGroup)的使用,先给大家展示下效果图吧: xml文件 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_heig

  • vue中的双向数据绑定原理与常见操作技巧详解

    本文实例讲述了vue中的双向数据绑定原理与常见操作技巧.分享给大家供大家参考,具体如下: 什么是双向数据绑定? vue是一个mvvm框架,即数据双向绑定,即当数据发生变化的时候,视图也就发生变化,当视图发生变化的时候,数据也会跟着同步变化.这也是算是vue的精髓之处了.值得注意的是,我们所说的数据双向绑定,一定是对于UI控件来说的,非UI控件不会涉及到数据双向绑定.单向数据绑定是使用状态管理工具的前提,如果我们使用vuex,那么数据流也是单向的,这时就会和双向数据绑定有冲突,我们可以这么解决.

  • Android开发中滑动分页功能实例详解

    本文实例讲述了Android开发中滑动分页功能.分享给大家供大家参考,具体如下: android UI 往右滑动,滑动到最后一页就自动加载数据并显示 如图: Java代码: package cn.anycall.ju; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import android.app.Activity; import andro

随机推荐