Web performance optimization is the primary issue that every web application developer pays attention to. Task runners such as Grunt play a key role in the development process. They automate tasks such as code stitching and compression, which is also the focus of this tutorial. We will use a range of Grunt plugins to ensure AngularJS applications can be compressed safely. Before discussing AngularJS and compression, I want to emphasize that developers of all skill levels can benefit from this tutorial, but it is better to have the basics of Grunt. In this article, we will use Grunt to generate new folders, so Grunt beginners can also have a good understanding of how it works.
Key Points
- Task runners like Grunt automate code stitching and compression, optimizing page speed during development. The Grunt plug-in ensures that AngularJS applications can be compressed safely.
- By default, AngularJS applications are not compressible and must be written in array syntax. When UglifyJS is running, it renames the parameters, but the existence of DI annotations in the array prevents them from being renamed, ensuring that the renamed parameters still have access to the necessary dependencies.
- Grunt can be used to automate the annotation, splicing, and compression processes of AngularJS applications. After installing the necessary plug-ins and configuring Grunt to read the "package.json" file, the task will be loaded and registered. Then configure the plugin to locate specific files.
- Grunt helps write safer Angular code by automating tasks that capture and prevent errors. It can run unit tests on the code every time the file is saved, sending an error alert to the developer immediately. Automation saves time and ensures that important tasks are not ignored.
Angular application compression problem
ArticleJS applications are not compressible safe by default. They must be written in array syntax. If you're not sure what the array syntax is, you've probably written code to use it. Let's look at two examples of AngularJS controllers that are passing the $scope
and $http
parameters. In the first example below, the module's factory and controller are wrapped in an array starting with a DI annotation, which, as you can see, does not follow the DRY (Don't repeat yourself) principle.
var form = angular.module('ControllerOne', []) form.factory('Users', ['$http', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }]); form.controller('InputController', ['$scope', '$http', 'Users', function($scope, $http, Users) { formData = {}; $scope.createUser = function () { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function (data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; }]);
In the next example, the crud.config
module code is still not compressible safe, but the code is shorter than previous ones. It just names the services and then passes the necessary dependencies into the function as parameters without writing them out as strings first. As long as there is no compression, this code will run normally. Therefore, it is easy to understand why people often choose this syntax when writing AngularJS code.
var form = angular.module('ControllerTwo', []) form.factory('Users', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }); form.controller('InputController', function($scope, $http, Users) { formData = {}; $scope.createUser = function() { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function(data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; });
Now that you have understood the physical differences between these two pieces of code, I will quickly explain why this syntax is not suitable for compression.
How array notation works
As mentioned above, array notation begins with DI annotations, which plays a key role in making code compression safe. When UglifyJS is run, it renames our parameters from $scope
and $http
to a
and b
respectively. The existence of DI annotations passed as strings into an array prevents them from being renamed. Therefore, these renamed parameters still have access to the necessary dependencies. If these annotations do not exist, the code will break. As you can see, it is very inefficient to write code manually in this way. To help you avoid this, I will now show you how to annotate, splice, and compress AngularJS applications in a fully optimized way using Grunt and prepare them for production.
Use Grunt
The entire repository of the project can be found on GitHub, including the files we will locate. For those who are used to using Grunt, you can continue and create your own build, or add this code to an existing project. If you are using an empty directory, you must make sure there is a "package.json" file in the directory. This file can be created by running the command npm init
. Once you have the "package.json" file in your project, you can download the plugin by running the following command:
var form = angular.module('ControllerOne', []) form.factory('Users', ['$http', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }]); form.controller('InputController', ['$scope', '$http', 'Users', function($scope, $http, Users) { formData = {}; $scope.createUser = function () { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function (data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; }]);
This will install Grunt into your project, and the three plugins we will use:
grunt-contrib-concat
grunt-contrib-uglify
grunt-ng-annotate
While ng-annotate can be used without Grunt, you will soon see how seamless Grunt makes the process of annotating, splicing, and compressing your code. It provides a simple and effective solution for compressing AngularJS code. If you are tracking this project from scratch, you should have a Gruntfile.js in the project root directory, which will contain all the Grunt code. If you haven't already, create it now.
Three steps to generate compression-safe code
Step 1 – Configure Grunt to read the "package.json" file
To access the plugin we installed earlier, you first need to configure the pkg
property of the Gruntfile to read the contents of the "package.json" file. The config
object starts at the top of the Grunt wrapper function and extends from line 3 to line 5 in the following example, but will soon include most of the code.
var form = angular.module('ControllerTwo', []) form.factory('Users', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }); form.controller('InputController', function($scope, $http, Users) { formData = {}; $scope.createUser = function() { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function(data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; });
Step 2 – Loading and Registering Grunt Tasks
After configuring Grunt to read our "package.json" files, plugins need to be loaded so that Grunt can access them. This is done by passing the plugin's name as a string into grunt.loadNpmTask()
. Be sure to make sure that these plugins are loaded inside the wrapper function but outside the object. If these conditions are not met, Grunt will not work properly. What we need to do next is create a default task that will be performed when Grunt is called without a specific target. You should be careful about the order in which these tasks are added, as they will run according to their configuration. Here, ngAnnotate is configured to run first, then concat and UglifyJS, which I believe is the best way to build your code. Also, it is important to remember that config
must be placed after the plugin is loaded. Based on what we just discussed, Gruntfile.js should look like this:
grunt.registerTask()
var form = angular.module('ControllerOne', []) form.factory('Users', ['$http', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }]); form.controller('InputController', ['$scope', '$http', 'Users', function($scope, $http, Users) { formData = {}; $scope.createUser = function () { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function (data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; }]);
ngAnnotate
Now that our Gruntfile is ready, let's go back to theobject and specify the file we want the ngAnnotate plugin to locate. To do this, we first have to create a part for ngAnnotate and create a target, called
in this case. In this target, you specify the file to which you want to add DI annotations, and the folder to which it should be generated. In this example, Grunt takes the three files specified in and generates them into a new folder named config
. Once the configuration is complete, you can run spApp
and see how the code is generated. Additionally, you can visit the GitHub page of grunt-ng-annotate and see the different options it allows you to specify. public/js
public/min-safe
grunt ngAnnotate
Split
var form = angular.module('ControllerTwo', []) form.factory('Users', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }); form.controller('InputController', function($scope, $http, Users) { formData = {}; $scope.createUser = function() { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function(data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; });Now that you have generated a folder with the AngularJS code with the new annotations, let's continue by compiling or splicing this code into a single file. Same way we created the section for ngAnnotate, we will now do the same for concat and UglifyJS. Like ngAnnotate, both tasks accept a target, in this case
. Many configuration options can be passed into these tasks, but we simply specify
and to point to the correct file. As you might guess, these plugins will get the file contents passed to the js
object and process them into the folder specified after src
. Let's try to understand what's going on here. You can test this by running dest
in your terminal, it should result in the creation of src
. dest
var form = angular.module('ControllerOne', []) form.factory('Users', ['$http', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }]); form.controller('InputController', ['$scope', '$http', 'Users', function($scope, $http, Users) { formData = {}; $scope.createUser = function () { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function (data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; }]);
Compression
The last thing we need to do is remove useless space from the code by compressing it. This is where the UglifyJS plugin comes into play. When using UglifyJS, we want Grunt to complete the final process of compressing the application. Therefore, we want to locate the file containing all the new splicing codes, in this case public/min/app.js
. To test this, run grunt uglify
and view your newly compressed file. The following are the relevant configurations for this task:
var form = angular.module('ControllerTwo', []) form.factory('Users', function($http) { return { get: function() { return $http.get('/api/users'); }, create: function(userData) { return $http.post('/api/users', userData); }, delete: function(id) { return $http.delete('/api/users/' + id); } }; }); form.controller('InputController', function($scope, $http, Users) { formData = {}; $scope.createUser = function() { if ($scope.formData != undefined) { Users.create($scope.formData) .success(function(data) { $scope.users = data; $scope.formData = {}; $scope.myForm.$setPristine(true); }); } }; });
In this course, we used all of these tasks separately. Now, let's use the default task we created earlier. It allows Grunt to run all specified tasks one by one in registration order. Now, just run grunt
in your project and your code will be annotated, spliced, and compressed.
Conclusion
I hope that with this short tutorial you will be able to understand array notation well and why it is essential to make AngularJS application compression safe. If you are new to Grunt, I highly recommend you try these plugins as well as others as they can save a lot of time. As always, feel free to comment below or if you have any questions please email me at the address in my profile.
FAQ (FAQ) on Compression-Safe Angular Code with Grunt
(The same FAQ part should be included here as in the original text, but the language is smoother and more natural)
The above is the detailed content of 5 Minutes to Min-Safe Angular Code with Grunt. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undress AI Tool
Undress images for free

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

JavaScript's garbage collection mechanism automatically manages memory through a tag-clearing algorithm to reduce the risk of memory leakage. The engine traverses and marks the active object from the root object, and unmarked is treated as garbage and cleared. For example, when the object is no longer referenced (such as setting the variable to null), it will be released in the next round of recycling. Common causes of memory leaks include: ① Uncleared timers or event listeners; ② References to external variables in closures; ③ Global variables continue to hold a large amount of data. The V8 engine optimizes recycling efficiency through strategies such as generational recycling, incremental marking, parallel/concurrent recycling, and reduces the main thread blocking time. During development, unnecessary global references should be avoided and object associations should be promptly decorated to improve performance and stability.

There are three common ways to initiate HTTP requests in Node.js: use built-in modules, axios, and node-fetch. 1. Use the built-in http/https module without dependencies, which is suitable for basic scenarios, but requires manual processing of data stitching and error monitoring, such as using https.get() to obtain data or send POST requests through .write(); 2.axios is a third-party library based on Promise. It has concise syntax and powerful functions, supports async/await, automatic JSON conversion, interceptor, etc. It is recommended to simplify asynchronous request operations; 3.node-fetch provides a style similar to browser fetch, based on Promise and simple syntax

JavaScript data types are divided into primitive types and reference types. Primitive types include string, number, boolean, null, undefined, and symbol. The values are immutable and copies are copied when assigning values, so they do not affect each other; reference types such as objects, arrays and functions store memory addresses, and variables pointing to the same object will affect each other. Typeof and instanceof can be used to determine types, but pay attention to the historical issues of typeofnull. Understanding these two types of differences can help write more stable and reliable code.

Hello, JavaScript developers! Welcome to this week's JavaScript news! This week we will focus on: Oracle's trademark dispute with Deno, new JavaScript time objects are supported by browsers, Google Chrome updates, and some powerful developer tools. Let's get started! Oracle's trademark dispute with Deno Oracle's attempt to register a "JavaScript" trademark has caused controversy. Ryan Dahl, the creator of Node.js and Deno, has filed a petition to cancel the trademark, and he believes that JavaScript is an open standard and should not be used by Oracle

Which JavaScript framework is the best choice? The answer is to choose the most suitable one according to your needs. 1.React is flexible and free, suitable for medium and large projects that require high customization and team architecture capabilities; 2. Angular provides complete solutions, suitable for enterprise-level applications and long-term maintenance; 3. Vue is easy to use, suitable for small and medium-sized projects or rapid development. In addition, whether there is an existing technology stack, team size, project life cycle and whether SSR is needed are also important factors in choosing a framework. In short, there is no absolutely the best framework, the best choice is the one that suits your needs.

IIFE (ImmediatelyInvokedFunctionExpression) is a function expression executed immediately after definition, used to isolate variables and avoid contaminating global scope. It is called by wrapping the function in parentheses to make it an expression and a pair of brackets immediately followed by it, such as (function(){/code/})();. Its core uses include: 1. Avoid variable conflicts and prevent duplication of naming between multiple scripts; 2. Create a private scope to make the internal variables invisible; 3. Modular code to facilitate initialization without exposing too many variables. Common writing methods include versions passed with parameters and versions of ES6 arrow function, but note that expressions and ties must be used.

CacheAPI is a tool provided by the browser to cache network requests, which is often used in conjunction with ServiceWorker to improve website performance and offline experience. 1. It allows developers to manually store resources such as scripts, style sheets, pictures, etc.; 2. It can match cache responses according to requests; 3. It supports deleting specific caches or clearing the entire cache; 4. It can implement cache priority or network priority strategies through ServiceWorker listening to fetch events; 5. It is often used for offline support, speed up repeated access speed, preloading key resources and background update content; 6. When using it, you need to pay attention to cache version control, storage restrictions and the difference from HTTP caching mechanism.

Promise is the core mechanism for handling asynchronous operations in JavaScript. Understanding chain calls, error handling and combiners is the key to mastering their applications. 1. The chain call returns a new Promise through .then() to realize asynchronous process concatenation. Each .then() receives the previous result and can return a value or a Promise; 2. Error handling should use .catch() to catch exceptions to avoid silent failures, and can return the default value in catch to continue the process; 3. Combinators such as Promise.all() (successfully successful only after all success), Promise.race() (the first completion is returned) and Promise.allSettled() (waiting for all completions)
