`
iaiai
  • 浏览: 2204425 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

[转]Android开发中,那些让你觉得相见恨晚的方法、类或接口

 
阅读更多
本篇文章内容提取自知乎Android开发中,有哪些让你觉得相见恨晚的方法、类或接口?,其实有一部是JAVA的,但是在android开发中也算常见。大多数的函数自己还是见过的,这里记录一下备忘。同时呢,也推荐一个github项目,里面记录了自己日常开发中见过的比较有用的东西开发中常用的工具、链接

  • Throwable类中的getStackTrace()方法,根据这个方法可以得到函数的逐层调用地址,其返回值为StackTraceElement[],而在StackTraceElement类中有四个方法getClassName(),getFileName(),getLineNumber(),getMethodName()在调试程序打印Log时非常有用。

try {
    int num = 1 / 0;
} catch (Exception e) {
    e.printStackTrace();
    StackTraceElement[] stackTrace = e.getStackTrace();
    for (StackTraceElement element : stackTrace) {
        String className = element.getClassName();
        int lineNumber = element.getLineNumber();
        String fileName = element.getFileName();
        String methodName = element.getMethodName();
        Log.e("TAG","fileName:"+fileName+" lineNumber:"+lineNumber+" className:"+className+" methodName"+methodName);
    }
}

  • UncaughtExceptionHandler接口,再好的代码异常难免,利用此接口可以处理未捕获的异常。比如NullPointerException空指针异常抛出时,用户没有try catch捕获,那么,Android系统会弹出对话框的“XXX程序异常退出”,给应用的用户体验造成不良影响。为了捕获应用运行时异常并给出友好提示,便可继承UncaughtExceptionHandler类来处理。

/**
 * 异常处理类
 * User:lizhangqu(513163535@qq.com)
 * Date:2015-08-04
 * Time: 14:48
 */
public class CrashHandler implements Thread.UncaughtExceptionHandler {
    private static final String TAG = CrashHandler.class.getSimpleName();
    private Context mContext;
    private static volatile CrashHandler instance;
    private Thread.UncaughtExceptionHandler defalutHandler;
    private DateFormat formatter = new SimpleDateFormat(
            "yyyy-MM-dd_HH-mm-ss.SSS", Locale.CHINA);
    private CrashHandler(){

    }
    /**
     * 获得单例
     * @return 单例
     */
    public static CrashHandler getInstance() {
        if (instance==null){
            synchronized (CrashHandler.class){
                if (instance==null){
                    instance=new CrashHandler();
                }
            }
        }
        return instance;
    }

    public void init(Context context){
        mContext=context.getApplicationContext();
        defalutHandler=Thread.getDefaultUncaughtExceptionHandler();
        // 获取系统默认的UncaughtException处理器
        Thread.setDefaultUncaughtExceptionHandler(this);
        // 设置该CrashHandler为程序的默认处理器
    }
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        boolean hasHandle=handleException(ex);
        //是否处理
        if (!hasHandle && defalutHandler!=null){
            defalutHandler.uncaughtException(thread,ex);
            //如果用户没有处理则让系统默认的异常处理器来处理
        }else{
            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                Log.e(TAG, "error : ", e);
            }
            android.os.Process.killProcess(android.os.Process.myPid());
            System.exit(1);
        }


    }


    private boolean handleException(final Throwable ex){
        if (ex==null){
            return false;
        }
        new Thread(){
            @Override
            public void run() {
                Looper.prepare();

                ex.printStackTrace();

                String err="["+ex.getMessage()+"]";
                Toast.makeText(mContext, "程序出现异常,5秒后自动退出", Toast.LENGTH_LONG).show();
                Looper.loop();
            }
        }.start();
        String str = collectDeviceInfo(ex);
        // 收集设备参数信息,日志信息
        saveCrashInfoToFile(str);
        // 保存日志文件
        return true;
    }

    /**
     * 收集设备信息,日志信息
     * @param ex Throwable
     * @return 收集的信息
     */
    private String collectDeviceInfo(Throwable ex){
        Log.e(TAG,"collectDeviceInfo:"+ex.getMessage());
        StringBuilder builder=new StringBuilder();

        return builder.toString();
    }

    /**
     * 保存出错信息
     * @param error 待保存的出错信息
     */
    private void saveCrashInfoToFile(String error){
        Log.e(TAG,"saveCrashInfoToFile:"+error);
    }
}

程序入口进行初始化,之后未捕获的异常均由此类处理。
CrashHandler.getInstance().init(this);

  • Resources类中的getIdentifier(name, defType, defPackage)方法,根据资源名称获取其ID,做UI时经常用到。

/**
 * 根据资源名获得资源id
 * User:lizhangqu(513163535@qq.com)
 * Date:2015-08-04
 * Time: 15:18
 */
public class ResourcesUtil {
    public static final String LAYTOUT="layout";
    public static final String DRAWABLE="drawable";
    public static final String MIPMAP="mipmap";
    public static final String MENU="menu";
    public static final String RAW="raw";
    public static final String ANIM="anim";
    public static final String STRING="string";
    public static final String STYLE="style";
    public static final String STYLEABLE="styleable";
    public static final String INTEGER="integer";
    public static final String ID="id";
    public static final String DIMEN="dimen";
    public static final String COLOR="color";
    public static final String BOOL="bool";
    public static final String ATTR="attr";
    //TODO please add other strings by yourself
    public static int getResourceId(Context context,String name,String type){
        Resources resources=null;
        PackageManager pm=context.getPackageManager();
        try {
            resources=context.getResources();
            return resources.getIdentifier(name, type, context.getPackageName());
        } catch (Exception e) {
            e.printStackTrace();
        }
        return 0;
    }
}

比如获得主布局的id
ResourcesUtil.getResourceId(getApplicationContext(),"activity_main",ResourcesUtil.LAYTOUT);

  • View中的isShown()方法,以前都是用view.getVisibility() == View.VISIBLE来判断的,但是与这个函数还是有区别的。也就是只有当view本身和它的所有父容器都是visible时,isShown()才返回TRUE。而平常我们调用if(view.getVisibility() == View.VISIBLE)只是对view本身而不对父容器的可见性进行判断。
  • 集合与数组的转化,Arrays类中的asList(T… array)方法,数组转List集合;反过来List.toArray();
  • android.text.format.Formatter类中formatFileSize(Context, long)方法,用来格式化文件大小(B → KB → MB → GB)

android.text.format.Formatter.formatFileSize(getApplicationContext(),1024);
//返回1.00KB
android.text.format.Formatter.formatFileSize(getApplicationContext(),1024*1024)
//返回1.00MB

  • android.media.ThumbnailUtils类,用来获取媒体(图片、视频)缩略图,该类从Android 2.2开始系统新增,不向下兼容

/** 
 * 创建一张视频的缩略图 
 * 如果视频已损坏或者格式不支持可能返回null 
 *  
 * @param filePath 视频文件路径  如:/sdcard/android.3gp 
 * @param kind kind可以为MINI_KIND或MICRO_KIND 
 *  
 */ 
ThumbnailUtils.createVideoThumbnail(filePath,kind);
/** 
 *  
 * 创建一个指定大小的缩略图 
 * @param source 源文件(Bitmap类型) 
 * @param width  压缩成的宽度 
 * @param height 压缩成的高度 
 */
ThumbnailUtils.extractThumbnail(source , width, height);
/** 
 * 创建一个指定大小居中的缩略图 
 *  
 * @param source 源文件(Bitmap类型) 
 * @param width  输出缩略图的宽度 
 * @param height 输出缩略图的高度 
 * @param options 如果options定义为OPTIONS_RECYCLE_INPUT,则回收@param source这个资源文件 
 * (除非缩略图等于@param source) 
 *  
 */
ThumbnailUtils.extractThumbnail(source , width, height,options);

  • 格式化字符串,可以使用String类的format(String,Object…)方法,如果要格式化资源文件strings.xml中的字符串,可以使用getResources().getString(int,Object…)方法

String.format("money:¥%.2f",1.00);

<resources>
    <string name="format">money:$%.2f</string>
</resources>

getResources().getString(R.string.format,1.00);

  • View类中的三个方法:callOnClick(),performClick(),performLongClick(),可以直接用于触发View的点击事件,不用我们手动点击才触发;

findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        Log.e("TAG", "onClick");

    }
});
findViewById(R.id.btn).setOnLongClickListener(new View.OnLongClickListener() {
    @Override
    public boolean onLongClick(View v) {
        Log.e("TAG", "onLongClick");
        return true;
    }
});
findViewById(R.id.btn).callOnClick();
findViewById(R.id.btn).performClick();
findViewById(R.id.btn).performLongClick();

至于callOnClick()和performClick的区别,相信看过源码后你会一目了然。
    /**
     * Call this view's OnClickListener, if it is defined.  Performs all normal
     * actions associated with clicking: reporting accessibility event, playing
     * a sound, etc.
     *
     * @return True there was an assigned OnClickListener that was called, false
     *         otherwise is returned.
     */    
     public boolean performClick() {
        sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);

        ListenerInfo li = mListenerInfo;
        if (li != null && li.mOnClickListener != null) {
            playSoundEffect(SoundEffectConstants.CLICK);
            li.mOnClickListener.onClick(this);
            return true;
        }

        return false;
    }

    /**
     * Directly call any attached OnClickListener.  Unlike {@link #performClick()},
     * this only calls the listener, and does not do any associated clicking
     * actions like reporting an accessibility event.
     *
     * @return True there was an assigned OnClickListener that was called, false
     *         otherwise is returned.
     */
    public boolean callOnClick() {
        ListenerInfo li = mListenerInfo;
        if (li != null && li.mOnClickListener != null) {
            li.mOnClickListener.onClick(this);
            return true;
        }
        return false;
    }

  • TextUtils类中的isEmpty(String)方法,判断字符串是否为null或”“,不要再自己写判断字符串非空的代码了。

String str = null;
String str1 = "";
String str2 = "a";
Log.e("TAG", TextUtils.isEmpty(str)+" "+TextUtils.isEmpty(str1)+" "+TextUtils.isEmpty(str2));
//输出true true false

类似的方法还有TextUtils.isDigitsOnly()是否是纯数字

  • TextView类中的append(String)方法,添加文本,不要再使用getText()方法拿到旧的字符串再拼接,拼接好了之后再调用setText()方法了

TextView textview= (TextView) findViewById(R.id.tv);
textview.setText("aaa");
textview.append("bbb");

  • View类中的getDrawingCache()等一系列方法,目前只知道可以用来截图

    /**
     * / 获取指定Activity的截屏,保存到png文件
     *
     * @param activity activity
     * @return 截屏Bitmap
     */
    private static Bitmap takeScreenShot(Activity activity) {
        // View是你需要截图的View
        View view = activity.getWindow().getDecorView();
        view.setDrawingCacheEnabled(true);
        view.buildDrawingCache();
        Bitmap b1 = view.getDrawingCache();

        // 获取状态栏高度
        Rect frame = new Rect();
        activity.getWindow().getDecorView().getWindowVisibleDisplayFrame(frame);
        int statusBarHeight = frame.top;
        Log.i("TAG", "" + statusBarHeight);

        // 获取屏幕长和高
        int width = activity.getWindowManager().getDefaultDisplay().getWidth();
        int height = activity.getWindowManager().getDefaultDisplay()
                .getHeight();
        Bitmap b = Bitmap.createBitmap(b1, 0, statusBarHeight, width, height
                - statusBarHeight);
        view.destroyDrawingCache();
        return b;
    }

  • DecimalFormat类,用于字串格式化包括指定位数、百分数、科学计数法等

DecimalFormat df=new DecimalFormat("0.0");
df.format(12.34);

  • System类中的arraycopy(src, srcPos, dest, destPos, length)方法,用来copy数组;Arrays.copyOf()里的一系列方法也是间接调用System.arraycopy()方法
  • Fragment类中的onHiddenChanged(boolean)方法,使用FragmentTransaction中的hide(),show()时貌似Fragment的其它生命周期方法都不会被调用

调用hide或者show你就会发现fragment的生命周期不走了!onPause方法不调用了!onResume只调用一次!这时此时方法onHiddenChanged派上用场
当fragment隐藏时,该方法会调用传入参数为true表示该fragment被隐藏了,
当fragment调用了show方法后,该方法传入的参数为false,表示该fragment正在显示
所以总结起来,如果使用hide/show方法来控制fragment的使用时,原本需要在onResume以及onPause方法做的事情就可以迁移到onHiddenChanged时进行管理
@Override
public void onHiddenChanged(boolean hidden) {
    super.onHiddenChanged(hidden);
    if(hidden){
        this.onPause();
    }else{
        this.onResume();
    }

}

  • Activity类中的onWindowFocusChanged(boolean)方法,使用一个view的getWidth() getHeight() 方法来获取该view的宽和高,返回的值却为0。
  • 如果这个view的长宽很确定不为0的话,那很可能是你过早的调用这些方法,也就是说在这个view被加入到rootview之前你就调用了这些方法,返回的值自然为0.,解决该问题的方法有很多,主要就是延后调用这些方法。可以试着在onWindowFocusChanged()里面调用这些方法。
  • View类中的getLocationInWindow(int[])方法和getLocationOnScreen(int[])方法,获取View在窗口/屏幕中的位置

TextView tv= (TextView) findViewById(R.id.tv);
int loc[]=new int[2];
tv.getLocationInWindow(loc);
Log.e("TAG",loc[0]+" "+loc[1]);

  • TextView类中的setTransformationMethod(TransformationMethod)方法,可用来实现“显示密码”功能;

TextView tv= (TextView) findViewById(R.id.tv);
tv.setText("123456");
tv.setTransformationMethod(PasswordTransformationMethod.getInstance());

  • TextWatcher接口,用来监听文本输入框内容的改变,可以做的事很多
  • View类中的setSelected(boolean)方法结合android:state_selected=”“用来实现图片选中效果
  • Surface设置透明,但是会挡住其它控件

SurfaceView.setZOrderOnTop(true);
SurfaceView.getHolder().setFormat(PixelFormat.TRANSLUCENT);

ListView或GridView类中的setFastScrollEnabled(boolean)方法,用来设置快速滚动滑块是否可见,当然前提是item够多

PageTransformer接口,用来自定义ViewPager页面切换动画,用setPageTransformer(boolean, PageTransformer)方法来进行设置

apache提供的一系列jar包:commons-lang.jar,commons-collections.jar,commons-beanutils.jar等,里面很多方法可能是你曾经用几十几百行代码实现过的,但是执行效率或许要差很多,比如:ArrayUtils,StringUtils……

AndroidTestCase类,Android单元测试

Activity类中的onNewIntent(intent)方法,具体看这篇文章Android:onNewIntent()触发机制及注意事项

Activity.startActivities() 常用于在应用程序中间启动其他的Activity。和startActivity()类似,startActivities也是界面跳转,但是传入的intent是一个数组,也就是说是多个。假设我传入的是两个intent: I1和I2,则调用startActivities之后,直接到I2界面,按返回键,到I1界面。其中到I2的过程中,不会经过I1界面,也就是说,不过存在I1的生命周期之说。

Html.fromHtml() 用于生成一个Html,参数可以是一个字符串.但是它不是很快,所以不要经常去用.取而代之的是请手动构建 Spannable 来替换 Html.fromHtml,但是它对渲染从 web 上获取的文字还是很不错的。

TextView.setError() 在验证用户输入的时候很棒

Build.VERSION_CODES 这个标明了当前的版本号,在处理兼容性问题的时候经常会用到.点进去可以看到各个版本的不同特性

Log.getStackTraceString() 方便的日志类工具,方法Log.v()、Log.d()、Log.i()、Log.w()和Log.e()都是将信息打印到LogCat中,有时候需要将出错的信息插入到数据库或一个自定义的日志文件中,那么这种情况就需要将出错的信息以字符串的形式返回来,也就是使用static String getStackTraceString(Throwable tr)方法的时候。

LayoutInflater.from() 顾名思义,用于Inflate一个layout,参数是layout的id.很多地方都会用到
LayoutInflater.from(getApplicationContext()).inflate(int resource, ViewGroup root, boolean attachToRoot)

ViewConfiguration.getScaledTouchSlop() 使用 ViewConfiguration 中提供的值以保证所有触摸是的交互都统一的。这个方法获取的值表示:用户的手滑动这个距离后,才判定为正在进行滑动.当然这个值也可以自己来决定.但是为了一致性,还是使用标准的值较好。

PhoneNumberUtils.convertKeypadLettersToDigits 顾名思义.将字母转换为数字,类似于T9输入法
String abcd = PhoneNumberUtils.convertKeypadLettersToDigits("abcd");
Log.e("TAG",abcd);
//结果为2223

Context.getCacheDir() 获取缓存数据文件夹的路径,这个路径通常在SD卡上(这里的SD卡指的是广义上的SD卡,包括外部存储和内部存储)Adnroid/data/您的应用程序包名/cache/ 下面.测试的时候,可以去这里面看是否缓存成功.缓存在这里的好处是:不用自己再去手动创建文件夹,不用担心用户把自己创建的文件夹删掉,在应用程序卸载的时候,这里会被清空,使用第三方的清理工具的时候,这里也会被清空。

ArgbEvaluator 用于处理颜色的渐变。在使用动画的时候可能用的比较多。可以看下他的实现
public Object evaluate(float fraction, Object startValue, Object endValue) {
        int startInt = (Integer) startValue;
        int startA = (startInt >> 24) & 0xff;
        int startR = (startInt >> 16) & 0xff;
        int startG = (startInt >> 8) & 0xff;
        int startB = startInt & 0xff;

        int endInt = (Integer) endValue;
        int endA = (endInt >> 24) & 0xff;
        int endR = (endInt >> 16) & 0xff;
        int endG = (endInt >> 8) & 0xff;
        int endB = endInt & 0xff;

        return (int)((startA + (int)(fraction * (endA - startA))) << 24) |
                (int)((startR + (int)(fraction * (endR - startR))) << 16) |
                (int)((startG + (int)(fraction * (endG - startG))) << 8) |
                (int)((startB + (int)(fraction * (endB - startB))));
    }

ContextThemeWrapper 方便在运行的时候修改主题。这里以改变系统自带Dialog字体大小为例
//将一个style的parent设置为@android:style/Theme.Dialog
//修改其中的 <item name="android:textSize">30sp</item>
//利用context和该style生成ContextThemeWrapper
//利用ContextThemeWrapper生产Builder对象
ContextThemeWrapper contextThemeWrapper = new ContextThemeWrapper(MainActivity.this, R.style.dialog);
AlertDialog.Builder builder =  new AlertDialog.Builder(contextThemeWrapper);
Dialog dialog=builder.create();

Space 它是Android 4.0中新增的一个控件,它实际上可以用来分隔不同的控件,其中形成一个空白的区域.这是一个轻量级的视图组件,它可以跳过Draw,对于需要占位符的任何场景来说都是很棒的。
<Space
    android:layout_width="match_parent"
    android:layout_height="10dp"/>

ValueAnimator.reverse() 用在动画中,将动画逆向。

EditText类的setKeyListener(KeyListener)方法,设置DigitsKeyListener类将只能输入数字,通过DigitsKeyListener.getInstance(String accepted)方法即可指定EditText可输入的字符集。

DateUtils.formatDateTime() 用来进行区域格式化工作,输出格式化和本地化的时间或者日期。
android.text.format.DateUtils.formatDateTime(getApplicationContext(),System.currentTimeMillis(), DateUtils.FORMAT_SHOW_DATE|DateUtils.FORMAT_SHOW_YEAR|DateUtils.FORMAT_SHOW_TIME);

Linkify.addLinks() 在Text上添加链接。很实用。比如将TextView中的超链接识别为可点击进入浏览器的链接。
TextView tv= (TextView) findViewById(R.id.tv);
Linkify.addLinks(tv,Linkify.WEB_URLS);

StaticLayout 在自定义 View 中渲染文字的时候很实用。
使用Canvas的drawText绘制文本是不会自动换行的,即使一个很长很长的字符串,drawText也只显示一行,超出部分被隐藏在屏幕之外。可以逐个计算每个字符的宽度,通过一定的算法将字符串分割成多个部分,然后分别调用drawText一部分一部分的显示, 但是这种显示效率会很低。
StaticLayout是android中处理文字换行的一个工具类,StaticLayout已经实现了文本绘制换行处理

Activity.onBackPressed() 很方便的管理back键的方法,有时候需要自己控制返回键的事件的时候,可以重写一下。比如加入 “点两下back键退出” 功能。

GestureDetector 用来监听和相应对应的手势事件,比如点击,长按,慢滑动,快滑动,用起来很简单,比你自己实现要方便许多。

ActivityManager.getMemoryClass() 告诉你你的机器还有多少内存,在计算缓存大小的时候会比较有用。

ViewStub 它是一个初始化不做任何事情的 View,但是之后可以载入一个布局文件。在慢加载 View 中很适合做占位符。

SystemClock.sleep() 这个方法在保证一定时间的 sleep 时很方便,通常用来进行 debug 和模拟网络延时。

Thread.sleep()是java提供的函数。在调用该函数的过程中可能会发生InterruptedException异常。
SystemClock.sleep()是android提供的函数。在调用该函数的过程中不会发生InterruptedException异常。

DisplayMetrics.density 这个方法你可以获取设备像素密度,大部分时候最好让系统来自动进行缩放资源之类的操作,但是有时候控制的效果会更好一些.(尤其是在自定义View的时候)。

Pair.create(),这个类在v4包下也存在,内部就两个泛型对象,一个叫first,一个叫second,可以类比map,一个为key,一个为value,但是这个Pair不是key,value,而是一组数据

Fragment.setArguments,因为在构建 Fragment 的时候不建议加参数,所以这是个很好的东西,可以在创建 Fragment 之前设置参数
public class BlankFragment extends Fragment {
    private static final String ARG_PARAM1 = "param1";
    private static final String ARG_PARAM2 = "param2";

    private String mParam1;
    private String mParam2;

    public static BlankFragment newInstance(String param1, String param2) {
        BlankFragment fragment = new BlankFragment();
        Bundle args = new Bundle();
        args.putString(ARG_PARAM1, param1);
        args.putString(ARG_PARAM2, param2);
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (getArguments() != null) {
            mParam1 = getArguments().getString(ARG_PARAM1);
            mParam2 = getArguments().getString(ARG_PARAM2);
        }
    }

}

DialogFragment.setShowsDialog () 这是一个很巧妙的方式,DialogFragment 可以作为Dialog显示。可以参考这篇文章Android 官方推荐 : DialogFragment 创建对话框

FragmentManager.enableDebugLogging () 在需要观察 Fragment 状态的时候会有帮助。可以通过getFragmentManager().enableDebugLogging(true);来提供相关的debug功能。

LocalBroadcastManager 这个会比全局的 broadcast 更加安全,简单,快速。一个简单的应用就是退出程序。
/**
 * 全局Application基类
 * User:lizhangqu(513163535@qq.com)
 * Date:2015-07-22
 * Time: 09:35
 */
public class BaseApplication extends Application {
    public final static String ACTION_EXIT_APP = "package.exit";
    private static LocalBroadcastManager mLocalBroadcatManager;
    private static Context mContext;
    private static BaseApplication instance;

    public static Context getContext() {
        return mContext;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        instance = this;
        mContext = this.getApplicationContext();
        CorePageManager.getInstance().init(this);
    }

    /**
     * 发送本地广播退出程序
     */
    public void exitApp() {
        Intent intent = new Intent();
        intent.setAction(ACTION_EXIT_APP);
        intent.addCategory(Intent.CATEGORY_DEFAULT);
        BaseApplication.getLocalBroadcastManager().sendBroadcast(intent);
        BaseActivity.unInit();
    }

    /**
     * 获得LocalBroadcastManager对象
     * @return LocalBroadcastManager对象
     */
    public static LocalBroadcastManager getLocalBroadcastManager() {
        if (mLocalBroadcatManager == null) {
            mLocalBroadcatManager = LocalBroadcastManager.getInstance(mContext);
        }
        return mLocalBroadcatManager;
    }
}

public class BaseActivity extends FragmentActivity{
    public final static String ACTION_EXIT_APP = "package.exit";
    /**
     * 仅用于接受应用退出广播,程序退出时有机会做一些必要的清理工作
     */
    private BroadcastReceiver mExitReceiver = new BroadcastReceiver() {

        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
            if (action.equals(ACTION_EXIT_APP)) {
                Log.d(TAG,"exit from broadcast");
                finish();
            }
        }
    };



    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_base);

        IntentFilter filter = new IntentFilter();
        filter.addAction(Config.ACTION_EXIT_APP);
        filter.addCategory(Intent.CATEGORY_DEFAULT);
        BaseApplication.getLocalBroadcastManager().registerReceiver(mExitReceiver, filter);
        //注册本地广播,接收程序退出广播
    }


}

Application.registerActivityLifecycleCallbacks 就是注册 Activity 的生命周期的一些回调方法,就是一个方便的工具
Application通过此接口提供了一套回调方法,用于让开发者对Activity的生命周期事件进行集中处理。
以往若需监测Activity的生命周期事件代码,你可能是这样做的,重写每一个Acivity的onResume(),然后作统计和处理。
@Override  
protected void onResume() {  
  super.onResume();  
  //TODO 处理和统计代码  
  Log.v(TAG, "onResume");  
  Logger.v(TAG, "onResume");  
  Logging.v(TAG, "onResume");  
  ...  
}

ActivityLifecycleCallbacks接口回调可以简化这一繁琐过程,在一个类中作统一处理

android.app.Application.ActivityLifecycleCallbacks 它要求API 14+ (Android 4.0+),在我们自定义的Application注册回调。
public void onCreate() {  
  super.onCreate();  
  this.registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {  

    @Override  
    public void onActivityStopped(Activity activity) {  
        Logger.v(activity, "onActivityStopped");  
    }  

    @Override  
    public void onActivityStarted(Activity activity) {  
        Logger.v(activity, "onActivityStarted");  
    }  

    @Override  
    public void onActivitySaveInstanceState(Activity activity, Bundle outState) {  
        Logger.v(activity, "onActivitySaveInstanceState");  
    }  

    @Override  
    public void onActivityResumed(Activity activity) {  
        Logger.v(activity, "onActivityResumed");  
    }  

    @Override  
    public void onActivityPaused(Activity activity) {  
        Logger.v(activity, "onActivityPaused");  
    }  

    @Override  
    public void onActivityDestroyed(Activity activity) {  
        Logger.v(activity, "onActivityDestroyed");  
    }  

    @Override  
    public void onActivityCreated(Activity activity, Bundle savedInstanceState) {  
        Logger.v(activity, "onActivityCreated");  
    }  
  });  
}; 

这里只是使用Log日志工具作简要测试,如需满足较复杂的统计或调试需求时,此法可能会大大减少插入代码量,提高效率,更多详情请看这个项目AndroidLifecyle

versionNameSuffix这个 gradle 设置可以让你在基于不同构建类型的 manifest 中修改版本名这个属性,例如,如果需要在在 debug 版本中以”-SNAPSHOT”结尾,那么就可以轻松的看出当前是 debug 版还是 release 版。
buildTypes {
    release {
        versionNameSuffix "sample" 
    }

}

-nodpi 在没有特别定义的情况下,很多修饰符(-mdpi,-hdpi,-xdpi等等)都会默认自动缩放 assets/dimensions,有时候我们需要保持显示一致,这种情况下就可以使用 -nodpi。

Activity.recreate () 强制让 Activity 重建

BroadcastRecevier.setDebugUnregister ()什么用不知道,调试用的。

android:weightSum (LinearLayout)这个很有用,在百分比布局没有出来前,基本上就是通过这个属性以及layout_weight属性来间接达到百分比。

android:duplicateParentState (View) 此方法可以使得子 View 可以复制父 View 的状态。比如如果一个 ViewGroup 是可点击的,那么可以用这个方法在它被点击的时候让它的子 View 都改变状态。

android:tileMode (BitmapDrawable)可以指定图片使用重复填充的模式。

android:enterFadeDuration/android:exitFadeDuration (Drawables)此属性在 Drawable 具有多种状态的时候,可以定义它展示前的淡入淡出效果。

android:scaleType (ImageView)定义在 ImageView 中怎么缩放/剪裁图片,一般用的比较多的是centerCrop和centerInside。

Merge此标签可以在另一个布局文件中包含别的布局文件,而不用再新建一个 ViewGroup,对于自定义 ViewGroup 的时候也需要用到;可以通过载入一个带有标签的布局文件来自动定义它的子部件。

AtomicFile通过使用备份文件进行文件的原子化操作的类

ViewDragHelper视图拖动是一个比较复杂的问题。这个类可以帮助解决不少问题。可以参考这篇文章Android ViewDragHelper完全解析 自定义ViewGroup神器

**PopupWindow**Android到处都在使用PopupWindow ,甚至你都没有意识到(标题导航条ActionBar,自动补全AutoComplete,编辑框错误提醒Edittext Errors)。这个类是创建浮层内容的主要方法。

**SparseArray**Map的高效优化版本。推荐了解姐妹类SparseBooleanArray、SparseIntArray和SparseLongArray。在编写通用适配器的时候SparseArray可能会用到

PackageManager.setComponentEnabledSetting()可以用来启动或者禁用程序清单中的组件。对于关闭不需要的功能组件是非常赞的,比如关掉一个当前不用的广播接收器。

SQLiteDatabase.yieldIfContendedSafely()让你暂时停止一个数据库事务, 这样你可以就不会占用太多的系统资源。

Environment.getExternalStoragePublicDirectory()还是那句话,用户期望在SD卡上得到统一的用户体验。用这个方法可以获得在用户设备上放置指定类型文件(音乐、图片等)的正确目录。

Context.getExternalFilesDir()申请了SD卡写权限后,你可以在SD的任何地方写数据,把你的数据写在设计好的合适位置会更加有礼貌。这样数据可以及时被清理,也会有更好的用户体验。此外,Android 4.0 Kitkat中在这个文件夹下写数据是不需要权限的,每个用户有自己的独立的数据存储路径。该API从V8才开始支持。

View.generateViewId()每次我都想要推荐动态生成控件的ID。需要注意的是,不要和已经存在的控件ID或者其他已经生成的控件ID重复。

ActivityManager.clearApplicationUserData() 一键清理你的app产生的用户数据,可能是做用户退出登录功能,有史以来最简单的方式了。以前自己的做法真是不忍直视啊。

ActivityOptions方便的定义两个Activity切换的动画。 使用ActivityOptionsCompat 可以很好解决旧版本的兼容问题。使用ActivityOptionsCompat 类可以很方便的实现Material Design的切换动画

ViewParent.requestDisallowInterceptTouchEvent() Android系统触摸事件机制大多时候能够默认处理,不过有时候你需要使用这个方法来剥夺父级控件的控制权。

Fragment 的 setUserVisibleHint方法,可实现 fragment 对用户可见时才加载资源(延迟加载)

IntentService一个可以干完活后自己去死且不需要我们去管理子线程的Service

Executors. newSingleThreadExecutor()单线程顺序执行的任务队列

**android:animateLayoutChanges=”true”**LinearLayout中添加View的动画的办法,支持通过setLayoutTransition()自定义动画

GradientDrawable渐变,可实现阴影效果

PointF,graphics包中的一个类,我们经常见到在处理Touch事件的时候分别定义一个downX,一个downY用来存储一个坐标,如果坐标少还好,如果要记录的坐标过多那代码就不好看了。用PointF(float x, float y);来描述一个坐标点会清楚很多。

StateListDrawable,定义Selector通常的办法都是xml文件,但是有的时候我们的图片资源可能是从服务器动态获取的,比如很多app所谓的皮肤,这种时候就只能通StateListDrawable来完成了,各种addState即可。

includeFontPadding=”false”,TextView默认上下是有一定的padding的,有时候我们可能不需要上下这部分留白,加上它即可。

onTrimMemory,在Activity中重写此方法,会在内存紧张的时候回调(支持多个级别),便于我们主动的进行资源释放,避免OOM。

Fragment在onAttach方法中接收回调
@Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            mPageSelectedListener = (PageSelectedListener) activity;
            mMenuBtnOnclickListener = (MenuBtnOnClickListener) activity;
            mCommitBtnOnClickListener = (CommitBtnOnClickListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString() + "must implements listener");
        }
    }

通过 WindowManager.addView 在其他app界面添加一个view时,经常会无法显示,特别在miui,emui固件上,需要指定type为LayoutParams.TYPE_TOAST。

Paint.setXfermode(porterDuffXfermode),在ApiDemo里面有专门的介绍,实现了穿透,叠加,覆盖等多种绘制效果,非常实用。在自定义View中用的比较多。

通过View.getDrawingCache()可以获取截图,但是需要setDrawingCacheEnabled(true)频繁使用可能会oom,还有一种方法直接用canvas
Bitmap bm = Bitmap.createBitmap((int) (w * scale), (int) (h*scale), Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas();
canvas.setBitmap(bm);
View.draw(canvas);
return bm;

由于fragment的缓存机制决定的。默认情况下,viewpager切换页面时会缓存上一个页面,非相邻页面被销毁,可以使用viewPaper.setOffscreenPageLimit()函数来解决

support library 里的任何东西你都值得去看看

fragment嵌套时,内部fragment的manager通过getChildFragmentManager()获得

布局中,view.bringTofont(),把该view在层叠布局中置于最前面。通样的viewgroup的bringChildtoFont(),都是一样的效果。
分享到:
评论

相关推荐

    那些“相见恨晚”的APP.pdf

    那些“相见恨晚”的APP.pdf

    让你相见恨晚的十个Python骚操作

    Python 的代码之所以这么优雅,完全归功于其独有的特性,只要你掌握了这些特性,同样可以写出像诗一样的代码。 下面我们就一起看看 Python 中的骚操作。 0x00 Hello World 对于大部分程序员来说,第一个程序应该都...

    相见恨晚经典话语.docx

    "相见恨晚经典话语" 本文涵盖了十五个经典的相见恨晚话语,每一个话语都包含了深刻的哲理和人生经验。这些话语涵盖了人生的多个方面,如友谊、爱情、人际关系、自我成长等。 1. 《用人之道》:高人的面子,发挥...

    50个相见恨晚的CAD技巧总结

    标题:50个相见恨晚的CAD技巧总结 描述:在现代工程设计领域,CAD(计算机辅助设计)软件是设计人员不可或缺的工具之一。随着技术的不断发展,掌握一系列高效的CAD技巧能够显著提升工作效率,减少设计过程中的不便...

    Android 高仿QQ 沉浸式状态栏

    接下来我就给大家介绍怎样快速打造沉浸式状态栏吧,虽然感觉有点相见恨晚,但其实不完! 一:何为沉浸式状态栏? 沉浸式状态栏是Google从Android 4.4开始,给我们开发者提供的一套能透明的系统ui样式,这样样式是给...

    2009-相见恨晚的30句话-@Lonely Fish.zip

    标题“2009-相见恨晚的30句话-@Lonely Fish.zip”和描述“2009-相见恨晚的30句话-@Lonely Fish”看似与IT行业知识关联不大,但我们可以从这个标题中推测这可能是一个包含了一些人生感悟或者智慧格言的资料包。...

    一些相见恨晚的 JavaScript 技巧

    ES6的解构赋值允许你从对象或数组中提取值并直接赋给变量,如`var {name: bandName, year: releaseYear} = band;`,这在处理复杂数据结构时非常有用。 通过掌握这些JavaScript技巧,你可以写出更高效、更简洁的...

    读书笔记:记录一些看到过的相见恨晚的技术文章, 收录原则 通俗易懂, 详细..zip

    读书笔记:记录一些看到过的相见恨晚的技术文章, 收录原则 通俗易懂, 详细.

    60 个相见恨晚的神器工具!.zip

    计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料计算机技术、IT咨询、人工智能AI理论介绍,学习...

    Django JavaScript AJAX and jQuery使用教程

    在描述中,读者表达了对这本书的高度评价,称其为“好书”,并提到书籍来自国外作者,给读者带来了很多收获,并有一种“相见恨晚”的感觉。这表明该书内容质量高,深受读者喜欢,特别是对于希望在Django中使用AJAX和...

    idea zookeeper插件

    **标题与描述解析** 标题提及的是"Idea Zookeeper插件",这指的是在JetBrains的IntelliJ IDEA集成开发环境中使用的Zookeeper插件...对于那些需要频繁与Zookeeper打交道的开发者来说,它的确是一款“相见恨晚”的工具。

    手工清理病毒(偶之遇,相见恨晚之感,亦有受益匪浅)

    当大家看到这个题目的时候...笔者写这个文章的目的就是让所有菜鸟在面对病毒的时候能轻而易举的狙杀掉它,而不是重装系统,或者在重装N次系统以后无奈的选择格式化,结果却依然无法将讨厌的病毒驱逐出你可怜的电脑。

    系统增强文件管理器 Total Commander 9.51 + x64 Final 中文.zip

    在接触、了解这个完美的工具后,相信你会和我一样有相见恨晚的感觉。 Total Commander 的最大特点是由两个列表窗口组成,这种设计避免了 Windows资 源管理器目录树在文件操作的一系列弊端: 首先,资源管理器在选择...

    Total commander 7.02a

    在接触、了解这个完美的工具后,相信你会和我一样有相见恨晚的感觉。  Total Commander的最大特点是由两个列表窗口组成,这种设计避免了windows资源治理器目录树在文件操作的一系列弊端:  首先,资源治理器在...

    FSCapture(截图录屏工具)

    在本文中,我们将深入探讨FSCapture的主要特点、功能以及如何使用它来提升你的工作或学习效率。 首先,FSCapture的**截屏功能**非常全面。它支持多种截图模式,包括全屏截图、窗口截图、区域截图、固定大小截图以及...

    导游带团脑筋急转多篇弯题目大全及答案大全.pdf

    导游带团是旅游中不可或缺的重要环节,它不仅担负着引领游客游览景点的任务,更承载着传播文化、娱乐大众的使命。在这一过程中,各种互动游戏,尤其是脑筋急转弯这种集趣味性与智慧性于一体的游戏,显得尤为重要。...

    C++矩阵库 Eigen 快速入门

    近需要用 C++ 做一些数值计算,之前一直采用Matlab 混合编程的方式处理矩阵运算,非常麻烦,直到发现了 Eigen 库,简直相见恨晚,好用哭了。 Eigen 是一个基于C++模板的线性代数库,直接将库下载后放在项目目录下,...

    全国大学生电子设计竞赛教程 基于TI器件设计方法

    《全国大学生电子设计竞赛教程:基于TI器件设计方法》是一本专门为参与全国大学生电子设计竞赛的学生准备的指导书籍。本书的核心在于介绍了如何利用德州仪器(TI)的集成电路进行电子产品的设计与实现,从而帮助参赛...

Global site tag (gtag.js) - Google Analytics