diff --git a/blogs/Android/Framework/深入理解 Android 内核设计思想/View 事件分发.md b/blogs/Android/Framework/深入理解 Android 内核设计思想/View 事件分发.md new file mode 100644 index 0000000..ee3f727 --- /dev/null +++ b/blogs/Android/Framework/深入理解 Android 内核设计思想/View 事件分发.md @@ -0,0 +1,35 @@ +--- +View 事件分发 +--- + +#### 目录 + +1. View 中 TouchEvent 的投递流程 +2. ViewGroup 中 TouchEvent 的投递流程 + +#### View 中 TouchEvent 的投递流程 + +View 类对整个输入事件的处理流程是: + +```xml +InputChannel -> InputEventReceiver -> ViewRootImpl#deliverInputEvent -> ViewPostImeInputStage#onProcess -> View#dispatchPointerEvent -> View#dispatchTouchEvent -> onTouch -> onTouchEvent +``` + +在最新的 Android 系统中,事件的处理者不再是由 InputEventReceiver 独自承担,而是通过多种形式的 InputStage 来分别处理它们都有一个回调接口 onProcess 函数,这些都声明在 ViewRootImpl 的内部类里面,并且在 setView 方法里面进行注册,我们重点关注与 View 事件分发相关的 ViewPostImeInputStage。在它的 onProcess 函数中,如果判断事件类型是 SOURCE_CLASS_POINTER 即触摸屏的 MotionEvent 事件,就会调用 mView 的 dispatchPointerEvent 方法处理。 + +```java +public final boolean dispatchPointerEvent(MotionEvent event) { + if (event.isTouchEvent()) { + return dispatchTouchEvent(event); + } else { + return dispatchGenericMotionEvent(event); + } +} +``` + +如果判断是 TouchEvent,就会调用其 dispatchTouchEvent 方法处理。 + +```java + +``` + diff --git a/blogs/Java/口水话/JVM 相关口水话.md b/blogs/Java/口水话/JVM 相关口水话.md index b42c7d7..3cda100 100644 --- a/blogs/Java/口水话/JVM 相关口水话.md +++ b/blogs/Java/口水话/JVM 相关口水话.md @@ -197,10 +197,16 @@ JVM 也提供了内联缓存来加快动态绑定,它能够缓存虚方法调 #### JVM 是如何实现泛型的? -Java 中的泛型不过是一个语法糖,在编译时还会将实际类型给擦除掉,不过会新增一个 checkcast 指令来做编译时检查,不过类型不匹配就抛出 ClassCastException。 +Java 中的泛型不过是一个语法糖,在编译时还会将实际类型给擦除掉,不过会新增一个 checkcast 指令来做编译时检查,如果类型不匹配就抛出 ClassCastException。 不过呢,字节码中仍然存在泛型参数的信息,如方法声明里的 T foo(T),以及方法签名 Signature 中的 "(TT;)TT",这些信息可以通过反射 Api getGenericXxx 拿到。 +除此之外,需要注意的是,泛型结合数组会有一些容易忽视的问题。数组是协变且具体化的,数组会在运行时才知道并检查它们的元素类型约束,可能出现编译时正常但运行时抛出 ArrayStoreException,所以尽可能的使用列表,这就是 Effective Java 中推荐的列表优先于数组的建议。这在我们看集合源码时也能发现的到,比如 ArrayList,它里面存数据是一个 Object[],而不是 E[],只不过在取的时候进行了强转。还有就是利用通配符来提升 API 的灵活性,简而言之即 PECS 原则,上取下存。典型的案例即 Collections.copy 方法了: + +```java +Collections.copy(List dest, List src); +``` + #### JVM 是如何实现异常的? 在 Java 中,所有的异常都是 Throwable 类或其子类,它有两大子类 Error 和 Exception。 当程序触发 Error 时,它的执行状态已经无法恢复,需要终止线程或者终止虚拟机,常见的比如内存溢出、对栈溢出等;Exception 又分为两类,一类是受检异常,比如 IOException,一类是运行时异常 RuntimeException,比如空指针、数组越界等。