android

Handler机制

Looper.prepare

sThreadLocal.set(new Looper(quitAllowed));//线程关联looper 

new Handler时  mLooper = Looper.myLooper();//获取looper对象 

    public static @Nullable Looper myLooper() { 

        return sThreadLocal.get();//通过线程获得looper 

    }   

线程与Looper关联

handler与looper关联,MessageQueue创建于looper中,looper与线程关联,handler与MessageQueue通过looper关联,handler与线程关联

Android中让多个线程顺序执行探究

1 利用优先队列实现多个线程顺序执行  每创建一个线程我都给它设置了一个自带的优先级

final PriorityQueue queue = new PriorityQueue<>(11, new MyThreadComparator());

        queue.add(thread1);

        queue.add(thread2);

public static class MyThreadComparator implements Serializable, Comparator {

        @Override

        public int compare(Thread lhs, Thread rhs) {

            int value = lhs.getPriority() < rhs.getPriority() ? 1 : lhs.getPriority() >

rhs.getPriority() ? -1 : 0;

            return value;

        }

    }

1用transient关键字标记的成员变量不参与序列化过程。

Serializable的实现,只需要实现Serializable接口即可。这只是给对象打了一个标记(UID),系统会自动将其序

列化。而Parcelabel的实现,不仅需要实现Parcelabel接口,还需要在类中添加一个静态成员变量CREATOR,这个变

量需要实现 Parcelable.Creator 接口,并实现读写的抽象方法。

 Parcelable的性能比Serializable好,在内存开销方面较小,所以Android应用程序在内存间数据传输时推荐使用

Parcelable,如activity间传输数据和AIDL数据传递,而Serializable将数据持久化的操作方便,因此在将对象序列

化到存储设置中或将对象序列化后通过网络传输时建议选择Serializable

2 当ListView自身接收到的滑动事件时,使ScrollView取消拦截。ListView区域内的滑动事件由自己处理,

scrollListView.setOnTouchListener(new View.OnTouchListener() {

        @Override

        public boolean onTouch(View v, MotionEvent ev) {

            switch (ev.getAction()) {

                case MotionEvent.ACTION_DOWN:

                case MotionEvent.ACTION_MOVE:

                    scrollListView.getParent().requestDisallowInterceptTouchEvent(true);

                    break;

                case MotionEvent.ACTION_UP:

                case MotionEvent.ACTION_CANCEL:

                    scrollListView.getParent().requestDisallowInterceptTouchEvent(false);

                    break;

            }

            return false;

        }

    });

3 HashMap为什么是线程不安全的?并发操作HashMap,是有可能带来死循环以及数据丢失的问题的。

HashMap:它根据键的hashCode值存储数据,大多数情况下可以直接定位到它的值,因而具有很快的访问速度,但遍历顺序却是不确定的。 HashMap最多只允许一条记录的键为null,允许多条记录的值为null。HashMap非线程安全,即任一时刻可以有多个线程同时写HashMap,可能会导致数据的不一致。如果需要满足线程安全,可以用 Collections的synchronizedMap方法使HashMap具有线程安全的能力,或者使用分段锁实现的ConcurrentHashMap

HashMap底层是一个Entry数组,当发生hash冲突的时候,hashmap是采用链表的方式来解决的,在对应的数组位置存放链表的头结点。对链表而言,新加入的节点会从头结点加入。

在hashmap做put操作的时候会调用到以上的方法。现在假如A线程和B线程同时对同一个数组位置调用addEntry,两个线程会同时得到现在的头结点,然后A写入新的头结点之后,B也写入新的头结点,那B的写入操作就会覆盖A的写入操作造成A的写入操作丢失

4 view的绘制流程在activity的onResume()之后才调用的  onFinishInflate()在xml解析完毕后调用

onFinishInflate() -->  onMeasure()--> onLayout() --> onDraw()

viewGroup默认不调用onDraw()

-------------------------------面试题--------------------------------------------

1 数据库操作类型 ?如何导入外部数据库?

  NULL、INTEGER、REAL、TEXT、BLOB

  INTEGER –整数,对应Java 的byte、short、int 和long。

  REAL – 小数,对应Java 的float 和double。

  TEXT – 字串,对应Java 的String。

  android系统下数据库应该存放在 /data/data/com.*.*(package name)/ 目录下,所以我们需要做的是把已有的数

据库传入那个目录下。操作方法是用FileInputStream读取原数据库,再用FileOutputStream把读取到的东西写入到

那个目录。  db.execSQL(参数)  db.insert("sql语句")

2 本地广播 与全局广播 

  2.1 核心用法

 使用LocalBroadcastManager来管理广播:

    调用LocalBroadcastManager.getInstance()来获得实例

    调用xx.registerReceiver()来注册广播

    调用xx.sendBroadcast()发送广播

    调用xx.unregisterReceiver()取消注册

    2.2 注意事项

    本地广播无法通过静态注册来接收,相比起系统全局广播更加高效

    在广播中启动activity的话,需要为intent加入FLAG_ACTIVITY_NEW_TASK的标记,不然会报错,      因为需要一个栈来存放新打开的activity。

    广播中弹出AlertDialog的话,需要设置对话框的类型为:TYPE_SYSTEM_ALERT不然是无法弹出的。

3  Activity launch

  3.1 standard模式的Activity, 每次启动都会创建一个新的实例, 放到启动他的那个Activity所在的Task中.

  3.2 singleTop模式的Activity, 仅当该Activity已经在Task的顶部了, 才会复用. 复用时onPause, 然后       

  onNewIntent唤起, 走onResume流程. 否则都要创建新的实例, 放进Task中.

  3.3 singleTask模式的Activity, 同一个Task中只会存在一个实例. 如果Task中还没有, 则新建, 放在Task顶部; 

        如果Task中已经有该Activity实例, 则复用.

singleTask模式的Activity的复用模式:

  如果已经在Task顶部, 如同singleTop的复用模式;

        如果不在Task顶部, 则销毁Task中该Activity顶部的所有其他Activity, 通过onNewIntent唤起该      

  Activity, 走onRestart流程.

  3.4 singleInstance模式的Activity, 会运行在一个单独的Task中, 且整个系统中只有一个该Activity实例. 相当 于单例模式. 复用模式和singleTask一样.

  设置A为singleInstance, 执行A -> B -> C -> A  最后A是复用  按back键  不会出现A了

Activity的android:configChanges="orientation|keyboardHidden"时,切屏不会重新调用各个生命周期,只会执行onConfigurationChanged方法

4 Activity的绘制流程    setContentView  只是创建DecorView, 把我们的布局加载到DecorView 

    performLaunchActivity->Activity.onCreate()

    handleResumeActivity()

    ->performResumeActivity()->Activity.onResume()

    ->wm.addView(decor,1);才开始把我们的DecorView加载windowManager,这时才开始View的绘制流程,measume(),layout(),draw()

->WindowManagerImpl.addView()

->root.setView(view,wparmas,panel); ->requestLayout ->scheduleTraversals()

->doTraversal()->performTraversals()(网上文章从这里开始)

你可能感兴趣的:(android)