Android超实用的Toast提示框优化分享

前言

相信每位Android开发者都用过Toast,都知道是弹出消息的。类似于js里面的alert,C#里面的MesageBox。当然android里面也有dialogdialog是有焦点的,可与用户交互。而toast是没有焦点的,时间到了自动消失,不能回应用户的交互,下面就跟大家分享下Android中Toast提示框的优化方法。

先看下源码:

public class Toast { 

 public static final int LENGTH_SHORT = 0; 

 public static final int LENGTH_LONG = 1; 

 /**
  * 构造一个空的toast。你必须在调动show()之前,线调用setView()
  * @param context 参数,application或者activity都可以
  */
 public Toast(Context context) {
  ...
  //获取系统内置的toast_y_offset常量值
  mTN.mY = context.getResources().getDimensionPixelSize(
    com.android.internal.R.dimen.toast_y_offset);
  mTN.mGravity = context.getResources().getInteger(
    com.android.internal.R.integer.config_toastDefaultGravity);
 } 

 /**
  * 在指定的时长显示view视图
  */
 public void show() {
  //如果mNextView为空,即没有设置view
  if (mNextView == null) {
   throw new RuntimeException("setView must have been called");
  } 

  //通过系统服务,获取通知管理器。看来这是使用系统通知?
  INotificationManager service = getService();
  String pkg = mContext.getOpPackageName();
  TN tn = mTN;
  tn.mNextView = mNextView; 

  try {
   service.enqueueToast(pkg, tn, mDuration);
  } catch (RemoteException e) {
   // Empty
  }
 } 

 /**
  * 关闭一个正在显示的toast, 或者取消一个未显示的toast.
  * 通常你不必调用它,在适当的时长后它会自动消失的。
  */
 public void cancel() {
  mTN.hide(); 

  try {
   getService().cancelToast(mContext.getPackageName(), mTN);
  } catch (RemoteException e) {
   // Empty
  }
 } 

 /**
  * 设置toast显示的视图内容,不单单是黑色的界面,你可以自己决定显示什么
  * @see #getView
  */
 public void setView(View view) {
  mNextView = view;
 } 

 /**
  * 设置时长,只能是下面这两个常量值,没什么卵用
  * @see #LENGTH_SHORT 2000毫秒
  * @see #LENGTH_LONG 3500毫秒
  */
 public void setDuration(@Duration int duration) {
  mDuration = duration;
 } 

 /**
  * 设置view的外间距,不用多说.
  *
  * @param horizontalMargin The horizontal margin, in percentage of the
  *  container width, between the container's edges and the
  *  notification
  * @param verticalMargin The vertical margin, in percentage of the
  *  container height, between the container's edges and the
  *  notification
  */
 public void setMargin(float horizontalMargin, float verticalMargin) {
  mTN.mHorizontalMargin = horizontalMargin;
  mTN.mVerticalMargin = verticalMargin;
 } 

 /**
  * 设置notification在屏幕中的方位,大家都知道.上中下左中右什么的都有
  * @see android.view.Gravity
  * @see #getGravity
  */
 public void setGravity(int gravity, int xOffset, int yOffset) {
  mTN.mGravity = gravity;
  mTN.mX = xOffset;
  mTN.mY = yOffset;
 } 

 /**
  * 构造一个只包含一个TextView的标准toast对象
  *
  * @param context 通常是application或者activity对象
  * @param text  用于显示的文本,可以是formatted text.
  * @param duration 显示时长. LENGTH_SHORT或LENGTH_LONG
  *
  */
 public static Toast makeText(Context context, CharSequence text, @Duration int duration) {
  Toast result = new Toast(context); 

  LayoutInflater inflate = (LayoutInflater)
    context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); 

  //包含了一个默认的TextView,这个textview的布局位置在 

com.android.internal.R.layout.transient_notification,可以去查看下内容
  View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);
  TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);
  tv.setText(text); 

  result.mNextView = v;
  result.mDuration = duration; 

  return result;
 } 

 /**
  * 更新通过makeText()方法创建出来的toast对象显示的文本内容
  * @param s 待显示的新文本内容.
  */
 public void setText(CharSequence s) {
  if (mNextView == null) {
   throw new RuntimeException("This Toast was not created with Toast.makeText()");
  } 

  /**
   * 看来com.android.internal.R.layout.transient_notification布局里面的唯一的
   * TextView标签的id是R.id.message。拿到这个textview,设置新文本内容
   */
  TextView tv = (TextView) mNextView.findViewById(com.android.internal.R.id.message);
  if (tv == null) {
   throw new RuntimeException("This Toast was not created with Toast.makeText()");
  }
  tv.setText(s);
 } 

 static private INotificationManager getService() {
  if (sService != null) {
   return sService;
  }
  //获取远程的通知服务
  sService = INotificationManager.Stub.asInterface(ServiceManager.getService("notification"));
  return sService;
 } 

 //TN是一个瞬态通知的子类,里面包含显示和隐藏两个任务对象
 private static class TN extends ITransientNotification.Stub {
  final Runnable mShow = new Runnable() {
   @Override
   public void run() {
    handleShow();
   }
  }; 

  final Runnable mHide = new Runnable() {
   @Override
   public void run() {
    handleHide();
    // Don't do this in handleHide() because it is also invoked by handleShow()
    mNextView = null;
   }
  }; 

  //出现Handler了哦
  final Handler mHandler = new Handler(); 

  /**
   * 调度handleShow任务到执行线程中
   */
  @Override
  public void show() {
   if (localLOGV) Log.v(TAG, "SHOW: " + this);
   //handler发送异步任务了
   mHandler.post(mShow);
  } 

  /**
   * 同上
   */
  @Override
  public void hide() {
   if (localLOGV) Log.v(TAG, "HIDE: " + this);
   mHandler.post(mHide);
  } 

  //...
 } 

} 

通过上面的源码解读,了解到有远程通知,handler异步任务等信息,不多说,自己看。

重点是toast的用法:

1、直接调用makeText静态方法即可,返回的是Toast对象,最后别忘了调用show方法显示:

Toast.makeText(context, text, duration).show(); 

Toast toast = Toast.makeText(context, text, duration);
pre name="code" class="html"> toast.setView(view);
toast.setText(s);
toast.setGravity(gravity, xOffset, yOffset);
toast.setDuration(duration);
toast.show();

2、还可以使用new的方式创建,别忘了setView()方法:

Toast toast = new Toast();
toast.setView(view);
toast.setText(s);
toast.setGravity(gravity, xOffset, yOffset);
toast.setDuration(duration);
toast.show(); 

以上这些都不值得一提,很简单。

在开发过程中,有这样的需求:在项目总,我们偷懒,想连串toast出多个变量的值或者其他任务,可在操作手机时直观可见。问题来了,弹出是无论我们的操作有多快,这些toast内容都是一个跟着一个显示,没办法快进。哪怕我们玩完了,退出了app,它还在弹。怎么办?有没有办法让toast的内容与我们的操作同步,快速反应?

public class T {
 private static Toast toast; 

 public static void show(Context context, String msg) {
  if (toast == null) {
   toast = Toast.makeText(context, msg, Toast.LENGTH_SHORT);
  } else {
   toast.setText(msg);
  }
  toast.show();
 }
} 

单例模式,每次创建toast都调用这个类的show方法,Toast的生命周期从show开始,到自己消失或者cancel为止。如果正在显示,则修改显示的内容,你持有这个对象的引用,当然可以修改显示的内容了。若每次你都makeText或者new一个toast对象,即每次通过handler发送异步任务,调用远程通知服务显示通知,当然是排队等待显示了。

结束语

以上就是Android中Toast提示框优化的全部内容,希望对大家开发Android能有所帮助,如果有大家有疑问可以留言交流。

时间: 2016-09-03

Android编程之Button控件配合Toast控件用法分析

本文实例讲述了Android编程之Button控件配合Toast控件用法.分享给大家供大家参考,具体如下: 在本章教程中,我们将会学习Button控件的使用,同时顺便说一下Toast提示控件. 在Android程序开发中,我们使用最多的用户交互控件可能就是Button的了,而我们使用最多的事件估计也就是onclick事件了. 这些事件也是最简单的事件,我们一般通过google自带的API接口就可以调用了,我们具体看看怎么做吧. 第一步.新建一个工程Ep.Toast,活动和主视图名称我都使用默认的

Android编程实现Toast只显示最后一条的方法

本文实例讲述了Android编程实现Toast只显示最后一条的方法.分享给大家供大家参考,具体如下: 在做Android开发中,时不时的可能会用到Toast,但用Toast的时候,连续使用会存在一个问题,就是一条条显示Toast.而不是直接显示最后一条.因此,根据此需求,现在写了ToastUtil这个类,该类中有三个方法供选择. ToastUtil.Java import android.content.Context; import android.graphics.PixelFormat;

Android实现Toast提示框图文并存的方法

本文实例讲述了Android实现Toast提示框图文并存的方法.分享给大家供大家参考,具体如下: 程序如下: import android.app.Activity; import android.graphics.Color; import android.os.Bundle; import android.text.util.Linkify; import android.view.Gravity; import android.view.View; import android.view.

超简单实现Android自定义Toast示例(附源码)

Bamboy的自定义Toast,(以下称作"BToast") 特点在于使用简单, 并且自带两种样式: 1)普通的文字样式: 2)带图标样式. 其中图标有√和×两种图标. BToast还有另外一个特点就是: 系统自带Toast采用的是队列的方式,当前Toast消失后,下一个Toast才能显示出来: 而BToast会把当前Toast顶掉, 直接显示最新的Toast. 那么,简单三步,我们现在就开始自定义一下吧! (一).Layout: 要自定义Toast, 首先我们需要一个XML布局. 但

Android Toast的用法总结(五种用法)

Toast大家都很熟,不多说.直接上图上代码.     具体代码如下: main.xml: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layou

android开发教程之实现toast工具类

Android中不用再每次都写烦人的Toast了,直接调用这个封装好的类,就可以使用了! 复制代码 代码如下: package com.zhanggeng.contact.tools; /** * Toasttool can make you  use Toast more easy ;  *  * @author ZHANGGeng * @version v1.0.1 * @since JDK5.0 * */import android.content.Context;import andro

android自定义toast(widget开发)示例

1.Toast控件: 通过查看源代码,发现Toast里面实现的原理是通过服务Context.LAYOUT_INFLATER_SERVICE获取一个LayoutInflater布局管理器,从而获取一个View对象(TextView),设置内容将其显示 复制代码 代码如下: public static Toast makeText(Context context, CharSequence text, int duration) {        Toast result = new Toast(c

Android 5.0以上Toast不显示的解决方法

原因分析 用户使用android 5.0以上的系统在安装APP时,将消息通知的权限关闭掉了.实际上用户本意只是想关闭Notification,但是Toast的show方法中有调用INotificationManager这个类,而这个类在用户关闭消息通知权限的同时被禁用了,所以我们的吐司无法显示. Toast.show() 效果图 自定义Toast(上)与Toast(下)比对 问题解决 既然系统不允许我们调用Toast,那么我们就自立门户--自己写一个Toast出来.我们总体的思路是:在Activ

android之自定义Toast使用方法

Android系统默认的Toast十分简洁,使用也非常的简单.但是有时我们的程序使用默认的Toast时会和程序的整体风格不搭配,这个时候我们就需要自定义Toast,使其与我们的程序更加融合. 使用自定义Toast,首先我们需要添加一个布局文件,该布局文件的结构和Activity使用的布局文件结构一致,在该布局文件中我们需设计我们Toast的布局,例如: 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> &

Android控件系列之Toast使用介绍

Toast英文含义是吐司,在Android中,它就像烘烤机里做好的吐司弹出来,并持续一小段时间后慢慢消失 Toast也是一个容器,可以包含各种View,并承载着它们显示. 使用场景: 1.需要提示用户,但又不需要用户点击"确定"或者"取消"按钮. 2.不影响现有Activity运行的简单提示. 用法: 1.可以通过构造函数初始化: 复制代码 代码如下: //初始化Toast Toast toast = new Toast(this); //设置显示时间,可以选择To

Android Service中使用Toast无法正常显示问题的解决方法

本文实例讲述了Android Service中使用Toast无法正常显示问题的解决方法.分享给大家供大家参考,具体如下: 在做Service简单练习时,在Service中的OnCreate.OnStart.OnDestroy三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法: DemoService代码如下: @Override public void onCreate() { su

在Android开发中替换资源图片不起作用的解决方法

现象 在android开发中,经常会需要替换res\drawable中的图片,打开res\layout下的文件预览布局页面发现图片已经被替换,但在模拟器或者真实机器上运行时发现该图片并没有被替换,还是使用的是原来的资源图片. 原因 在开发过程中,由于使用模拟器测试了程序,在首次运行后会将res文件夹下的图片资源文件(如drawable-hdpi.drawable-ldpi和drawable-mdpi)拷贝到bin文件夹下.在替换资源图片后,eclipse并不清楚是否有图片改变,所以会使用原来bi

Android Service中方法使用详细介绍

 service作为四大组件值得我们的更多的关注 在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务.例如,一个从service播放音乐的音乐播放器,应被设置为前台运行,因为用户会明确地注意它的运行.在状态栏中的通知可能会显示当前的歌曲并且允许用户启动一个activity来与音乐播放器交互. Service的两种实现形式 1.非绑定 通过调用应用程序组件(例如Activity)的startService()方法来启动一个服务.一旦启动,服务就会在

Toast类避免显示时间叠加的方法

本文为大家分享了Toast类避免显示时间叠加的方法,供大家参考,具体内容如下 import android.app.Activity; import android.app.Fragment; import android.content.Context; import android.widget.Toast; /** * Toast工具类 * Created by user on 2016/12/22. */ public class ToastUtil { private static T

Android Listview 滑动过程中提示图片重复错乱的原因及解决方法

主要分析Android中Listview滚动过程造成的图片显示重复.错乱.闪烁的原因及解决方法,顺便跟进Listview的缓存机制. 1.原因分析 Listview item 缓存机制:为了使得性能更优,Listview会缓存行item(某行对应的view).listview通过adapter的getview函数获得每行的item.滑动过程中, a.如果某行item已经划出屏幕,若该item不在缓存内,则put进缓存,否则更新缓存: b.获取滑入屏幕的行item之前会先判断缓存中是否有可用的it

移动端页面在ios中不显示图片的解决方法

在移动端开发中,有的时候可能遇到这样的问题,我从别人网站上下载下来的图片,然后做出H5页面,但是在浏览器中和android中都显示正常,可是一到ios中图片就不显示了,这个时候就需要注意了,可能是图片的格式问题导致ios中不认识,比如我从网上下载的图片保存到电脑中不能预览的图片就是这种. 在计算机中打开预览图片显示如下: 这样的图片在ios中就不显示,解决办法很简单,就是在下载的时候去掉后面的类型就可以了, 以上这篇移动端页面在ios中不显示图片的解决方法就是小编分享给大家的全部内容了,希望能给

Android开发中button按钮的使用及动态添加组件方法示例

本文实例讲述了Android开发中button按钮的使用及动态添加组件方法.分享给大家供大家参考,具体如下: MainActivity.java package com.example.lab2; import android.os.Bundle; import android.app.Activity; import android.content.Context; import android.view.Menu; import android.view.View; import andro

基于Linux系统中python matplotlib画图的中文显示问题的解决方法

最近想学习一些python数据分析的内容,就弄了个爬虫爬取了一些数据,并打算用Anaconda一套的工具(pandas, numpy, scipy, matplotlib, jupyter)等进行一些初步的数据挖掘和分析. 在使用matplotlib画图时,横坐标为中文,但是画出的条形图横坐标总是显示"框框",就去查资料解决.感觉这应该是个比较常见的问题,网上的中文资料也确实很多,但是没有任何一个彻底解决了我遇到的问题.零零碎碎用了快3个小时的时间,才终于搞定.特此分享,希望能帮到有同

5个Android开发中比较常见的内存泄漏问题及解决办法

android中一个对象已经不需要了,但是其他对象还持有他的引用,导致他不能回收,导致这个对象暂存在内存中,这样内存泄漏就出现了. 内存泄漏出现多了,会是应用占用过多的没存,当占用的内存超过了系统分配的内存容量,就会出现内存溢出了导致应用Crash. 了解了内存泄漏的原因及影响后,我们需要做的就是掌握常见的内存泄漏,并在以后的Android程序开发中,尽量避免它.下面搜罗了5个Android开发中比较常见的内存泄漏问题及解决办法,分享给大家,一起来看看吧. 一.单例造成的内存泄漏 android

详解Android Libgdx中ScrollPane和Actor事件冲突问题的解决办法

详解Android Libgdx中ScrollPane和Actor事件冲突问题的解决办法 在Libgdx的使用过程中,经常会用到ScrollPane这个widget,来实现滑动效果, 如下所示: 但是如果想在上面的效果上添加一点扩展,比如ScrollPane中的Actor可以从ScrollPane中移出来,并添加到Stage中,则需要添加额外的逻辑 具体代码参考如下: /** * Created by Danny.姜 on 17/7/26. */ public class TestAdapter