preface
Hi, everybody. Long time no see.
Today summarized before more than a month to see some of the interview questions, the difficulty is not big, big guy can pass by directly, of course, send a kind point praise can also .
Enter the topic, the following is the content of October, every three questions for a section, that is, a special article, I do not specifically distinguish, due to the number of words, also only excerpted some questions, we forgive. In addition to answer the bad place we can also leave a message knock me, thank you.
To declare that we do not recite the question, the interview question is just to help you review, expand their knowledge, learn from each other, the answer is for reference only. The place that does not understand still wants oneself to take a good look relevant knowledge point, just see a topic also is useless, hope everybody can find satisfactory job finally ~
Contents of the October issue
The entire process from entering the URL to rendering the entire interface, and what protocol is used in the process?
1) Process analysis: mainly divided into three steps
The DNS
. After entering the URL, the user needs to find the IP address corresponding to the domain name through DNS resolution. After the IP address is found, the user can find the server. The browser cache is searched first to see if there is a corresponding DNS record. Then, perform DNS resolution in the sequence of operating system cache, route cache, ISP DNS server, and root server until the corresponding IP address is found.The client (browser) interacts with the server
. The browser makes an HTTP request based on the resolved IP address and port number. The request reaches the transport layer, in this case the TCP layer, and the three-way handshake is initiated to establish the connection. After receiving the request, the server sends a packet to the client (browser). The client receives the packet and parses it to obtain HTML data, including HTML, JS, and CSS.The client (browser) parses the HTML data
, construct the DOM tree, then construct the render tree, and finally draw to the browser page.
2) It involves TCP/IP protocol cluster, including DNS, TCP, IP, HTTP and so on.
This section describes TCP/IP in detail
TCP/IP refers to the TCP/IP protocol cluster, which consists of various protocols involved in the transmission of information between different networks, including the following layers:
The application layer
: Provides data and services. Such as HTTP, FTP, DNS, etcThe transport layer
: Responsible for data assembly, block. Such as TCP, UDP, etcThe network layer
: Is responsible for telling the destination of the communication, such as IPData link layer
: Is responsible for connecting the hardware part of the network, such as Ethernet, WIFI, etc
TCP three handshakes and four waves, why not two handshakes? Why wave more?
Client A, server B 1) A three-way handshake is required to establish A TCP connection
- A sends A message to B that it wants to connect with B
syn
Bag, A entersSYN_SENT
State) - B receives a message indicating that I am ready to connect with you as well (B received
syn
Package, need to confirmsyn
Package, and send one yourselfsyn
The packet is sentsyn+ack
Packet, B entersSYN_RECV
State) - A receives the message and tells B that I received the signal that you are also ready to connect (A received
syn+ack
Packet, which sends an acknowledgement packet to the serverack
AB intoestablished
Status) starts the connection.
2) TCP disconnection requires four waves of the hand
- A tells B that it wants to disconnect from B
fin
And into theFIN_WAIT_1
State) - B receives the message but does not finish sending the message. It can only tell A that I received your disconnection message (B receives A FIN, sends an ACK, and enters
CLOSE_WAIT
State, A entersFIN-WAIT-2
State) - After A while, B finishes sending data, tell A, I can disconnect from you (B sends fin, enter
LAST_ACK
State, A entersTIME-WAIT
State) - A receives A message and tells B that he can disconnect (A receives A FIN, sends an ACK, and B enters
closed
State)
3) Why does one more wave actually require four normal disconnections and connections?
- A sends A message to B
- User B sends the message to user A, indicating that the message is correctly received
- B sends A message to A
- A sends A message to B, indicating that the message is correctly received.
But in the join, the second and third steps can be combined, because A and B are unconnected before the join, so there is no other case to deal with. In the case of disconnection, since both ends were normally connected before, the second step cannot guarantee that the message sent by B has been completed, so A cannot be informed of the message to be disconnected immediately. This is why the connection can be one step less.
4) Why do you need three connections instead of two? Normally, if I send you a message and you tell me I can receive it, doesn’t that mean our previous correspondence is normal?
- The short answer is,
TCP is a two-way communication protocol
If you shake hands twice, the message sent by B to A will not arrive correctly.
To ensure reliable transmission of TCP packets, both communication parties need to check whether the packets they have sent are received by the receiver. If they are not received, they need to resend the packets.
How does TCP guarantee reliable transmission?
Serial number and confirmation number
. For example, if one side of the connection sends an 80byte piece of data, it will carry a serial number, such as 101. The recipient receives the data and replies with the acknowledgement number 181 (180+1), so that the next sent message will be sent from 181.
Therefore, during the handshake, for example, USER A sends A SYN signal to user B with the initial sequence number 120. Then, user B receives an ACK message with the sequence number 120+1. At the same time, B sends the SYN signal to A with the initial sequence number of 256. If it does not receive the reply message from A, it will resend the syn signal. Otherwise, the subsequent communication cannot be normally completed if the sequence number is lost.
That’s the reason for the three handshakes.
The difference between TCP and UDP?
TCP provides a connection-oriented, reliable byte stream service. That is, a TCP connection (three-way handshake) must be established between the client and server before data can be exchanged. It also provides functions such as timeout retransmission, discarding duplicate data, verifying data, and flow control to ensure that data can be transmitted from one end to the other.
UDP is a simple datagram – oriented transport layer protocol. It provides no reliability, just sending datagrams from the application to the IP layer, but with no guarantee that they will reach their destination. UDP does not need to establish a connection between the client and the server before transmitting data packets, and there is no timeout retransmission mechanism, so the transmission speed is very fast.
So it comes down to this:
- TCP is connection-oriented and UDP is connectionless
- TCP data headers include serial numbers, acknowledgement numbers, and so on. In contrast, UDP program structure is simpler.
- TCP is byte stream oriented and UDP is datagram based
- TCP ensures data correctness, while UDP may lose packets
- TCP guarantees data order, UDP does not
It can be seen that TCP is suitable for stable application scenarios. It ensures the correctness and sequence of data. Therefore, TCP is generally used for web browsing and interface access, so the three-way handshake ensures the connection stability. UDP is a simple protocol that does not take into account packet loss, connection establishment, etc. The advantage is that data transmission is fast, so it is suitable for live broadcast, games and other scenes.
Several HTTP request methods are introduced
There are four common ones:
GET
Get the resource, no body, idempotentPOST
Add or modify resources, including bodyPUT
Modify resources, body, idempotentDELETE
Delete resources, idempotent
Format of HTTP request and response packets and common status codes
1) Request message:
// Request line (method, path, HTTP version)
GET /s HTTP/1.1
//Headers
Host: www.baidu.com
Content-Type: text/plain
//BodySearch for * * * *Copy the code
2) Response message
// Status line (including HTTP version, status code, status information)
HTTP/1.1 200 OK
//Headers
Content-Type: application/json; charset=utf-8
//Body
[{"info":"xixi"}]
Copy the code
3) Common status codes
It is mainly divided into five types:
1 the beginning
, for temporary messages, such as 100 (continue to send)2 at the beginning
, for successful request, such as 200 (OK)3 at the beginning
, for redirection, such as 304 (unchanged content)4 the beginning
, representing some errors on the client side, such as 403 (access forbidden)The beginning of may
Represents some errors in the server, such as 500
This section describes symmetric and asymmetric encryption
1) Symmetric encryption, that is, encryption and decryption algorithms are different, but the key is the same. Such as DES, AES algorithm.
Data A --> Algorithm D (key S) --> Encrypted data B --> Algorithm E (key S) --> Data ACopy the code
Advantages: Disadvantages: The key can be cracked and forged easily. Once the key is known by others during transmission, data can be decrypted.
2) Asymmetric encryption, that is, the encryption and decryption algorithm is the same, but the key is different. The private key is kept and the public key is provided to the other party. Such as RSA, DSA algorithm.
Data A -> Algorithm D (public key) -> Encrypted data B -> Algorithm D (private key) -> Data ACopy the code
Advantages: Security. The public key cannot decrypt data even if it is known by others. Disadvantages: Both communication parties need to have a set of public and private keys
Principle of digital signature
1) Why do you need a digital signature in the first place?
To protect against attack and counterfeiting. Because the public key is public, others can intercept the public key and forge data for transmission. Therefore, we need to verify the source of the data.
2) How do I sign my name?
Since the public key can decrypt the data encrypted by the private key, the private key can also decrypt the data encrypted by the public key. (Asymmetric encryption A and B code can be exchanged) so we use public key encryption, generate encrypted data. Encrypt the hash of the data with the private key again, then the encryption of the private key is called a signature, that is, only I can encrypt the operation. Therefore, the data transmission process becomes encrypted data and signature data. If the solution is the same data and its summary information, then the data is safe and reliable.
Data A --> Algorithm D (public key) --> Encrypted data B Data A Abstract --> Algorithm D (private key) --> Signature data C Encrypted data B --> Algorithm D (private key) --> Data A Abstract Signature data C --> Algorithm D (public key) --> Data A AbstractCopy the code
In this way, the signed data is added to ensure that the original data, namely data A, was indeed sent from the correct sender and is the complete original data that has not been modified.
What is a Base64 algorithm? Is it an encryption algorithm?
-
Base64 is an encoding algorithm that converts binary data into a string of 64 characters. It is mainly used for the transmission of non-text data, such as pictures. Binary data such as pictures can be converted into specific strings for saving and transmission.
-
Technically, no. Although it does convert one piece of binary data into another, its encryption and decryption are public, so there is no secret. So I prefer to think of it as encoding, and anyone can encode and decode binary data in Base64.
-
Interview bonus: To reduce confusion, facilitate replication, and reduce data length, a base58 encoding was derived. Removed some confusing numbers and letters (digit 0, O, I, digit 1, symbol +, symbol /) in Base64
Bitcoin is known for its improved Base58 encoding, or Base58Check encoding, which incorporates a verification mechanism and hash values.
Why is there a concurrency problem when multiple threads are accessing the same variable at the same time?
- The Java memory model specifies that all variables are stored in main memory and that each thread has its own working memory.
- The working memory of a thread holds a main memory copy of variables used by the thread. All operations on variables must be performed in the working memory of the thread instead of reading or writing directly to the main memory.
- When a thread accesses a variable, it copies the variable from main memory to working memory. Writes to the variable are not immediately synchronized to main memory.
- Different threads cannot directly access variables in each other’s working memory, and the transfer of variables between threads requires data synchronization between their own working memory and main memory.
What do we mean by atomicity, visibility, and order?
-
Atomicity: In an operation, the CPU may not pause and then schedule, i.e., the operation will either complete or not be executed without interruption.
-
Visibility: When multiple threads access the same variable and one thread changes the value of the variable, other threads can immediately see the changed value.
-
Orderliness: The order in which a program is executed is the order in which the code is executed.
Are there any examples of multi-threaded concurrency problems in real projects?
There are, for example, singleton patterns. Because of the particularity of singleton mode, it may be called by multiple threads at different places in the program at the same time. Therefore, in order to avoid multi-threaded concurrency, volatile+Synchronized is generally used to protect variables and methods.
private volatile static Singleton singleton;
public static Singleton getSingleton4(a) {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = newSingleton(); }}}return singleton;
}
Copy the code
This section describes several startup modes.
standard
By default, each startup creates a new Activity instance and enters the current task stacksingleTop
If the Activity you want to start has an instance at the top of the stack, the Activity is not recreated. Instead, the Activity instance at the top of the stack is used directly and the onNewIntent method is called back.singleTask
If the Activity to be started has an instance in the stack, the Activity is not recreated. Instead, the Activity instance in the stack is used directly and the onNewIntent method is called back. It also pushes the instance to the top of the stack, and everything previously on top of the Activity is destroyed.singleInstance
The onNewIntent method is called and called back when the Activity is called repeatedly.
Activity A→B→C→B, where B startup mode is singleTask, AC is standard, how to call the life cycle respectively? What if B starts in singleInstance mode? The startup mode of B is singleInstance unchanged. When A→B→C is clicked twice to return, how to call the life cycle?
1)A→B→C→B,B’s startup mode is singleTask
- To start the process of A, the lifecycle call is (A)onCreate→(A)onStart→(A)onResume
- To restart B, the lifecycle calls are (A)onPause→(B)onCreate→(B)onStart→(B)onResume→(A)onStop
- B goes to C the same way
- In the process of C→B, since B’s startup mode is singleTask, B will call onNewIntent and remove the instances on B, that is, C will be removed from the stack. So the lifecycle calls are (C)onPause→(B)onNewIntent→(B)onRestart→(B)onStart→(B)onResume→(C)onStop→(C)onDestory
2)A→B→C→B,B’s startup mode is singleInstance
- If B is singleInstance, then C will not be removed from C→B, because B and C are not in the same task stack. So the lifecycle calls are (C)onPause→(B)onNewIntent→(B)onRestart→(B)onStart→(B)onResume→(C)onStop
3)A→B→C,B’s startup mode is singleInstance, and click the return key twice
-
If B is singleInstance, the process A→B→C is called as before. However, when clicking return, AC is the same as the task stack, so C clicking return will return to A, and then clicking return will return to B. So the lifecycle is :(C)onPause→(A)onRestart→(A)onStart→(A)onResume→(C)onStop→(C)onDestory.
-
Click back again, and you’ll go back to B, so the lifecycle is :(A)onPause→(B)onRestart→(B)onStart→(B)onResume→(A)onStop→(A)onDestory.
How to prevent an Activity from rebuilding while the screen rotates.
-
The life cycle of the toggle screen is: onConfigurationChanged->onPause->onSaveInstanceState->onStop->onDestroy->onCreate->onStart->onRestoreInstanceState->onRe sume
-
If you want to prevent rotation and recreate your Activity, you need to do the following configuration:
In targetSdkVersion value less than or equal to 12, the configuration of the android: configChanges = “orientation” | keyboardHidden “, when the targetSdkVersion value is greater than 12, Configure the android: configChanges = “orientation” | keyboardHidden | screenSize”
Three ways to start a thread
1) Inherit the Thread class
class MyThread :Thread() {override fun run(a) {
super.run()
}
}
fun test(a){
var t1=MyThread()
t1.start()
}
Copy the code
2) Implement runnable interface
class MyRunnable : Runnable {
override fun run(a){}}fun test(a) {
var t1 = Thread(MyRunnable(),"test")
t1.start()
}
Copy the code
3) Implement Callable interface
class MyCallThread : Callable<String> {
override fun call(a): String {
return "i got it"}}fun test(a) {
var task = FutureTask(MyCallThread())
var t1 = Thread(task, "test")
t1.start()
try {
// Get the result
var result = task.get()}catch (e: Exception) {
}
}
Copy the code
The third method, FutureTask, also implements Runnable, but behaves differently and returns a value. This everyone interview time can say, and then say their own views, after all, to let the interviewer see more of your knowledge.
The difference between threads run and start
-
Start method, with the start method to start the thread, the true implementation of multithreading, then do not wait for the run method body of the code to complete and directly continue to execute the subsequent code. A Thread is started by calling the start() method of the Thread class. The Thread is ready but not running. Once the CPU slice is available, the run() method is executed. The thread then terminates.
-
Run method, the run method is just an ordinary method of class, if direct call run method, this thread is still only the main thread in the program, the program execution path or there is only one, or to order, still want to wait until after the run method body has been completed can continue to execute the following code, so that no writer threads.
The run method is a normal method call in the thread class. Instead of starting a new thread, the run method is executed in the main thread.
How do thread states transition from one another
1) Initial state (New). A new thread object is created to enter the initial state, which can be entered through the above methods of creating a new thread.
2) RUNNABLE state After the thread object is created, other threads, such as the main thread, call the start() method of the object. Threads in this state are in the runnable thread pool, waiting to be selected by thread scheduler for CPU usage. There are several ways to enter the runnable state:
- Call the start method.
- Get object lock
- Call the yield method
3) RUNNING state. The runnable thread gets the CPU time slice and executes the program code. The thread scheduler enters the run state when it selects a thread from the runnable pool as the current thread.
4) BLOCKED. A thread is suspended while it is running, usually to wait for a certain time to occur (such as when a resource is ready) before continuing. Methods like wait, sleep, and suspend can all cause threads to block.
5) DEAD. A thread terminates its life cycle when its run(), main() methods finish executing, or it exits the run() method due to an exception. Dead threads cannot be resurrected.
Is String a basic data type in Java? Is it mutable? Is it thread-safe?
String
Java defines big data types as:byte, short, int, long, char, float, double, boolean
String
Is immutableString
Is immutable. Once we create a String, we can’t change its value. So it isThread safetyCan be safely used in multithreaded environments
Why make it immutable? If String is immutable, what do we change in our usual assignment?
1) Why is design immutable
security
. Because String is widely usedjava
Class parameters, so security is a very important consideration. This includes thread safety, opening files, storing data passwords, and so on.- The immutability of String guarantees that the hash code is always one, so you don’t need to recalculate the hash code when using classes like HashMap,
To improve efficiency
. - Because Java strings are immutable, you can save a lot at Java runtime
The Java heap space
. Because different string variables can refer to the same string in the pool. If strings are mutable, any change in the value of one variable will reflect the other variables, soString pooling
It doesn’t make any sense.
2) Normally, double quotation marks are used to return valuesString reference
I’m not changing the string object
What is the difference between String, StringBuffer, and StringBuilder? Are the two ways of creating a String stored in the JVM the same?
Strings are immutable, so whenever we operate on a String, we always create a new String. Manipulating Strings is very resource-intensive, so Java provides two utility classes to manipulate StringBuffer and StringBuilder.
StringBuffer and StringBuilder are mutable classes, StringBuffer is thread-safe, StringBuilder is not thread-safe. So we should use StringBuffer when multiple threads are working on the same string. StringBuilder is more efficient than StringBuffer because it doesn’t have to handle multiple threads.
1) There are two common ways to create a String
- String s1 = “Java”
- String s2 = new String(“Java”)
2) Different storage methods
-
First, S1 looks for the string “Java “in the string constant pool and returns the constant handle if it has the same character. If it doesn’t, s1 creates the string in the constant pool and returns the constant handle, or string reference.
-
Second, s2 creates a variable object directly on the heap, but does not store it in the string pool, and calls intern to store the string in the constant pool
What is a thread pool and what are its advantages?
Thread pools are mainly used to manage child threads and have the following advantages:
- Reuse threads in the thread pool to avoid the frequent creation and destruction of threads
Memory overhead
. - Effectively control the maximum number of concurrent threads to avoid resource preemption between threads
Blocking phenomenon
. - Can be simple management of threads, provide
Regularly perform
As well as the specifiedThe interval is executed in a loop
, and other functions.
What does the constructor of a thread pool mean for each parameter to execute the task’s flow
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler) {}
Copy the code
corePoolSize
: Number of core threads. By default, the thread pool is empty and threads are created only when a task is submitted. If the number of threads currently running is less than corePoolSize, a new thread is created to process the task; If equal to or equal to corePoolSize, it is no longer created. If the thread pool’s prestartAllcoreThread method is called, the thread pool creates and starts all core threads ahead of time to wait for the task.maximumPoolSize
: Maximum number of threads allowed to be created in the thread pool. If the task queue is full and the number of threads is less than maximumPoolSize, the thread pool still creates new threads to process the task.keepAliveTime
: Timeout event when a non-core thread is idle. Anything beyond this event is reclaimed. If there are many tasks and the execution time of each task is short, you can increase keepAliveTime to improve thread utilization. In addition, keepAliveTime is applied to the core thread if the allowCoreThreadTimeOut property is set to true.TimeUnit
: keepAliveTime Time unit of a parameter. The options are Days, HOURS, MINUTES, SECONDS, and MILLISECONDS.workQueue
: Task queue. If the current thread count is greater than corePoolSzie, the task is added to this task queue. The task queue is of type BlockingQueue, or BlockingQueue.ThreadFactory
: Thread factory. You can use a thread factory to give each created thread a name. Generally, you do not need to set this parameter.RejectedExecutionHandler
: Rejection policy. This is the current task queue and thread pool full of coping strategies, taken when default is AbordPolicy, unable to process a new task, and throw RejectedExecutionException anomalies.
Among them, there are four rejection strategies:
AbordPolicy
: unable to process a new task, and throw RejectedExecutionException anomalies.CallerRunsPolicy
: uses the caller’s thread to process the task. This strategy provides a simple feedback control mechanism that slows down the delivery of new tasks.DiscardPolicy
: Indicates an unexecutable task and deletes the task.DiscardOldestPolicy
: Discards the latest task in the queue and executes the current task.
Task execution process:
- If the number of threads in the thread pool is not reached
Number of core threads
, will directly start a core thread to execute the task. - If the number of threads in the thread pool has reached or exceeded the number of core threads, the task will be inserted into
Task queue
Is queued for execution. - If the task queue cannot insert new tasks, it indicates that the task queue is full. If the maximum number of threads has not been reached, one is started
Non-core thread
To carry out the mission. - Execute if the number of threads exceeds the specified maximum
Rejection policies
– RejectedExecutionHandler.
What are the main categories of Android thread pools and what do they represent?
There are four main types: FixedThreadPool, CachedThreadPool, SingleThreadExecutor, and ScheduledTheadPool
1) FixedThreadPool — a reusable thread pool with a fixed number of threads
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>());
}
Copy the code
- thread
The number of fixed
Both are core threads: the number of core threads and the maximum number of threads are nThreads; - Are core threads and will not be recycled, quickly respond to external requests;
- There is no timeout mechanism and no size limit on the task queue;
- New Task use
Core thread
Process, and if there are no free core threads, queue up for execution.
- CachedThreadPool — Thread pools created on demand
public static ExecutorService newCachedThreadPool(a) {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue<Runnable>());
}
Copy the code
- The number of threads is variable, only non-core threads, maximum number of threads
Any large
The number of core threads passed in is 0, and the maximum number of threads is integer.max_value. - For new tasks
The idle thread
Execute, and create a new thread to process if there are no idle threads. - Each idle thread in the thread pool has a timeout mechanism, usually 60 SECONDS (parameter: 60L, timeUnit.seconds). If the idle thread exceeds 60 SECONDS, the idle thread will be reclaimed.
- Suitable for performing a large number of less time-consuming tasks when all threads are idle
More than 60 s
Will be stopped, so there is very little use of system resources.
- SingleThreadExecutor — single-threaded thread pool
public static ExecutorService newSingleThreadExecutor(a) {
return new FinalizableDelegatedExecutorService
(new ThreadPoolExecutor(1.1.0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>()));
}
Copy the code
- only
One core thread
All tasks are executed sequentially on the same thread. - All external tasks are unified into one thread, so there is no need to deal with thread synchronization.
- ScheduledThreadPool – A pool of timed and periodic threads
private static final long DEFAULT_KEEPALIVE_MILLIS = 10L;
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE,
DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS,
new DelayedWorkQueue());
}
Copy the code
- Core thread
The number of fixed
, number of non-core threadsunlimited
; - Non-core threads that are idle for more than 10 seconds are recycled.
- It is mainly used to perform scheduled tasks and repetitive tasks with a fixed period.
What is an index? Pros and cons
A database index is a sorted data structure in a database management system to assist in quick queries to update the data of tables in a database. Index implementation usually uses B tree and variant B+ tree (mysql common index is B+ tree)
advantages
- By creating an index, you can
Improve system performance
- By creating a unique index, each row in a database table is guaranteed to be valid
uniqueness
- When using grouping and sorting clauses for data retrieval, you can reduce the number of queries
Grouping and sorting time
disadvantages
- To create and maintain indexes
Time consuming
And the time increases with the increase of data volume - Indexes need to take up physical space, which is required if you want to build clustered indexes
There will be more space
- Required when adding, deleting, or modifying data in a table
It takes more time
Because indexes are also maintained dynamically
Four features of transactions
Database transactions must have ACID feature, which is shorthand for **Atomic, Consistency, Isolation, and persistence **.
- atomic
All operations in a transaction either complete or do not complete and do not end up somewhere in the middle. If a transaction fails during execution, it is rolled back to the state before the transaction began, as if the transaction had never been executed.
- consistency
Transaction consistency means that the database must be in a consistent state both before and after a transaction is executed. If the transaction completes successfully, all changes in the system are applied correctly and the system is in a valid state. If an error occurs during a transaction, all changes in the system are automatically rolled back and the system is returned to its original state.
- Isolation,
In a concurrent environment, when different transactions manipulate the same data at the same time, each transaction has its own complete data space. Changes made by concurrent transactions must be isolated from changes made by any other concurrent transactions. When a transaction views a data update, the data is either in the state it was in before another transaction modified it, or in the state it was in after another transaction modified it, and the transaction does not see the data in the intermediate state.
- persistence
This means that as long as a transaction successfully terminates, its updates to the database must be persisted forever. Even if a system crash occurs, the database system can be restarted to the state it was at the successful end of the transaction.
A couple of paradigms
The English name of Normal Form is Normal Form, which is summarized after e.F. Codd (ancestor of relational database) put forward the relational database model in 1970s. Paradigm is the basis of relational database theory, and it is also the rule and guiding method that we should follow in the process of designing database structure. Usually only the first three paradigms are used, namely: first normal (1NF), second normal (2NF), and third normal (3NF).
-
The first paradigm is that attributes are indivisible, and every field should be indivisible. For example, if a field is NAME, it is generally understood in China that NAME is a unit that cannot be split again, which conforms to the first normal form. But in foreign countries, it is also divided into FIRST NAME and LAST NAME. At this time, the NAME field can be divided into smaller units, which does not conform to the FIRST normal form.
-
The second normal form requires that there be a primary key in the table; all other fields in the table depend on the primary key, so the second normal form just remembers the primary key constraint. Table, for example, there is a table is the student, the student have a value in the table only field student id, so students can all other fields in a table according to the student id field to obtain, depend on the primary key of the meaning is the meaning of related, because student id value is unique, so will not cause the information stored in question, Student 001’s name will not be saved to Student 002.
-
The third normal form requires that a table cannot have fields that store the same information that exist in other tables. Usually, the implementation is to establish associations through foreign keys, so the third normal form only needs to remember the foreign key constraints. Table, for example, there is a table is the student, the student has a student number in the table, name field, such as that if you want to tie his number, head of the department, the dean and also endures the student table, it can cause a lot of redundant data, one is the information existing in the system information table, 2 it is department if there are 1000 students in the information storage, 1000 times. Therefore, the third normal form approach is to add a department number field (foreign key) to the student table to associate with the department information table.
Recycleview is different from ListView
Recycleview layout effect more
Added vertical, table, waterfall flow and other effectsRecycleview got rid of some of the apis
, such as setEmptyview, onItemClickListener, etc., to give the user more customization possibilitiesRecycleview removed the ability to set items at the bottom of the header
, specifically implemented through different types of viewholderRecycleview implements some local refreshes
, such as notifyitemchangedRecycleview comes with some animations for layout changes
, you can also implement custom animation effects through the custom ItemAnimator classRecycleview cache mechanism is more comprehensive
, add two levels of cache, and support custom cache logic
How many levels of cache does Recycleview have, the cache process?
Recycleview has four levels of cache, namely mAttachedScrap(in-screen), mCacheViews(off-screen), mViewCacheExtension(custom cache), mRecyclerPool(cache pool).
MAttachedScrap (inside screen)
For quick reuse of in-screen ItemView, without the need to re-createView and bindViewMCacheViews (Off-screen)
Save the ViewHolder that has been moved off the screen recently, which contains data and position information. The ViewHolder must be in the same position when reused. Application Scenarios For those lists that need to be swiped back and forth, the ViewHolder data can be reused directly without re-bindView.MViewCacheExtension (custom cache)
Is not directly used, but requires user-defined implementation. The default implementation is not implemented.MRecyclerPool (Cache pool)
When the cacheView is full or the Adapter is changed, the ViewHolder removed from the cacheView will be removed from the Pool. The ViewHolder data will be erased before the cacheView is removed from the Pool. Therefore, the bindView will need to be re-used.
Level 4 caches need to be read in order. So the full cache flow is:
- Save cache process:
- Insert or delete
itemView
, first save the ViewHolder on the screen toAttachedScrap
In the - When you swipe the screen, the itemView that disappears first is saved to
CacheView
CacheView defaults to a CacheView size of 2. If there are more than one CacheView, we will remove the header itemView and save it to the first in, first out ruleRecyclerPool buffer pool
RecyclerPool cache pool is created according to itemViewitemtype
For each itemTyep cache number of 5, more than will be recycled.
- Get the cache process:
- AttachedScrap, failed to obtain from pos match holder —
CacheView
The holder cache is also obtained through pos
CreateViewHolder — createViewHolder and bindview.
Note that if you find the cache from the cache pool, you also need to re-bindView.
Talk about RecyclerView performance optimization.
bindViewHolder
The method is executed in the UI thread, so this method should not be time-consuming or it will affect the smoothness of the slide. Like formatting dates.- This parameter can be used when adding or deleting data
diffutil
Do local refresh rather than global refresh - for
itemVIew
Make layout optimizations, such as less nesting. - 25.1.0 (>=21) and above
Prefetch
Function, also is the function of prefetching, nested and use LinearLayoutManager, child RecyclerView number can be set up by setInitialPrefatchItemCount prefetch - increase
RecyclerView cache
For example, the cacheView defaults to a size of 2. You can use a larger size to trade space for time and improve flow - If the height is fixed, you can set it
setHasFixedSize(true)
To prevent requestLayout from wasting resources and remeasuring the height every time the data is updated.
void onItemsInsertedOrRemoved(a) {
if (hasFixedSize) layoutChildren();
else requestLayout();
}
Copy the code
- If multiple
RecycledView
The Adapter is the same. For example, the nested RecyclerView has the same Adapter, which can be setRecyclerView.setRecycledViewPool(pool);
To share oneRecycledViewPool
. This reduces the overhead of creating a VIewholder. - When the elements of RecyclerView are relatively high and one screen can only display one element, sliding to the second element for the first time will lag. This can be overridden by setting up extra cache space
getExtraLayoutSpace
Method can.
new LinearLayoutManager(this) {
@Override
protected int getExtraLayoutSpace(RecyclerView.State state) {
returnsize; }};Copy the code
- Set up the
RecyclerView.addOnScrollListener();
To stop the loading operation while sliding. - To reduce object creation, such as setting listener events, you can create a listener globally, all views share a listener, and place it in
CreateView
Create a listener inside, because CreateView calls are less than bindView calls. This reduces the overhead of object creation - with
notifyDataSetChange
, the adapter does not know what is in the entire data set and then matches againViewHolder
Peanuts flash when. Set Adapter.sethasstableids (true) and rewritegetItemId()
To give each Item a unique ID, that is, a unique identifier, which fixes the focus of the ItemView and resolves the flicker problem.
Talk about double check locks, and the role of volatile
Let’s review the prototype of the double check lock, namely the implementation of the singleton pattern:
public class Singleton {
private volatile static Singleton mSingleton;
private Singleton(a) {}public Singleton getInstance(a) {
if (null == mSingleton) {
synchronized (Singleton.class) {
if (null == mSingleton) {
mSingleton = newSingleton(); }}}returnmSingleton; }}Copy the code
There are several questions that need to be addressed:
- Why lock it?
- Why not just lock the getInstance method?
- Why is it necessary to double check whether it is empty?
- Why volatile?
Here are the answers:
- If it’s unlocked, yes
Thread insecurity
That is, it is possible for multiple threads to access the getInstance method at the same time to get two instantiated objects. - If you lock the getInstance method, you will need to lock it every time you access the mSingleton
Performance overhead
- The first nullation is to determine whether the variable has been instantiated. If it has been instantiated, the variable will be returned directly without locking. The second nullation is when you get to the lock step, if thread A is instantiated, and when thread B gets the lock, it’s actually instantiated by the time it gets in, and if you don’t nullation it’s going to
Instantiate again
. - Volatile is used for
Disallow command reordering
. Instruction rearrangement refers to that in the process of running the program, the execution is not completely in accordance with the code order. It is possible to exchange the instruction order that does not affect the result in consideration of performance and other reasons. So the initialization sequence would have been this:
1) Allocate memory space 2) initialize objects 3) point objects to allocated space
If an order is rearranged, it is possible that 2 and 3 will be swapped because the result is not affected. So it becomes:
1) Allocate memory space 2) point objects to allocated space 3) initialize objects
It is possible that if thread A has reached the second step and thread B enters the second nullity, it will judge that the mSingleton is not empty and directly return, but actually the mSingleton has not been initialized at this time.
The difference between synchronized and volatile
volatile
Essentially telling the JVM that the current value of a variable in a register is indeterminate and needs to be read from main memory,synchronized
The current variable is locked. Only the current thread can access the variable. Other threads are blocked.volatile
Can only be used at the variable level,synchronized
Can be used in variables, methods.volatile
Can only achieve variable change visibility, andsynchronized
The change visibility and atomicity of variables can be guaranteed.volatile
Does not cause the thread to block, whilesynchronized
It may block the thread.- When the value of a field depends on the value before it,
volatile
It doesn’t work,n =n+1,n++, etc., so it doesn’t guarantee atomicity. - use
volatile
Rather thansynchronized
The only safe case is when there is only one mutable field in a class.
What is the difference between synchronized modifying static methods and ordinary methods
-
Synchronized modifies a nonstatic method and essentially locks the object on which the method is invoked, colloquially called “object locking.” So we’re locking this object, this. If the same object accesses two synchronized methods of the object on two threads separately, a mutex is created. This is called an object lock, and an object can only enter one operation at a time.
-
Synchronized refers to static methods that actually lock objects of that class. That is, the locked class is xx.class. If an object calls a statically synchronized method and a nonstatically synchronized method in two separate threads, since the static method receives the class lock limit, but the nonstatic method receives the object limit, the two methods are not the same object lock and therefore are not excluded.
What are memory leaks and why do they happen?
A Memory Leak refers to a program that fails to release dynamically allocated heap Memory for some reason, resulting in a waste of system Memory, slowing down the program and even crashing the system. To put it simply, mobile phones provide our applications with a certain amount of heap memory, which is constantly GC(Java’s garbage collection mechanism) during object creation, so memory normally stays at a stable value. However, a memory leak can cause an instance, such as an Activity instance, to be referenced somewhere and not released properly, resulting in a larger memory footprint. This is a memory leak.
What happens when a memory leak occurs?
There are four main types of cases:
- Collection class leakage
- Memory leaks caused by singletons/static variables
- Anonymous inner classes/non-static inner classes
- Memory leak caused by resource not being closed
1) Collection class leakage
After the collection class adds elements, it still refers to the collection element object, so the element object in the collection cannot be reclaimed, resulting in a memory leak.
List<Object> mList = new ArrayList<>();
for (int i = 0; i < 100; i++) {
Object obj = new Object();
mList.add(obj);
obj = null;
}
Copy the code
The solution is to let go of the set as well.
mList.clear();
mList = null;
Copy the code
2) Memory leaks caused by singletons/static variables
Because of the static nature of the singleton pattern, which has a lifetime equal to the lifetime of the application, it is often prone to memory leaks.
public class SingleInstance {
private static SingleInstance mInstance;
private Context mContext;
private SingleInstance(Context context){
this.mContext = context;
}
public static SingleInstance newInstance(Context context){
if(mInstance == null){
mInstance = new SingleInstance(context);
}
returnsInstance; }}Copy the code
For example, in this singleton pattern, if we pass the Activity context when we call newInstance, then we have a reference with a long lifetime that has a short lifetime, causing a memory leak. To modify the context to context. GetApplicationContext ().
3) Anonymous inner classes/non-static inner classes
Non-static inner classes hold strong references to their outer classes, which can cause non-static inner classes to have a longer lifetime than their outer classes and are prone to memory leaks. The most common example is the Handler.
public class TestActivity extends Activity {
private TextView mText;
private Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg); }};@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_test);
new Thread(new Runnable() {
@Override
public void run(a) {
try {
Thread.sleep(100000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}).start();
mHandler. sendEmptyMessageDelayed(0.100000);
}
Copy the code
How do I fix it? Change to a static inner class, and then modify the outer class in a weak reference fashion
public class TestActivity extends Activity {
private TextView mText;
private MyHandler myHandler = new MyHandler(TestActivity.this);
private MyThread myThread = new MyThread();
private static class MyHandler extends Handler {
WeakReference<TestActivity> weakReference;
MyHandler(TestActivity testActivity) {
this.weakReference = new WeakReference<TestActivity>(testActivity);
}
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
weakReference.get().mText.setText("do someThing"); }}@Override
protected void onDestroy(a) {
super.onDestroy();
myHandler.removeCallbacksAndMessages(null);
}
Copy the code
4) Memory leaks caused by unclosed resources
Such as:
- Networks, files, and other streams forget to close
- Manually register when broadcasting, exit when forgetting
unregisterReceiver()
- Service is forgotten after execution
stopSelf()
- Observer mode frameworks such as EventBus forget to manually unregister
How do I find and resolve memory leaks?
1. Using tools such as Memory Profiler, you can view the real-time Memory of your app, capture heap dumps, and generate a Memory snapshot, hprof file. By looking at the files, you can see which classes are leaking memory.
2. Use the LeakCanary library, which is known as LeakCanary. Import the library and run it to find the memory leak in the app.
Here’s how LeakCanary works:
- Listening to the
First by ActivityLifecycleCallbacks and FragmentLifeCycleCallbacks listening Activity and fragments of life cycle.
- judge
It then determines whether the object is reclaimed during the life cycle of destruction. When a weak reference is defined, it can specify a reference object and a ReferenceQueue. Whether the weak reference is added to the ReferenceQueue can determine whether the object is reclaimed.
- Analysis of the
Finally, the Hprof file is analyzed through the HAHA library to find out the previous reference relationship of the class.
What is the class loading mechanism?
The Java files that we write will compile into.class files, and the class loader is responsible for loading the class bytecode files, and the class files have a file identifier at the beginning of the file, and loading the class file bytecode contents into memory, This is converted into a runtime data structure in the method area and the ClassLoader is only responsible for loading the class file. Whether it can run is determined by the Execution Engine.
Simply put, class loading is the act of reading a series of class files from the file system into JVM memory to provide resources for subsequent program execution.
Class loader type.
There are four main types of classloaders:
- BootstrapClassLoader: starts the class loader, implemented in C++
- ExtClassLoader: Extended class loader, implemented in Java
- AppClassLoader: An application class loader that loads all classes of the current application’s classpath
- UserDefinedClassLoader: user-defined class loader
Belongs to sequential inheritance, that is, the parent loader of the next level.
What is the parent delegate mechanism and why is it designed this way?
When a class loader received the request of the class loading, it will not go directly to load this class, but rather to delegate the request to parent to complete, and, in turn, will pass to the most superior, that is, to start the class loader, and then the parent loader will check whether they have been loaded the class, if not loaded, would be to load, load will give son loader for the failure, All the way down to the bottom level, if none of them can be loaded correctly, a ClassNotFoundException will be thrown.
For example:
- When an Application ClassLoader receives a classload request, it first does not attempt to load the class itself. Instead, it delegates the request to the parent ClassLoader, Extension ClassLoader.
- When the Extension ClassLoader receives a classload request, it does not attempt to load the class itself in the first place. Instead, it delegates the request to the parent ClassLoader, Bootstrap ClassLoader.
- If the Bootstrap ClassLoader fails to load (the required class is not found in
\lib), the Extension ClassLoader will try to load.
- If the Extension ClassLoader also fails to load, it will be loaded using the Application ClassLoader.
- If the Application ClassLoader also fails to load, a custom loader is used to try to load it.
- If all load fails, a ClassNotFoundException is thrown.
The reason for this design is to prevent dangerous code placement, such as String class, if the AppClassLoader is loaded directly, it will be tampered with, so it has to be checked by the first, that is, BootstrapClassLoader, already loaded classes do not need to load again.
WebView communicates with JS
1) Android calls JS code
There are two main methods:
- LoadUrl () via WebView
// Call the javascript callJS() method
mWebView.loadUrl("javascript:callJS()");
Copy the code
But this is not often used because it automatically refreshes the page and returns no value, somewhat affecting the interaction.
- Through the WebView
EvaluateJavascript ()
MWebView. EvaluateJavascript ("javascript:callJS()".new ValueCallback<String>() {
@Override
public void onReceiveValue(String value) {
// This is the result returned by js}});Copy the code
This is more comprehensive. Call the method and get the return value.
2) JS calls Android code
There are two main methods:
- Through the WebView
AddJavascriptInterface ()
Object mapping
public class AndroidtoJs extends Object {
// Define the method that JS needs to call
// Methods called by javascript must be annotated with @javascriptInterface
@JavascriptInterface
public void hello(String msg) {
System.out.println("JS calls Android's Hello method");
}
}
mWebView.addJavascriptInterface(new AndroidtoJs(), "test");
/ / in js:
function callAndroid(a){
// Because of the object mapping, calling the test object is equal to calling the Android mapping object
test.hello("Js calls the Hello method in Android");
}
Copy the code
This method works well, but it is important to note that after 4.2, @javascriptInterface annotations for called functions are easy to exploit, because js side can call some local commands through reflection, which is dangerous.
- Through the WebViewClient
shouldOverrideUrlLoading ()
Method callback intercepts url
This method intercepts the URL with a shouldOverrideUrlLoading callback, then parses it, and calls the corresponding method if the protocol is agreed.
// Override the shouldOverrideUrlLoading method of the WebViewClient class
mWebView.setWebViewClient(new WebViewClient() {
@Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
Uri uri = Uri.parse(url);
// If the url protocol = the pre-agreed JS protocol
if ( uri.getScheme().equals("js")) {
// If authority = pre-specify webViews in the protocol, that is, represent the protocol that conforms to the convention
if (uri.getAuthority().equals("webview")) {
System.out.println("Js calls Android methods");
// Can take parameters on the protocol and pass them to Android
HashMap<String, String> params = new HashMap<>();
Set<String> collection = uri.getQueryParameterNames();
}
return true;
}
return super.shouldOverrideUrlLoading(view, url); }});Copy the code
How do I avoid WebView memory leaks
WebView memory leaks are mainly caused by WebView resources not being released immediately after the page is destroyed. There are two main approaches:
1) Instead of adding webView tags to the XML layout, use new tags in the code and free webView resources when the page is destroyed
//addview
private WeakReference<BaseWebActivity> webActivityReference = new WeakReference<BaseWebActivity>(this);
mWebView = new BridgeWebView(webActivityReference .get());
webview_container.addView(mWebView);
/ / destroy
ViewParent parent = mWebView.getParent();
if(parent ! =null) {
((ViewGroup) parent).removeView(mWebView);
}
mWebView.stopLoading();
mWebView.getSettings().setJavaScriptEnabled(false); mWebView.clearHistory(); mWebView.clearView(); mWebView.removeAllViews(); MWebView. Destroy (); mWebView=null;Copy the code
2) Start another process to load the WebView and kill the process after the page is destroyed. But the trouble with this approach is interprocess communication.
To use it, simply write the process name in the XML file and call System.exit(0) when it is destroyed.
<activity android:name=".WebActivity"
android:process=":remoteweb"/>
System.exit(0)
Copy the code
What else can be improved about webView
-
Pre-initialize or use the global WebView. The first initialization of the WebView is much slower than the second. After initialization, even if the WebView is released, some global service/resource pairs shared by multiple WebViews are not released, and the second initialization does not need to be generated, so the initialization is faster.
-
DNS uses the same domain name as the client API, and DNS resolution takes a lot of time. Therefore, DNS will be cached when using the same domain name as the client API, so opening webView will not take much time on DNS
-
For JS optimizations, try not to use heavy frameworks such as React. The second is whether high-performance pages or back-end rendering is required. Finally, the web framework in app should be unified, so that JS can be cached and reused.
Here is the summary plan of Meituan team, as follows:
- WebView initialization is slow
Initialize the
Request data first at the same time to keep the back end and the network busy. - Back-end processing is slow and can make the server
Points trunk output
The front-end also loads network static resources while the back-end calculates. - Script execution is slow, just
The script runs at the end
Does not block page parsing. - At the same time, reasonable
Preloading and precaching
Can make the loading speed bottleneck is smaller. - WebView initialization is slow, so anytime
Initialize the
What a WebView to use. - DNS and link slow, find a way to reuse client use
Domain names and links
. - Script execution is slow
Framework code split
Go out and do it before you request the page.
The relationship between activities, Views, and Windows.
Each Activity contains a Window object, which is implemented by PhoneWindow. PhoneWindow uses a DecorView as the root View of an application window, which in turn divides the screen into two regions: One is TitleView, one is ContentView, and the layout that we normally write in an Xml file happens to be displayed in a ContentView.
The Activity sets the View to the PhoneWindow using setContentView, and the View manages the View through addView(), removeView(), and updateViewLayout() of WindowManager. The process of adding a Window and the process of starting an Activity are both IPC processes. The Activity starts with AMS; The process of adding Windows is done through WindowSession.
Let’s talk about how Android’s event distribution mechanism works, which is how events are delivered from the moment the screen is clicked.
In my opinion, the process of event distribution mechanism can be divided into three parts: from out, from inside out, and after consumption.
1) First, from the outermost layer to the innermost layer:
If it is currently in the ViewGroup hierarchy, onInterceptTouchEvent is checked to see if it is true. If it is true, the event will be consumed in this hierarchy and will not be passed down. The onTouchEvent method of the current ViewGroup is then executed. If onInterceptTouchEvent is false, the event is passed to the dispatchTouchEvent method at the next level, followed by the same code logic, all the way to the innermost view.
Pseudocode interpretation:
public boolean dispatchTouchEvent(MotionEvent event) {
boolean isConsume = false;
if (isViewGroup) {
if (onInterceptTouchEvent(event)) {
isConsume = onTouchEvent(event);
} else{ isConsume = child.dispatchTouchEvent(event); }}else {
//isView
isConsume = onTouchEvent(event);
}
return isConsume;
}
Copy the code
2) After arriving at the innermost view, the view itself can still choose to consume or transmit to the outside.
Does the view have the right to reject the consumption event when it goes to the innermost layer and executes the onTouchEvent method? View, as the lowest level, should have no right to speak. However, as a matter of fairness, the View can also be rejected by returning false in the onTouchEvent method to indicate that it does not want to consume the event. Its parent’s onTouchEvent will be called again, and if the parent’s onTouchEvent returns false, the parent’s onTouchEvent will be called again. All the way up to the top level, the onTouchEvent for the Activity is called.
Pseudocode interpretation:
public void handleTouchEvent(MotionEvent event) {
if (!onTouchEvent(event)) {
getParent.onTouchEvent(event);
}
}
Copy the code
3) After consumption
When the onInterceptTouchEvent of a layer viewGroup is true, it means that the current layer will consume events. If its onTouchListener is set, onTouch will be called. If onTouch returns true, onTouchEvent will not be called. If false is returned or onTouchListener is not set, onTouchEvent continues to be called. The onClick method is set to onClickListener and will be called normally.
Pseudocode interpretation:
public void consumeEvent(MotionEvent event) {
if (setOnTouchListener) {
int tag = onTouch();
if (!tag) {
onTouchEvent(event);
}
} else {
onTouchEvent(event);
}
if(setOnClickListener) { onClick(); }}Copy the code
Solutions to sliding conflicts.
The fundamental solution to sliding conflicts is to intercept in the right place, so there are two solutions:
External intercept
: The event is processed from the parent view side, depending on whether the event is distributed to the child viewInternal intercept
: From the child view side processing, according to the situation to decide whether to prevent the parent view to intercept, the key isrequestDisallowInterceptTouchEvent
Methods.
OnInterceptTouchEvnet = onInterceptTouchEvnet = onInterceptTouchEvnet
// External interception: parent view.java
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
boolean intercepted = false;
// The parent view intercepts the condition
boolean parentCanIntercept;
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
intercepted = false;
break;
case MotionEvent.ACTION_MOVE:
if (parentCanIntercept) {
intercepted = true;
} else {
intercepted = false;
}
break;
case MotionEvent.ACTION_UP:
intercepted = false;
break;
}
return intercepted;
}
Copy the code
Return true to intercept, false to not intercept, and pass to the child view. Note that ACTION_DOWN status should not be intercepted. If intercepted, subsequent events will be directly handed over to the parent view, and there will be no interception.
- Internal interception methodIs through
requestDisallowInterceptTouchEvent
Method tells the parent view not to block.
/ / the parent view. Java
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (ev.getActionMasked() == MotionEvent.ACTION_DOWN) {
return false;
} else {
return true; }}/ / child view. Java
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
// The parent view intercepts the condition
boolean parentCanIntercept;
switch (event.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
getParent().requestDisallowInterceptTouchEvent(true);
break;
case MotionEvent.ACTION_MOVE:
if (parentCanIntercept) {
getParent().requestDisallowInterceptTouchEvent(false);
}
break;
case MotionEvent.ACTION_UP:
break;
}
return super.dispatchTouchEvent(event);
}
Copy the code
RequestDisallowInterceptTouchEvent (true) means to prevent the parent view intercept events, is introduced to true, the parent view will no longer call onInterceptTouchEvent. On the other hand, passing false means that the parent view can intercept, that is, the onInterceptTouchEvent method that goes to the parent view. So if you want the parent view to intercept, you pass flase, and if you want the parent view not to intercept, you pass true.
Fragment life cycle, life cycle changes when hide, show, replace
1) Life cycle:
onAttach()
Called when Fragment is associated with an Activity. You can get Activity references through this method, and you can get parameters through getArguments().onCreate()
: Fragment is called when the Fragment is created.onCreateView()
: Creates the Fragment layout.onActivityCreated()
: called when the Activity completes onCreate().onStart()
: called when the Fragment is visible.onResume()
: called when the Fragment is visible and interactive.onPause()
: called when the Fragment is not interactive but visible.onStop()
: called when Fragment is not visible.onDestroyView()
: called when the Fragment’s UI is removed from the view structure.onDestroy()
: called when destroying the Fragment.onDetach()
: called when the Fragment and Activity are disassociated.
Lifecycle changes for each called method:
add()
: onAttach () – >… – > onResume ().remove()
: onPause () – >… – > onDetach ().replace()
: is equivalent to the old Fragment calling remove() and the new Fragment calling add(). Add the life cycle of remove()+add()show()
Do not call any lifecycle methods. This method is called only if the Fragment has been added to the container, so that the Fragment UI setVisibility is true.hide()
Do not call any lifecycle methods. This method is called only if the Fragment has been added to the container and the Fragment UI setVisibility is false.
Activities and Fragments communicate with each other.
- Activities communicate with Fragments
The Activity has an instance of a Fragment, so you can either execute the Fragment method or pass in an interface. Similarly, fragments can get an instance of an Activity via getActivity() and can also execute methods.
- Fragment Indicates the communication between fragments
1) Get another instance of Fragmetn directly
getActivity().getSupportFragmentManager().findFragmentByTag("mainFragment");
Copy the code
2) The interface calls back into one Fragment to implement the interface, and the other Fragment sends in the instance of the interface.
3) Eventbus and other frameworks.
Fragment encountered any problems with viewPager.
-
While sliding, call the setCurrentItem method, paying attention to the second argument, smoothScroll. Pass false to jump straight to fragment, pass true to smooth over. In general, the home page switch page is false.
-
To disable preloading, the call to setOffscreenPageLimit(0) is invalid because the method determines if it is less than 1. You need to override the setUserVisibleHint method to determine whether the fragment is visible.
-
Do not use getActivity() to get an activity instance. It is easy to create a null pointer because the fragment will report a null pointer if it is already onDetach(). So inside the onAttach method, get the context of the activity.
-
FragmentStatePagerAdapter to the fragments destroyed beyond the limit, The life cycle is onPause->onStop->onDestoryView->onDestory->onDetach, onAttach->onCreate->onCreateView->onStart->onResume. It is possible to switch the Fragment onCreateView multiple times, so you need to be careful with the data.
-
Since onCreateView can be onCreateView multiple times, we can save the view and initialize the data if it is empty. See the code:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
if (null == mFragmentView) {
mFragmentView = inflater.inflate(getContentViewLayoutID(), null);
ButterKnife.bind(this, mFragmentView);
isDestory = false;
initViewsAndEvents();
}
return mFragmentView;
}
Copy the code
The principle of ARouter
First of all, what does ARouter do? ARouter is a framework developed by Alibaba to solve the problem of interface jump between components and modules. So simply speaking, is used to jump interface, different from the normal use of explicit or implicit jump, just need to add annotations on the corresponding interface, you can realize the jump, take a look at a case:
@Route(path = "/test/activity")
public class YourActivity extend Activity {... }/ / jump
ARouter.getInstance().build("/test/activity").navigation();
Copy the code
Easy to use, through a path can be jumped, so what is the principle?
In fact, if you think about it carefully, you can think that since the key jump process is to jump to a specific activity through the path, then the principle is nothing more than to match the path and activity. Yes, in fact, through annotations, apt technology, which is annotation processing tools, path and activity associated. There are mainly the following steps:
- It’s in the code
@Route
Apt generates class files that store the mapping between Path and activity.class at compile time - When the app starts, it loads these class files and reads the data that holds these mappings into memory (stored in a map).
- During route redirection, the route passes
build()
Method passes in a routing address to reach the page, and ARouter finds the activity.class corresponding to the routing address through its own stored routing table - then
new Intent
Method, if there is a callARouter
thewithString()
Method is calledintent.putExtra(String name, String value)
Method add parameter - The last call
navigation()
Method, which internally calls startActivity(Intent) to jump
How does ARouter implement page blocking
Here’s an example of an interceptor that can be used to verify login and then redirect to the login or target page:
@Interceptor(name = "login", priority = 6)
public class LoginInterceptorImpl implements IInterceptor {
@Override
public void process(Postcard postcard, InterceptorCallback callback) {
String path = postcard.getPath();
boolean isLogin = SPUtils.getInstance().getBoolean(ConfigConstants.SP_IS_LOGIN, false);
if (isLogin) {
// Do not intercept if you have logged in
callback.onContinue(postcard);
} else {
// If there is no login, interceptcallback.onInterrupt(postcard); }}@Override
public void init(Context context) {
LogUtils.v("Initialization successful"); }}/ / use
ARouter.getInstance().build(ConfigConstants.SECOND_PATH)
.withString("msg"."123")
.navigation(this.new LoginNavigationCallbackImpl());
// The second argument is the route jump callback
// Intercepting callback
public class LoginNavigationCallbackImpl implements NavigationCallback{
@Override
public void onFound(Postcard postcard) {}@Override
public void onLost(Postcard postcard) {}@Override
public void onArrival(Postcard postcard) {}@Override
public void onInterrupt(Postcard postcard) {
// Intercept and jump to the login pageString path = postcard.getPath(); Bundle bundle = postcard.getExtras(); ARouter.getInstance().build(ConfigConstants.LOGIN_PATH) .with(bundle) .withString(ConfigConstants.PATH, path) .navigation(); }}Copy the code
Interceptor implements the IInterceptor interface. Using the @Interceptor annotation, the Interceptor is automatically registered, using APT technology to automatically generate mapping classes. There is also a priority parameter, the smaller the value, the sooner it is executed.
How does that apply to componentization
First, add dependencies to the common component’s build.gradle:
dependencies {
api 'com. Alibaba: arouter - API: 1.4.0'
annotationProcessor 'com. Alibaba: arouter - compiler: 1.2.1'
}
Copy the code
Second, must be in each business component, that is, with the help of arouter components statement annotationProcessorOptions, otherwise you will not be apt to generate the index file, also can’t jump properly:
// Build. Gradle for business components
android {
defaultConfig {
javaCompileOptions {
annotationProcessorOptions {
arguments = [AROUTER_MODULE_NAME: project.getName()]
}
}
}
}
dependencies {
annotationProcessor 'com. Alibaba: arouter - compiler: 1.2.1'
implementation 'Common Component'
}
Copy the code
[AROUTER_MODULE_NAME: The project.getName()] key/value pair is passed to Arouter to make it easier for Arouter to use apt.
Then it’s ready for normal use.
Tell me what you know about coroutines
In my opinion, coroutines and threads are both solutions for concurrent tasks (asynchronous tasks). So coroutines and threads are on the same level, but coroutines in Kotlin are different from coroutines in general. The coroutine in Kotlin is actually a wrapper around threads, or a thread framework, to make asynchronous tasks easier to use.
Let’s talk about how coroutines are used
For example, when an asynchronous task needs to be called back to the main thread, the normal thread needs to use handler to switch threads and then perform UI updates, etc. It is even more inconvenient when multiple tasks need to be called sequentially, such as the following:
// The client makes three network asynchronous requests in sequence and updates the UI with the final results
thread{
iotask1(parameter) { value1 ->
iotask1(value1) { value2 ->
iotask1(value2) { value3 ->
runOnUiThread{
updateUI(value3)
}
}
}
}
}
Copy the code
It’s a hell of a call, if not three, but five, six times, that’s ok.
Coroutines can be used to solve this problem:
// Concurrent requests
GlobalScope.launch(Dispatchers.Main) {
// Three requests are made concurrently
val value1 = async { request1(parameter1) }
val value2 = async { request2(parameter2) }
val value3 = async { request3(parameter3) }
// Update the UI after all results are returned
updateUI(value1.await(), value2.await(), value3.await())
}
// Switch to the IO thread
suspend fun request1(parameter : Parameter){withContext(Dispatcher.IO){}}
suspend fun request2(parameter : Parameter){withContext(Dispatcher.IO){}}
suspend fun request3(parameter : Parameter){withContext(Dispatcher.IO){}}
Copy the code
Just like sequential execution in the same thread, or let’s say I want to do one asynchronous task sequentially, and then update the UI when I’m done, three asynchronous tasks. How should I write it normally?
thread{
iotask1() { value1 ->
runOnUiThread{
updateUI1(value1)
iotask2() { value2 ->
runOnUiThread{
updateUI2(value2)
iotask3() { value3 ->
runOnUiThread{
updateUI3(value3)
}
}
}
}
}
}
}
Copy the code
Dizzy dizzy, isn’t it an asynchronous task, a UI update. So let’s see what we can do with coroutines:
GlobalScope.launch (Dispatchers.Main) {
ioTask1()
updateUI1()
ioTask2()
updateUI2()
ioTask3()
updateUI3()
}
suspend fun ioTask1(a){
withContext(Dispatchers.IO){}
}
suspend fun ioTask2(a){
withContext(Dispatchers.IO){}
}
suspend fun ioTask3(a){
withContext(Dispatchers.IO){}
}
fun updateUI1(a){}fun updateUI2(a){}fun updateUI3(a){}Copy the code
How do coroutines cancel
- cancel
Coroutine scope
Will cancel all of its subcoroutines.
// Coroutine scope scope
valJob1 = scope.launch {... }valJob2 = scope.launch {... } scope.cancel()Copy the code
- cancel
The child coroutines
// Coroutine scope scope
valJob1 = scope.launch {... }valJob2 = scope.launch {... } job1.cancel()Copy the code
However, calling Cancel does not immediately stop work in the coroutine, and it does not prevent the code from running. For example, joB1 is normally in the active state. After calling the cancel method, the coroutine will become Cancelling and Cancelled after the work is completed. Therefore, the coroutine can be stopped by judging its status.
The coroutine scopes (viewModelScope and lifecycleScope) defined in Jetpack help you automatically cancel tasks, more on that next time, but otherwise bind and cancel yourself.
Before you should have read my startup process analysis, that article I said that the purpose of source analysis has not been to learn knowledge and learn, but to understand these foundations, we can better solve the problem. So today, let’s look at how we can specifically optimize the startup process by analyzing the app startup process.
- What can we optimize in the App launch process?
- What are the specific optimization methods?
- Method for analyzing the startup time
What are the specific startup optimization methods?
- Blinding screen page
To eliminate the white/black screen at startup, you can set Android :windowBackground to make it feel as if the startup is finished by clicking the icon.
<activity android:name=". The UI. The activity. Start the activity"
android:theme="@style/MyAppTheme"
android:screenOrientation="portrait">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<style name="MyAppTheme" parent="Theme.AppCompat.NoActionBar">
<item name="android:windowBackground">@drawable/logo</item>
</style>
Copy the code
- To create the Activity in advance
When an object is first created, the Java virtual machine first checks whether the corresponding Class object has been loaded. If it is not loaded, the JVM looks for the.class file based on the class name and loads its class object. When the same class is new the second time, it does not need to load the class object, but directly instantiates it, and the creation time is shortened.
- Lazy loading of third-party libraries
Many third party open source libraries say that initialization is done in the Application, so you can make the Application lighter by putting the initialization of some third party libraries that are not initialized on demand.
- WebView startup optimization
The first startup of webView will be very time-consuming. For specific optimization methods, please refer to my previous article on webView optimization.
- Thread optimization
Threads are the basic unit in which programs run, and frequent creation of threads is costly, so everyone should use thread pools. In the case of a single CPU, even if multiple threads are open, only one thread can work at the same time, so the size of the thread pool depends on the number of cpus.
- MultiDex optimization
Due to 65536 method limitations, so the general class file to generate multiple dex files, below Android5.0, ClassLoader will only load from the class. Dex (main dex), Therefore, you need to run multidex.install (context) to read the dex class normally.
The install method is time-consuming. It decompresses APK, traverses dex files, compresses DEX, converts DEX files into DexFile objects through reflection, and reflects replacement arrays.
The solution needed is the Toutiao solution:
1. In the attachBaseContext method of the Application, start the LoadDexActivity of another process to asynchronously execute the MultiDex logic, showing Loading. 2. Then the main process Application enters a while loop to check whether the MultiDex operation is complete. The ContentProvider initializer and the Application onCreate method, which executes the normal logic of the main process.
So the key is to open the process to execute the MultiDex logic, so that the APP does not affect the startup.
Method for analyzing the startup time
- Systrace + function peg
That is, by adding statistical codes at the entry and exit points of the method, the method takes time
class Trace{
public static void i(String tag){
android.os.Trace.beginSection(tag);
}
public static void o(a){ android.os.Trace.endSection(); }}void test(a){
Trace.i("test");
System.out.println("doSomething");
Trace.o();
}
Copy the code
- BlockCanary
BlockCanary monitors the main thread time by logging where the main thread message loop takes place. When a message takes longer than the threshold, it records the status of various resources in the system and displays them. So we set the threshold low so that if a method takes more than 200 milliseconds to execute, we get the stack information.
As we have said before, when looper() is used to cycle to fetch MSG from MessageQueue, there will be logging log printing before and after dispatchMessage method, so only a Printer needs to be customized. Rewrite the println(String x) method to implement the time statistics.
OnCreate, onResume, onStart, where can we get wide high
If you call the getWidth/getHeight method on onCreate, onStart, and onResume, you will not get the correct information about the View’s width and height because the View’s measure process is not synchronized with the Activity’s life cycle. So there is no guarantee that the measure of the view has been completed during these life cycles. So it’s very possible to get a width and height of 0.
So there are three main methods to get the width and height of a view:
- The post () method
The runnable object in this method ensures that the view has been drawn by executing the measure, Layout, and draw methods.
view.post(new Runnable() {
@Override
public void run(a) {
int width = view.getWidth();
inthight = view.getHeight(); }});Copy the code
- OnWindowFocusChanged method
The onWindowFocusChanged method can be overwritten in the Activity, which indicates when the Activity window gets focus or loses focus. Therefore, when Activitiy gets focus, the View must be drawn, and there is no problem in obtaining width and height:
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
if(hasFocus){
int width = view.getWidth();
inthight = view.getHeight(); }}Copy the code
- ViewTreeObserver registers the OnGlobalLayoutListener interface
The ViewTreeObserver is an observer used to observe changes to the view tree. The OnGlobalLayoutListener method will be called back when the state of the View tree changes or the visibility of a View changes. Therefore, it is also possible to obtain the width and height of the view.
ViewTreeObserver observer = title_name.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout(a) {
int width = view.getWidth();
inthight = view.getHeight(); }});Copy the code
View. post can obtain width and height.
The reason we get the width and height is because the view has already been drawn, so the task added by view.post () is guaranteed to be executed after all the view drawing processes have finished.
Look at the source code for post:
public boolean post(Runnable action) {
final AttachInfo attachInfo = mAttachInfo;
if(attachInfo ! =null) {
return attachInfo.mHandler.post(action);
}
// Assume that post will succeed later
ViewRootImpl.getRunQueue().post(action);
return true;
}
//RunQueue .class
void post(Runnable action) {
postDelayed(action, 0);
}
void postDelayed(Runnable action, long delayMillis) {
HandlerAction handlerAction = new HandlerAction();
handlerAction.action = action;
handlerAction.delay = delayMillis;
synchronized(mActions) { mActions.add(handlerAction); }}void executeActions(Handler handler) {
synchronized (mActions) {
final ArrayList<HandlerAction> actions = mActions;
final int count = actions.size();
for (int i = 0; i < count; i++) {
finalHandlerAction handlerAction = actions.get(i); handler.postDelayed(handlerAction.action, handlerAction.delay); } actions.clear(); }}Copy the code
So when the view.post () method is executed, the Runnable is not executed immediately, but is saved to the RunQueue and then executed through the executeActions method, i.e. by the handler, a delayed task Runnable is posted. And when will the executeActions method be executed?
private void performTraversals(a) { getRunQueue().executeActions(attachInfo.mHandler); . performMeasure(); . performLayout(); . performDraw(); }Copy the code
You can see that it is executed in performTraversals, but before the View is drawn, this is because the runnable that needs to be executed is wrapped as a Message and queued in MessageQueue before rendering, but Looper doesn’t fetch the Message immediately, Since Looper fetches messages sequentially, what messages are left on the main thread? So the performMeasure, performLayout, and performDraw traversals will all be executed. The main reason we can get the width and height of the runnable in the view.post method.
How is SharedPreferences thread-safe, and what locks are used by its internal implementation
The essence of SharedPreferences is to store data as key-value pairs to XML files and then read and write the files.
- For read operations, a lock is sufficient:
public String getString(String key, @Nullable String defValue) {
synchronized (mLock) {
String v = (String)mMap.get(key);
returnv ! =null? v : defValue; }}Copy the code
- For write operations, there are two locks, one is editor.put and the other is commit or apply:
// The first lock to operate on the map object
public final class EditorImpl implements Editor {
@Override
public Editor putString(String key, String value) {
synchronized (mEditorLock) {
mEditorMap.put(key, value);
return this; }}}// Write to file
synchronized (mWritingToDiskLock) {
writeToFile(mcr, isFromSyncCommit);
}
Copy the code
Is the process safe? What do we as developers do if it’s not secure?
1) SharedPreferences is process insecure because cross-process locks are not used. Since the process is not secure, data exceptions may occur during multi-process operations.
2) There are two ways to secure the process:
- Use a cross-process component, also known as a ContentProvider, which is officially recommended. Multiple processes are processed through ContentProvider so that different processes access SharedPreferences through ContentProvider.
- Add file lock, because the nature of SharedPreferences is to read and write files, so we can lock the file, to ensure that the process is safe.
Why plug-in
I think the main reason is the ability to dynamically expand functionality. Some uncommon functions or modules into plug-ins, can reduce the size of the original installation package, so that some functions in the form of plug-ins when needed to be loaded, that is, to achieve dynamic loading.
For example, dynamic skin, festival promotion, some shady functions, you can download the apK of the corresponding mode when you need it, and then load the function dynamically. Therefore, this function is generally applicable to some platform projects, such as Dianping Meituan. There are many functions, and users are very likely to only use some of them. Moreover, these modules can be run as an app when taken out separately.
But now it is rarely used. See point 3 for details.
The principle of plug-in
To implement plug-in, that is, reading all data from APK, consider three issues:
Reading plug-in code
To complete the loading of the code in the plug-in and the mutual call with the main projectReading plug-in resources
To complete the loading of resources in the plug-in and mutual access with the main projectFour Components Management
1) Read the plug-in code, which is essentially a plug-in class load. So you can just use the class loader. There are two kinds of loaders commonly used in Android, DexClassLoader and PathClassLoader, which inherit from BaseDexClassLoader. The difference is that DexClassLoader passes a parameter optimizedDirectory, indicating that we need to cache the dex file, and create a DexFile object, and this path must be an internal storage path. The PathClassLoader parameter is null, which means that the file will not be cached in the internal storage space, but will be directly loaded using the original file path. Therefore, DexClassLoader is more powerful and can load external DEX files.
Also, because of the parent delegate mechanism, the main project’s ClassLoader is passed as the parent loader when the plug-in’s ClassLoader is constructed, so the plug-in can directly refer to the main project’s class by class name. The main project call plug-in needs to load the class through the DexClassLoader and then call the method reflectively.
2) Read plug-in resources, mainly through AssetManager access. The specific code is as follows:
/** * Load the plugin's resources: Add the plugin's APK resource path */ through the AssetManager
protected void loadPluginResources(a) {
// reflection loads resources
try {
AssetManager assetManager = AssetManager.class.newInstance();
Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);
addAssetPath.invoke(assetManager, mDexPath);
mAssetManager = assetManager;
} catch (Exception e) {
e.printStackTrace();
}
Resources superRes = super.getResources();
mResources = new Resources(mAssetManager, superRes.getDisplayMetrics(), superRes.getConfiguration());
}
Copy the code
To access the resources of the plug-in, pass the plug-in path through the addAssetPath method.
3) Four components management why single out the four components? Because the big four components not only load their classes, but also manage their life cycle, registered in androidmanifest.xml. This is also an important part of plug-in. Here we focus on activities.
The main implementation method is through Hook technology. The main scheme is to use an Activity registered in androidmanifest.xml to trap the Activity, which is used to pass the AMS verification, and then replace the Activity with plug-in Activity at the appropriate time.
Hook technology is also called Hook function. Before the system calls the function, the Hook program will first capture the message, and the Hook function will first get control. At this time, the Hook function can not only process (change) the execution behavior of the function, but also force to end the message transmission. In a nutshell, it’s about pulling out the system’s programs and making snippets of code that we execute ourselves.
A hook can change the internal behavior of a function.
Here, the plug-in Activity is loaded using hook technology, there are two points can hook, respectively:
- Hook IActivityManager
As mentioned above, the Activity registered in androidmanifest.xml will be the first to trap, and then replace the Activity loaded when appropriate. So we need two main steps: Step 1: Use this Activity to complete the AMS validation. Let AMS know that the Activity we want to start is registered with XML. The specific code is as follows:
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if ("startActivity".contains(method.getName())) {
/ / a change
Intent intent = null;
int index = 0;
for (int i = 0; i < args.length; i++) {
Object arg = args[i];
if (arg instanceof Intent) {
// The Intent parameter of startActivity is found
intent = (Intent) args[i];
This intent cannot be started because Acitivity is not registered in the manifest fileindex = i; }}// Forge a proxy Intent that launches a proxyActivity
Intent proxyIntent = new Intent();
ComponentName componentName = new ComponentName(context, proxyActivity);
proxyIntent.setComponent(componentName);
proxyIntent.putExtra("oldIntent", intent);
args[index] = proxyIntent;
}
return method.invoke(iActivityManagerObject, args);
}
Copy the code
Step 2: Replace back our Activity. The previous step is to change the actual Activity we want to start to the Activity registered in our XML to avoid validation, so we need to change the Activity back later. The last step in the Activity launch process is actually handled by the handleMessage method overridden in H (a handler), which eventually calls the Activity’s onCreate method. The dispatchMessage method to the Handler is called to process the message. If the Handler’s mCallback type is not null, the handleMessage method of the mCallback is executed. So the point we can hook is this mCallback.
public static void hookHandler(a) throws Exception { Class<? > activityThreadClass = Class.forName("android.app.ActivityThread");
Object currentActivityThread= FieldUtil.getField(activityThreadClass ,null."sCurrentActivityThread");/ / 1
Field mHField = FieldUtil.getField(activityThread,"mH");/ / 2
Handler mH = (Handler) mHField.get(currentActivityThread);/ / 3
FieldUtil.setField(Handler.class,mH,"mCallback".new HCallback(mH));
}
public class HCallback implements Handler.Callback{
/ /...
@Override
public boolean handleMessage(Message msg) {
if (msg.what == LAUNCH_ACTIVITY) {
Object r = msg.obj;
try {
// Get the Intent in the message (the Intent that started the SubActivity)
Intent intent = (Intent) FieldUtil.getField(r.getClass(), r, "intent");
// Retrieve the previously saved Intent(the Intent that started the TargetActivity)
Intent target = intent.getParcelableExtra(HookHelper.TARGET_INTENT);
// Replace the Intent that starts SubActivity with the Intent that starts TargetActivity
intent.setComponent(target.getComponent());
} catch (Exception e) {
e.printStackTrace();
}
}
mHandler.handleMessage(msg);
return true; }}Copy the code
Replacing the mCallback in mH with a custom HCallback completes the Activity replacement.
- Hook Instrumentation
This method can be done by replacing Instrumentation because the startActivityForResult method calls the execStartActivity method of the Instrumentation to activate the Activity lifecycle. Then in the Instrumentation execStartActivity method using the pit SubActivity to pass the AMS validation, in the Instrumentation newActivity method restore TargetActivity.
public class InstrumentationProxy extends Instrumentation {
private Instrumentation mInstrumentation;
private PackageManager mPackageManager;
public InstrumentationProxy(Instrumentation instrumentation, PackageManager packageManager) {
mInstrumentation = instrumentation;
mPackageManager = packageManager;
}
public ActivityResult execStartActivity(
Context who, IBinder contextThread, IBinder token, Activity target,
Intent intent, int requestCode, Bundle options) {
List<ResolveInfo> infos = mPackageManager.queryIntentActivities(intent, PackageManager.MATCH_ALL);
if (infos == null || infos.size() == 0) {
intent.putExtra(HookHelper.TARGET_INTENsT_NAME, intent.getComponent().getClassName());/ / 1
intent.setClassName(who, "com.example.liuwangshu.pluginactivity.StubActivity");/ / 2
}
try {
Method execMethod = Instrumentation.class.getDeclaredMethod("execStartActivity",
Context.class, IBinder.class, IBinder.class, Activity.class, Intent.class, int.class, Bundle.class);
return (ActivityResult) execMethod.invoke(mInstrumentation, who, contextThread, token,
target, intent, requestCode, options);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
public Activity newActivity(ClassLoader cl, String className, Intent intent) throws InstantiationException,
IllegalAccessException, ClassNotFoundException {
String intentName = intent.getStringExtra(HookHelper.TARGET_INTENT_NAME);
if(! TextUtils.isEmpty(intentName)) {return super.newActivity(cl, intentName, intent);
}
return super.newActivity(cl, className, intent); }}public static void hookInstrumentation(Context context) throws Exception { Class<? > contextImplClass = Class.forName("android.app.ContextImpl");
Field mMainThreadField =FieldUtil.getField(contextImplClass,"mMainThread");/ / 1
Object activityThread = mMainThreadField.get(context);/ / 2Class<? > activityThreadClass = Class.forName("android.app.ActivityThread");
Field mInstrumentationField=FieldUtil.getField(activityThreadClass,"mInstrumentation");/ / 3
FieldUtil.setField(activityThreadClass,activityThread,"mInstrumentation".new InstrumentationProxy((Instrumentation) mInstrumentationField.get(activityThread),
context.getPackageManager()));
}
Copy the code
Some plug-in solutions on the market and your ideas
A few years ago, plugins were very popular, such as dynamic-load-apk, DroidPlugin, RePlugin (360), VirtualApk (didi), but now the opportunity is not in operation, many frameworks only support Android9 at most.
Why is that? I think one is that the maintenance cost is too high to be compatible, every time the source code is updated, it is necessary to maintain again. Second, it is true that plug-in technology is not used much now, why used to plug-in framework? For example, new functions are added to decouple functional modules. You now have RN for plug-in functionality, and you have componentization for project decoupling. So not many people use it.
Although plugins are not used much anymore, I think the technology can still be understood, and hot updates are mainly used by these technologies. Solutions can be obsolete, but technologies can’t.
reference
View. Post View. Post SharedPreferences
conclusion
Hope to give you a little help, of course, THE article I will continue to write, feel everyone to me before the point of praise, hey hey.
‘!
There is a small partner to study together can pay attention to my public number – code on the building blocks .