preface
There are several main reasons to introduce DevTools ð :
- The first is
DevTools
Itself isFlutter
An official recommended debugging tool. DevTools
Is to useFlutter
Prepared, very distinctive ðž.- Has a very comprehensive debugging function, can meet the size, aspects of different optimization needs âïļ.
DevTools
First, introduce today’s main character: DevTools ð
About the official website: DevTools is a suite of performance and debugging tools for Dart and Flutter. It’s currently in beta release, but is under active development.
DevTools is a set of performance and debugging tools for Dart and Flutter. It is currently in beta but under active development.
Install ð
To use DevTools, first of all, the premise of everything, of course, is to install it. Yes, this seemingly simple step actually took me quite a while. Good luck with the installation, and I’m going to show you four ways to open DevTools.
Android Studio / Intellij
For those of you who use the above two development tools, check out ðâïļ. There are three steps to start DevTools on these tools
- The installation
Flutter
Plug-in:
First make sure you have installed it in the development toolFlutter
Plug-in (won’t someone hasn’t installed it, won’t ð)
- Start debugging an application ðĐ
Yes, literally, allowing you to debug the project (Profile mode is recommended). Note: Profile mode only works on the real machine ðą.
- find
DevTools
The entrance of the
Actually,DevTools
The entrance is not obvious, but it is easy to find, at the bottom of the toolbar, we often usehot reload
Click the button ð to start the projectDevTools
.
PS: I use Android Studio myself, but I haven’t succeeded in this way, I’ve been Installing DevTools… If you know why, please feel free to comment in the comments section, Ali Gardo ðĪ)
VS Code
- The installation
Flutter
Plug-in:
Make sure you have the Flutter and Dart plug-ins installed in the developer tools.
-
Start the Debugging session by opening the root directory of your project (including pubspec.yaml) in VS Code and clicking Run > Debugging (F5).
-
Start development program
- Once the debug session is active and the application is open ððŧ, then
VS Code
Will be displayed on the command control boardDart: Open DevTools
- When you first run it (and when you update the development kit in the future), you will be reminded to activate or upgrade
DevTools
.
Command Line
- Installing developer Tools
If you have pub in your environment variable PATH, run ð :
pub global activate devtools
Copy the code
If the environment variable PATH contains a flutter, you can run ð :
flutter pub global activate devtools
Copy the code
- Start developer tools services
Next, start the local Web Server service to run the developer tools. Run one of the following two commands.
pub global run devtools # If you have `pub` on your path.
Copy the code
flutter pub global run devtools # If you have `flutter` on your path.
Copy the code
After running these two lines of code, you should get something like this output on the command line:
An Observatory debugger and profiler on iPhone X is at: http://127.0.0.1:50976/Swm0bjIe0ks=/Copy the code
at
And this isDevTools
You can use it directlyChrome
Just open it. The page that opens requires you to fill in a link to the project you want to debugDebug service
Listening address, where did you get this address ðĪ?
3. Start oneapp
æĨ debug
In fact, the above link ð address, we usually run the projectLog
In the. You run the project, which can be found in the first few lines of the logDebug service
Is the address in the figurelistening on
In the following part, fill this link into the web page we opened in the second step, and then enter the corresponding one of our projectDevTools
.
Open the violence
The above three ways of Flutter installation and opening are provided on the official website. Unfortunately, I failed to open ðĪŊ. The installation was not complete, or I opened a very old version of DevTools, which should be related to my Dart SDK path configuration. Found a very happy ðĪŦ open method (currently do not know what drawbacks).
- find
Flutter
Folder address.
This varies from person to person. You should know the path of flutter on your computer. For example, I put it in my personal development directory:
/Users/tys/development/flutter
Copy the code
- find
DevTools
This is the key to violence on my side ð, I find… This DevTools seems to have been quietly placed in the Flutter.
/Users/tys/development/flutter/.pub-cache/bin/devtools
Copy the code
In the above directory, you can directly find it, double click open, direct good guy ðĄ
Wuhu ~ directly to the last step with the command line open. And that’s what you’re currently usingFlutter
The latest and available corresponding to the versionDevTools
!
Using DevTools ðĩ ðŧ came ïļ ‘
I’ll walk you through each feature in the order of the toolbar at the top of DevTools ðĪ.
Flutter inspector
- Brief introduction: The function of this part is mainly interface module for you, which will first display your overall layout drawing tree ðē, click each node of the tree will display more detailed layout information on the right ðģ, at the same time, on the layout information you can see the visual view information and some simple selection operations. In official terms, this feature has two main purposes:
1. Know the existing layout.
2. Diagnose layout problems.
The contents of this module are shown at ð in the order highlighted in red in the figure below
Select widget mode
This is actually a selection view, you can see the Tab corresponding to your entire drawing tree ðē, such a tree structure complexity is generally related to your page complexity. If it is a complex page structure, the tree will be very large and look very fuzzy. In order to better see the detailed structure of the tree, we can click on a node in the tree to view the detailed content of a part of the tree ðŪ.
Detail Tree
As the name implies, in fact, here is the detailed information of the node we just selected, in fact, the actual information can really be said to be very heart âĪïļ. In the image, I select a Container node. The unassigned properties of the Container are displayed, such as the BG background color. If your page has color, color overlay and so on, it can be seen clearly ðŽ.
Layout Explorer
This is the layout browser for your currently selected node. For example, you can see that under the Container I selected, the detailed structure of the package is displayed, including the boundaries of each Wiget, and even the calculated height and width ð (according to the standard resolution size).
The Layout Explorer not only allows you to see the entire Layout, but also allows you to do some simple dynamic operations to identify Layout problems or defects without changing the code (see giFs at âŽïļ).
Slow Animations
Click this button to reduce the speed of all Animations in your App and experience the magical world at 0.5 times speed ðĪĐ. The Animations mentioned here include but are not limited to your interface jump, Hero and other system-level Animations. Personally, there are two main functions of these functions: 1. If there is no lag in your animation in this mode, your animation performance is very perfect, can achieve a better user experience ð. 2. Let some animators see their own animation drawing curve, path ðģ.
Debug Paint
Add visual debugging prompts to render to show borders, fills, alignments and intervals. Your simulator displays the overall layout, including your overall rendering orientation and base component classes.
Paint Baselines
Make each RenderBox draw a line at each of its text baselines. This is the green line âŽïļ in the picture below
Repaint Rainbow
Turn this on and your interface will change a border color ð on the redrawn part of the interface when redrawn. For example, if you have a banner and change images every 1s, the color around your banner will change to a random color every second. Personally, the goal is to let us see what part of the current page is being drawn or redrawn.
Timeline ð
Use the actual device to debug performance. Skia has two very different backends. The Flutter uses a pure CPU back end in the iOS emulator and a GPU hardware accelerated back end in real devices, so its performance features are quite different.
TimeLine
isDevTools
The more practical and meaningful part of the. It allows you to actually see yoursAPP
The existence ofUI
A problem orGPU
The problem. Now I’m going to take you to learn how to analyze yourself, rightFlutter APP
The rendering performance of ðĐš.
The first thing you can see in the top part of the figure is a bar chart ð. This part is a state diagram of what you just did, and you can see what it means based on the legend on the right, so LET me clarify it for you.
UI, Raster, Jank
If you look at the bar chart, you see red and blue, light blue is the drawing of the UI thread, and dark blue is your rasterization thread (GPU thread). Red, represents the appearance of Caton on your drawing. Generally speaking, we define lag in Flutter: if the render time of a frame exceeds 16ms, the frame is considered delayed. In order to achieve a frame rendering rate of 60fps (frames per second), the render time of each frame must be equal to or less than 16ms. If this goal is not achieved, you may find that the UI is not smooth or frames are lost ð.
Timeline Event
In the middle part of the whole interface, we can see the Timeline Event, which actually corresponds to the details of the drawing diagram above, and is the Event in the drawing process. We can use it to see in detail why we caton. Let’s take a closer look at ð§Ū. In the figure below, you can see that the Timeline Event is still divided into two parts. The upper part is UI Event, and the lower part is Raster Event. During this period, we can obviously feel that our UI events take a very long time â°. In order to have a better understanding of the meaning of the whole TimeLine Event, the official explanation is quoted here:
The Flame TAB is used to display selected frame events, sample CPU information. The diagram shows the top-down call stack information, with the top stack frame calling the bottom stack frame. The width of each stack frame represents how long the CPU executes. The longer stack frames consume CPU time, the more likely they are to be a good place for us to make performance improvements.
We can see in the stack information above that the uppermost Event is called VsyncProgressCallback. In the figure below, we can see at the bottom that the stack takes up to 40.6ms, which is almost unacceptable ð―. In fact, when we click on a smooth frame, the VsyncProgressCallback will also be there. So why does this frame get extra long and stall?
So in order for us to further explore why that is, we need to go further, we need more information.
- More detailed flame map ðĨ (
CPU Flame Chart
)
This has all the details you want, including the internal time spent on a particular stack. Why does VsyncProgressCallback take so long? What are the details of internal operation, here is clear ðŪ. Here we can see that a long VsyncProgressCallback has several more parts than a short VsyncProgressCallback. Once we know the internal stack information of the VsyncProgressCallback, we can analyze the internal time stack again. Let’s go to our second Tab.
Recommended reading for the Vsync signal in Flutter: The Flutter Rendering mechanism – UI Threads ð.
- Call tree
This section is closer to home for developers as we can see in detail which functions are called by the VsyncProgressCallback and which are time-consuming. And if you look at the above picture, the extra work that we’re doing in this carton frame, or the function that we’re calling a lot, which is performRebuild, if WE click on the call to this function, we can see that this function is constantly being called in a loop. PerformRebuild is essentially a process that iterates through your draw tree ðē, which means that you have updated a large draw tree in the current frame and redrawn many of its children. So what we need to check is what part of the interface we call redraw multiple components in, or what part of view_mode drives such a redraw process, and whether there is unnecessary redraw. To improve rendering performance ð.
The above section explains some of the reasons why our UI thread blocks, and how to troubleshoot it, but what about the GPU? Also, we took a typical GPU time operation, and in this stack we can see that our GPU method does take a lot of time âģ, also GPURasterizer::Draw Why does a particular frame take a lot of time?
In fact, DevTools’ Raster view does not show this very clearly. But there’s another way to get to the bottom of it. You can just go to the link that you filled in DevTools. If you are running in profile mode, you will find the link ð in your log.
Click on the link, and in the middle of the page, you’ll find the Timeline tool (which I’ve highlighted in red).
After entering this part of timeline, here is the most detailed information of each frame ð. You can select an area with your mouse, and the specific function calls for the selected area will be displayed below. Includes function name, total time, number of calls, time spent per call, and a visualized bar chart ð. At a glance. The function calls related to the GPU part require you to have a good understanding of the source code of the Flutter Engine so that you can solve the problem specifically. The associated call stack is very complex.
Official skia function calls that are time-consuming:
saveLayer
: very time-consuming, each call needs to reallocate a drawing buffer in the CPU, and tellGPU
Toggles drawing targets. Especially on older equipment.clipPath
: Time consuming. Each invocation will affect each subsequent drawing command. When the drawing command is complex, the sum will be made multiple timesclipPath
The intersection operation of theclipPath
Remove the parts that are not included.
You may say that these functions are rarely programmed to use ðĪŊ, but they are widely used in widgets and their properties. The good news is that the Flutter Team has eliminated many unnecessary and time-consuming operations like the one described above ðģ in its previous iterations. Therefore, if you find similar time-consuming operations in your timeline, you can still analyze the source code of some widgets, which is much easier than before.
In the function call scenario above, a short operation does not mean it is not likely to cause a lag. Chances are you found a call to saveLayer, but it didn’t take as long as you thought. In fact, the Skia GPU’s back end will perform a simple pre-processing after receiving saveLayer’s instructions and immediately return, creating the illusion of no time consuming. However, the actual subsequent asynchronous process will also call SkCanvas::Flush, causing a lot of time. The purpose of asynchronous processing here is to package multiple pre-processed messages ðĶ and send them to the GPU for drawing.
conclusion
This is the end of the article on interface drawing performance of Flutter. I hope this article has been helpful to you ðĪŠ. We will continue to cover memory management and other aspects related to Flutter. Reference: DevTools