Android小经验

你应该知道的那些Android小经验

1.枚举提供类型安全

Android代码替代枚举的正确之道

Paste_Image.png

2.匿名token

作用:防止被人恶意刷接口。
情景:当应用,在非登录状态下也可以使用的情况下,需要匿名token。
解决:要去服务器请求一个匿名token

3微信分享没有回调

.wxapi.WXEntryActivity必须在包名下面。
微信分享没有回调问题

4android中网络请求中页面关闭了会怎么样?

问题:android中网络请求中页面关闭了会怎么样
以activity为例子,
1.可以在onDestroy里cancel掉所有网络请求。
2.也可以采用虚引用的方式
虚引用 :不能单独使用,主要是用于追踪对象被垃圾回收的状态。通过PhantomReference类和引用队列ReferenceQueue类联合使用实现


可以每个请求的时候标志一个tag(一个页面一个 bject tag = new Object();不能采用当前activity.this,因为你的回调是个内部匿名类会默认持有当前activity,内部匿名类不是消失,activity得不到释放),tag的hashcode值和应用队列建立一个虚引用,需引用里又个list<call>,后台开启一个常驻线程,检测。当Gc时,就能得到ReferenceQueue得到PhantomReference,把所有没执行的call都cancel

 * 为每个  NetCall 建立一个与 Tag(Object) 的引用. 当 Tag 被 JVM 回收后,自动关闭 NetCall. * */
public class BJNetResourceManager {   
 private Object mDefaultObject = new Object();    
private ReferenceQueue<Object> mReferenceQueue = new ReferenceQueue<>();    
private Map<Integer, ResourceReference> mResourceRefMap = new ConcurrentHashMap<>();    
private ResourceCheckThread mResourceCheckThread = new ResourceCheckThread();    
public BJNetResourceManager() {       mResourceCheckThread.start();    
}    
public void release() {    }    
public void addNetCall(Object tag, BJNetCall call) {        int key = mDefaultObject.hashCode();        if (tag != null) {           key = tag.hashCode();        }        if (mResourceRefMap.containsKey(key)) {            ResourceReference reference = mResourceRefMap.get(key);            reference.add(call);        } else {            ResourceReference reference = new ResourceReference(tag == null ? mDefaultObject : tag, mReferenceQueue);            reference.add(call);            mResourceRefMap.put(key, reference);        }    }    public void removeNetCall(Object tag, BJNetCall call) {        int key = mDefaultObject.hashCode();        if (tag != null) {            key = tag.hashCode();        }        if (mResourceRefMap.containsKey(key)) {            ResourceReference reference = mResourceRefMap.get(key);            reference.remove(call);        } else {        }    }    public void removeAll(Object tag) {        int key = mDefaultObject.hashCode();        if (tag != null) {            key = tag.hashCode();        }        if (mResourceRefMap.containsKey(key)) {            ResourceReference reference = mResourceRefMap.get(key);            reference.cancelAll();            mResourceRefMap.remove(key);        }    }    private static class ResourceReference extends PhantomReference<Object> {        private int tagId;        private String tagName;        private List<BJNetCall> list;        /**         * Constructs a new phantom reference and registers it with the given         * reference queue. The reference queue may be {@code null}, but this case         * does not make any sense, since the reference will never be enqueued, and         * the {@link #get()} method always returns {@code null}.         *         * @param r the referent to track         * @param q the queue to register the phantom reference object with         */        public ResourceReference(Object r, ReferenceQueue<? super Object> q) {            super(r, q);            this.tagId = r.hashCode();            this.tagName = r.getClass().getSimpleName();            this.list = Collections.synchronizedList(new LinkedList<BJNetCall>());        }        public int getTagId() {            return tagId;        }        public String getTagName() {            return tagName;        }        public void add(BJNetCall call) {            list.add(call);        }        public void remove(BJNetCall call) {            list.remove(call);        }        public void cancelAll() {            for (BJNetCall call : list) {                try {                    call.cancel();                } catch (Exception e) {                    e.printStackTrace();                }            }            list.clear();        }    }    private class ResourceCheckThread extends Thread {        public ResourceCheckThread()  {            super("NetResourceCheckThread");            setPriority(Thread.MAX_PRIORITY);            setDaemon(true);        }        @Override        public void run() {            ResourceReference reference = null;            while (! interrupted()) {                if (reference != null) {                    reference.cancelAll();                    reference.clear();                    mResourceRefMap.remove(reference.getTagId());                    Log.i("BJNetResource", "["+reference.getTagName() +"("+reference.getTagId()+")"+" is released and cancel calls auto.]");                }                // default tag 域中可能有已经执行完成了的 call. 将其回收掉                ResourceReference defaultReference =  mResourceRefMap.get(mDefaultObject.hashCode());                if (defaultReference != null) {                    Iterator<BJNetCall> iterator = defaultReference.list.listIterator();                    while (iterator.hasNext()) {                        if (! iterator.next().isExecuted()) {                            iterator.remove();                            Log.i("BJNetResource", "["+defaultReference.getTagName() +"("+defaultReference.getTagId()+")"+" is cleanup and cancel calls auto.]");                        }                    }                }                try {                    // remove() 会 wait 住线程                    reference = (ResourceReference) mReferenceQueue.remove(1000);                }  catch (Exception e) {                    e.printStackTrace();                }            }        }    }}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 一、缓存介绍: (一)、Android中缓存的必要性: 智能手机的缓存管理应用非常的普遍和需要,是提高用户体验的有...
    温暖的外星阅读 4,367评论 0 12
  • 1.如何开启多进程?应用是否可以开启N个进程? 实现多进程可以通过设置service、broadcast、acti...
    她做了一个梦阅读 2,767评论 0 2
  • 2017年据说是找工作的寒冬,作为一个Android开发,下面的一些问题可能在面试的时候遇到哦: 1、java 内...
    小相柳阅读 3,493评论 0 4
  • iOS开发中不可避免要使用到推送,JPush还是个不错的选择。本文主要介绍极光推送的过程。 写在前面:极光推送本身...
    胖子程阅读 8,911评论 66 110
  • (作者原创,禁止抄袭) 有一天,我们相识在梦中,醒来后却发现我们早已形同陌路。 有一天,我们说下一个渡口再见,一转...
    兮若予阅读 2,629评论 1 1

友情链接更多精彩内容