Introduction to AOPThe main function of AOP (Aspect Oriented Programming) is to extract some functions that are not related to the core business logic module. These functions that are not related to business logic usually include log statistics, security control, exception handling, etc. After extracting these functions, they are integrated into the business logic module through "dynamic weaving". Aspect-oriented programming provides us with a way to inject code into existing functions or objects without modifying the target logic. While not required, injected code is meant to have cross-cutting concerns, such as adding logging functionality, debugging metadata, or other less general but additional behavior can be injected without affecting the content of the original code. To give you a good example, suppose you have written your business logic, but now you realize that you didn't add logging code. The usual approach is to centralize the logging logic into a new module and then add logging information function by function. However, it would certainly be a lot easier if you could get the same logger to be injected with just one line of code at a specific point in the execution of each method you want to log. Isn’t it? Aspects, advice, and pointcuts (what, when, where)To make the above definition more formal, let's take the logger as an example to introduce three concepts about AOP. If you decide to explore this paradigm further, these will help you:
With this explanation, you should find it relatively easy to create an AOP-based library to add logging logic to existing OOP-based business logic (for example). All you have to do is replace the existing matching method of the target object with a custom function that adds the aspect logic at the appropriate time and then calls the original method. Basic Implementation Since I'm a visual learner, I thought it would be a long shot to show a basic example of how to implement an The following examples will illustrate how easy it is to implement and the benefits it brings to your code. `/** Helper function for getting all methods in an object*/ const getMethods = (obj) => Object.getOwnPropertyNames(Object.getPrototypeOf(obj)).filter(item => typeof obj[item] === 'function') /** Replace the original method with a custom function that will call our aspect when the advice instructs */ function replaceMethod(target, methodName, aspect, advice) { const originalCode = target[methodName] target[methodName] = (...args) => { if(["before", "around"].includes(advice)) { aspect.apply(target, args) } const returnedValue = originalCode.apply(target, args) if(["after", "around"].includes(advice)) { aspect.apply(target, args) } if("afterReturning" == advice) { return aspect.apply(target, [returnedValue]) } else { return returnedValue } } } module.exports = { // Main method of export: inject aspects into the target when and where needed inject: function(target, aspect, advice, pointcut, method = null) { if(pointcut == "method") { if(method != null) { replaceMethod(target, method, aspect, advice) } else { throw new Error("Tryin to add an aspect to a method, but no method specified") } } if(pointcut == "methods") { const methods = getMethods(target) methods.forEach( m => { replaceMethod(target, m, aspect, advice) }) } } }` Pretty straightforward. As I mentioned, the code above doesn't cover all use cases, but it should be enough to cover the next example. But before we go any further, take a look at the Next, I will explain how to use this library: `const AOP = require("./aop.js") class MyBussinessLogic { add(a, b) { console.log("Calling add") return a + b } concat(a, b) { console.log("Calling concat") return a + b } power(a, b) { console.log("Calling power") return a ** b } } const o = new MyBussinessLogic() function loggingAspect(...args) { console.log("== Calling the logger function ==") console.log("Arguments received: " + args) } function printTypeOfReturnedValueAspect(value) { console.log("Returned type: " + typeof value) } AOP.inject(o, loggingAspect, "before", "methods") AOP.inject(o, printTypeOfReturnedValueAspect, "afterReturning", "methods") o.add(2,2) o.concat("hello", "goodbye") o.power(2, 3)` This is just a basic object with three methods, nothing special. We want to inject two generic aspects, one for logging incoming properties, and another for analyzing their return values and logging their types. Two aspects, two lines of code (it doesn’t need to be six). This is the end of the example, and here is the output you will get:
Advantages of AOPAfter knowing the concept and purpose of AOP, you may have guessed why people would want to use aspect-oriented programming, but let's make a quick summary:
Main Problems with AOPBecause not everything is perfect, this paradigm has some critics against it. The main issue they raise is that its main advantage is actually hiding code logic and complexity, which can have side effects when it is not clear. If you think about it, they have a point, AOP gives you a lot of power to add unrelated behavior to existing methods or even replace their entire logic. Of course, this may not be the exact reason why this paradigm was introduced, and it certainly was not the intention of the examples I provided above. However, it does allow you to do whatever you want, which combined with a lack of understanding of good programming practices can lead to a very big mess. In order not to sound too cliché, I will paraphrase Uncle Ben: With great power comes great responsibility If you want to use AOP correctly, you must understand the best practices of software development. In my opinion, just because you can cause a lot of harm using this tool doesn't mean it's a bad tool, because it also brings a lot of benefits (i.e. you can extract a lot of common logic into a centralized location and inject it with a single line of code wherever you need it). To me, this is a powerful tool that is worth learning and definitely worth using. Aspect-Oriented Programming is a perfect complement to OOP, especially thanks to the dynamic nature of JavaScript, we can implement it very easily (as demonstrated in the code here). It provides powerful capabilities to modularize and decouple large amounts of logic, and even share that logic with other projects later. Of course, if you don't use it correctly, you can make a real mess. But you can definitely use it to simplify and clean up a lot of code. That's my opinion on AOP, what about you? Have you ever heard of AOP? Have you used it before? Please leave a comment below and share your thoughts! This is the end of this article about AOP programming in JavaScript. For more relevant js AOP programming content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: Python virtual environment installation and uninstallation methods and problems encountered
>>: Detailed explanation of 4 common data sources in Spark SQL
In the process of using Vue to develop projects, ...
SVN service backup steps 1. Prepare the source se...
Table of contents 1. Description of functions in ...
Simple function: Click the plug-in icon in the up...
When OP opens a web page with the current firmwar...
In fact, this is also a clickbait title, and it c...
1. Vue--The first vue-cli program The development...
Original source: www.bamagazine.com There are nar...
1. Each function is an object and occupies memory...
1. Install tools and libraries # PCRE is a Perl l...
Need to export the fields and properties of the t...
Detailed example of removing duplicate data in My...
Because I have a database tutorial based on SQL S...
For commercial databases, database upgrade is a h...
js date time format Convert the date and time to ...