What is the architecture?
A: Architecture is the unified management of dependencies.
What is dependency? How many kinds are there? Why do we need to manage it.
A dependency is holding an object, or holding a non-null reference.
one-way dependence
As in project development, objects and objects will have the need to hold each other and call each other, and the holding of objects is a kind of dependency. The holding of objects is a kind of dependency. if A wants to complete a logical process, it needs to call a method of B to realize it, then we can say that A has created a dependency on B.
OK, now A holds a reference to B and can call B's methods directly, which constitutes a one-way dependency.
Single dependency is the most basic, the most common kind of dependency relationship, the first contact with the small comrades of the programming project is relatively easy to understand and use one-way dependency. However, a single dependency should also be based on the specific identity level of A and B to determine, just as the old man hit the boy seems to be natural, the son hit the old man is naturally considered "disobedient".
A normal and healthy one-way dependency is a parent object holding a child object.
This is also easier to understand, if the child object holds the parent object relationship, then with the increase in business, the parent object method logic content will be more and more, the child object if you want to complete a processing logic directly call the parent object method to achieve, will naturally result in a top-heavy unhealthy state.
Consider the hierarchy
The determination of whether a single dependency between objects is reasonable depends on the hierarchical relationship between the two objects that create the dependency.
If it is a parent-child relationship, then the principle is that it is more appropriate for the parent object to hold a dependency on the child object;
If you are brothers, try to communicate through a jointly held parent object. Holding dependencies between brothers also increases maintenance costs, so try to "communicate" by holding an object in common.
This means that the best dependencies are those where the higher level depends on the lower level, dependencies between objects of the same level are not recommended, and dependencies of child objects on parent objects should be avoided.
Of course this belongs to is a programming design ideas, not only limited to the implementation of the object and the dependency relationship between the object, module and the module in the middle, the hierarchy and the hierarchy should be followed by the design guidelines of the higher level holds the lower level.
The game project should also be following the surface (top level) to bottom level dependency.
Consider the interaction
The above section has shown three ways in which objects can interact with each other:
- methodologies
- commission
- event
Methods:
The parent object holds the child object and can directly call methods in the child object to complete the interaction.
bidirectional dependence
First of all, let's make it clear: two-way dependency is not recommended.
Because the coupling between mutually held objects is too high, not in line with our consistent pursuit of "low-coupling, high cohesion" criterion, for which we resolutely put an end to bidirectional dependencies, no matter whether it is a parent-child, brother level, any form of bidirectional dependencies should not be present.
What about the need to communicate with each other, given that the words are put in such absolute terms?
This brings us to the other two types of interaction from above - delegates and events.
Imagine an example of the typical need to communicate with each other.
Button clicks, button control script A needs to detect the user's click, after the user clicks to inform the specific execution of the script B. This is A has a communication need for B. If B's specific operation is to play an animation, during the animation to deactivate the button to prevent the user from clicking it again, and when B's animation finishes, B needs to inform A, so that A restores the button state and continues to detect the user's clicks, this is the dependency requirement B has on A. This is B's dependency requirement on A.
If you hold each other's objects, you can just call each other's mutual methods directly, but to decouple, you naturally need to use delegates and events.
Decoupling Unidirectional Dependencies with Delegates
The above sets out the principle that one-way dependence is subject to elders holding children and grandchildren, so what about children and grandchildren who have a need to communicate with their elders but cannot hold them dependent?
Here delegates are used to fulfill the communication needs of the lower level to the upper level.
namespace DependencyDemo
{
public class A : MonoBehaviour
{
B b.
B b; void Start()
{
b = ("Animation").GetComponent<B>();
// Register the completed event
+= ()=> {
("Animation finished playing");; // The registration is complete.
};
}
}
public class B : MonoBehaviour
{
// Define the delegate
public Action OnDoSomethingDone = ()=>{};;
// Called when the animation has finished playing
public void DoSomething()
{
//Trigger the execution of the function in the delegate
OnDoSomethingDone();
}
}
}
Declare a delegate in B, and call the function in the delegate when the animation is finished. A, which holds B, can get the delegate in B and add the operation function to the delegate when the animation finishes playing, thus realizing the communication between the lower level B and the higher level A. B does not hold A.
So what are the disadvantages of using delegates?
The register function and the logout function should appear in pairs in the delegate!
Delegates should be registered to ensure that the function is canceled at the end, that is, registration and cancellation should occur in pairs to avoid the problem of null references. Delegates should be used like a programming language without GC, to ensure that the memory is released after it has been requested and used, and to develop the habit of pairing.
In fact, whether it is delegated or memory use, in programming often have to pay attention to the use of the module, the life cycle of the object, for example, call a function found that the object is not instantiated, load resources found that the resource management class has not been initialized, and so on and so forth,, it is natural for us to clear to know the life cycle of the object to be called.
Delegate another point is that it will increase the amount of code, you can imagine delegate counted as a callback function container, a function execution time to have a corresponding callback, then with the increase in business, the function increases, the corresponding delegate is also correspondingly increased, and delegate the need for the corresponding declaration, not inevitably also increase the workload on the management of delegates.
So far, this section demonstrates the use of delegates to decouple subordinate-to-superior dependencies. However, the use of delegates is not recommended for single dependencies between the same levels.
Since then we know the principle or idea that communication from the bottom to the top can be done using delegates.