Android performance optimization startup speed optimization
Android app startup speed optimization, first of all to talk about why to optimize this step, if the start of the app project when the creation of this speed into consideration, then certainly do not need to optimize again. This is because in the era of mobile Internet, we are all in pursuit of fast, what is the first to do, the other can not be considered, first take the initiative, or verify whether it is worth doing. So why do it? My personal views are as follows
- If your app can’t be developed quickly, you may not even know if it’s worth doing. If you spend a lot of time doing a feature that doesn’t add value to users, you might as well not do it
- If the app is not built quickly, it may be captured by competitors and the best business opportunity may be missed
- If at the beginning of the regulation can not affect the start speed of this goal, then do the function of the time will be bound, not fast up
- In the early days of the app, everyone was so busy developing new features and iterating on new versions that they didn’t have time to stop and polish
- As the number of apps of the same type increases and the number of competitors increases, people begin to pay attention to startup performance and start optimization of startup speed (both active and passive optimization).
1. Causes of performance problems
With the continuous rapid iteration of the project, App startup is often slow, because a lot of logic for initializing other businesses may be put in the startup stage of the main process of the App or the startup stage of the main interface, and these businesses may not be needed at the beginning. In this paper, from the author’s personal experience to describe the start speed optimization related dribs and drabs, for the start speed optimization to provide a way of thinking for your reference.
Second, why do start speed optimization
The App startup delay affects the uninstallation rate and usage of an App. Fast startup speed will give a feeling of lightness and reduce user waiting time. If an App takes 10 seconds from clicking on the desktop icon to seeing the home screen, can you accept that? If you don’t want to uninstall it, you may want to press the Home button and kill it. As a result, App uninstallation rates increased and usage decreased. These performance details are important for apps with a large number of users, because users are money.
Analysis and formulation of optimization technical route
3.1 Analyzing startup performance bottlenecks
Before specific optimization, first we have to find the need to optimize the place, how to find? This requires understanding of the startup principle of Android App. We need to know what steps an App has gone through in the whole process from clicking the desktop icon to seeing the main interface of the App, and where we can optimize it. Below is a rough description of the App startup process.
Specific code flow, analysis of key function time
The sequence of onFirstDrawFinish and onWindowFocusChanged in the figure may be reversed, but the time difference is not significant.
3.2 Formulating optimization direction
As can be seen from the above analysis, we optimized the startup process of App including the main process startup process and the main interface startup process. The main process startup is the creation process of Application, and the main interface startup is the creation process of MainActivity. You just need to optimize these two parts separately.
- In the Application, attachBaseContext is called first, followed by onCreate. Try to avoid time-consuming operations in these two methods.
- In MainActivity, we focus on onCreate, onResume, onWindowFocusChange and the end mark of the Activity when it starts and finishes. Instead of using the life cycle function, we use the first drawing of the main interface View as the mark when the Activity starts and finishes. The View being drawn for the first time proves that the View is going to be displayed and we’re going to see it. So we add a custom View to the Activity root layout, with its onDraw method calling back for the first time as a signal that the Activity has started.
public class FirstDrawListenView extends View { private boolean isFirstDrawFinish = false; private IFirstDrawListener mIFirstDrawListener; public FirstDrawListenView(Context context, AttributeSet attrs) { super(context, attrs); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); if (! isFirstDrawFinish) { isFirstDrawFinish = true; if (mIFirstDrawListener ! = null) { mIFirstDrawListener.onFirstDrawFinish(); } } } public void setFirstDrawListener(IFirstDrawListener firstDrawListener) { mIFirstDrawListener = firstDrawListener; } public interface IFirstDrawListener { void onFirstDrawFinish(); }}
How to view the comparison of data before and after optimization
From the above analysis, we can count the time it takes to start a process at each stage, as well as the time it takes to start an Activity at each stage (this step requires adding an extra custom empty View to the main layout to listen for the first callback of its onDraw method). We can collect this data through buried data. Before optimization, buried point data can be added first to report buried points in each time period. Therefore, it is necessary to send a version to verify the situation before optimization. After the mechanism of statistics is added, we can start to optimize, and we can see the comparison before and after optimization very clearly.
5. Set optimized goals
Since the App startup speed varies greatly from device to device, it is not easy to set a goal, but you must have a goal to do things. First of all, we use a familiar concept of “second on”, followed by cold start and hot start separately, and then separate different models (high-end machine, mid-end machine, low-end machine). Finally, we need to see the startup data before optimization. This allows you to define goals like the following:
- Open in 1 second for high-end models (such as mi 5, Android6.0 and above)
- Open in 1.5 seconds for mid-range models
- Low end models open in 2.5 seconds
The above is the ultimate goal. When it comes to real optimization, you should set your own optimization goal based on the actual data of the App and the actual situation of the team.
6. Specific steps of optimization
In general, the best way to optimize quickly is to put operations that don’t need to be done in advance into asynchronous threads, which we often do asynchronously. In addition to asynchronous loading, some code with a real performance impact needs to be specifically optimized. The following are some specific optimization implementation steps.
6.1 Encapsulates a helper class for printing point-of-time logs
In order to quickly locate time-consuming code blocks during optimization, we need to add logs before and after time-consuming code blocks to count the specific elapsed time. This helps us to quickly analyze and locate the time-consuming code blocks in Debug mode, and then we can start the specific time-consuming code blocks to see how to optimize.
6.2 Asynchronous Loading I: Add asynchronous threads to Application
Encapsulate two methods in Application: OnSyncLoad (synchronous loading) and onAsyncLoad (asynchronous loading, executed in Thread), put all the parts that do not need synchronous loading into onAsyncLoad method, and put all the methods that need synchronization into onSyncLoad method. This simple classification can lead to a good optimization effect.
Public class StartUpApplication extends Application {@override public void onCreate() {public class StartUpApplication extends Application {@override public void onCreate() { Otherwise it will slow down the startup of the entire app super.oncreate (); onSyncLoadForCreate(); } @Override protected void attachBaseContext(Context base) { super.attachBaseContext(base); onSyncLoad(); onAsyncLoad(); } private void onSyncLoadForCreate() { AppStartUpTimeLog.isColdStart = true; Applog. log("StartUpApplication onCreate"); AppStartUpTimeLog.logTimeDiff("App onCreate start", false, true); BlockingUtil.simulateBlocking(500); / / simulation block 100 milliseconds AppStartUpTimeLog. LogTimeDiff (" App onCreate end "); } private void onSyncLoad() { AppLog.log("StartUpApplication attachBaseContext"); AppStartUpTimeLog.markStartTime("App attachBaseContext", true); BlockingUtil.simulateBlocking(200); / / simulation block 100 milliseconds AppStartUpTimeLog. LogTimeDiff (" App attachBaseContext end ", true); } public void asyncload () {new Thread(new Runnable() {@override public void run() { "ApplicationAsyncLoad").start(); }}
6.3 Asynchronous Loading 2: Add asynchronous threads to the MainActivity
This step is the same as the optimization idea of Application. It also encapsulates the onSyncLoad and onAsyncLoad methods to classify the existing code, but these two methods are called later when the first screen of the main interface is drawn. This step also requires a new Thead, which is an additional overhead, but it does not affect our overall performance.
6.4 Lazy Loading: Load after the first screen is drawn
There’s something else that you have to do in the UI thread, but it doesn’t have to be done that fast, so let’s put this in after the first screen is drawn, and we put an empty View in the main layout to listen for its first onDraw callback, We interface this event into our MainActivity (the Activity implements onFirstDrawFinish on the interface). To get the user to see the main screen as quickly as possible, we can put in onFirstDrawFinish operations that need to be performed in the UI thread but not as quickly.
6.5 Dynamically loaded Layout: Main layout file optimization
Use the ViewStub or addView to dynamically load all the layouts in the main screen that do not need to be used the first time.
6.6 In-depth optimization of the main layout file
If you’re still slow to get to the main screen after all these optimizations, focus on the main layout file. The complexity of the main layout file has a direct impact on the speed of Activity loading, which requires in-depth optimization of the main layout file. When the Activity loads the layout, it will parse, measure, layout and draw the entire layout file, so it is particularly important to design a simple and reasonable layout. Layout optimization is not detailed, many online articles. Several important optimizations are as follows:
- Reduce layout levels
- Reduce the number of views loaded for the first time
- Reduce overdrawing
If you want to see how long it took for the main layout to load, use a custom ViewGroup as the root element of the root layout, and then monitor its onInflateFinished, onMeasure, onLayout, and onDraw methods. You can print some critical logs to figure out the specific time-consuming steps, and you can also locate which View takes the most time to load.
6.7 In-depth optimization of function codes
In the previous optimization step, we put some time-consuming operations after onFirstDrawFinish is drawn on the first screen. This will cause an experience problem. Although it is faster to enter the main screen, the UI thread may be blocked for a short time after entering the main screen. Because onFirstDrawFinish hangs a lot of time-consuming operations, the UI thread needs to wait for them to finish before it can be idle. So we also need to optimize some of the functional code to make sure that it actually takes less time. Additionally we asynchronous loading of the operation there is a safety risk in the thread, if some operation is very time consuming, may cause we enter the main interface to use data isn’t ready, so the asynchronous loading we should pay attention to the order of the code block, if some very time-consuming operation considering on separate threads to deal with.
Seven,
Optimization is an ongoing process that allows us to learn what factors affect startup performance so that we can pay more attention to the performance of our code during coding. This article looks at the whole startup performance optimization from a global perspective. It may seem easy, but in practice the optimization may not go smoothly, it may behave differently on different devices, and it may sometimes take time to start a service. So if you really want to be time-efficient, that’s the big one: delete it.
More performance optimization solutions
Fragment+ViewPager/ Screen Adaptation /LeakCanary/ Boot Optimization/Memory /App Slimming