When the functions of the App become larger and more and more developers, various problems will occur, such as:
- Large amount of code, slow compilation
- Function interweave, difficult to pull out, the problem is more difficult to insert
- An error that made the whole project difficult to run
This is when a reasonable organizational structure becomes particularly important. The usual approach is to modularize the functionality within your App. After using modularity, rationalization of modularity can solve the above problems:
- Function is divided into different modules, different personnel to maintain different modules, clear responsibility
- Infrequently changed code is modularized so that it does not need to be compiled as often, increasing compilation speed
- Modules are independent of each other, reducing dependencies
Componentized scheme
There are three componentization schemes that I have learned so far:
Scheme 1 (Interface oriented)
Through the interface – oriented approach, each module through the interface to achieve the purpose of splitting and calling.
However, this scheme has obvious disadvantages: the caller needs to implement the modules that are used. Formally, the code is divided into modules, but there is still a strong dependency between each module, which is still not independent in nature.
Scheme 2 (register URL) :
-
At App startup, component modules are instantiated, and those components register urls with the ModuleManager. For those that do not need to be instantiated, class registration is used.
-
When component A needs to call component B, the URL is passed to the ModuleManager and the parameters follow the URL in GET mode, similar to openURL. More complex parameters need to be passed through A specialized dictionary. The ModuleManager is then responsible for scheduling component B and finally completing the task.
This scheme has the following disadvantages:
-
When registering urls with the ModuleManager, memory is resident, especially if instantiated classes are registered
-
In local calls, the USE of urls is unnecessary. If the business engineer gives the URL during local scheduling, it is inevitable that the dictionary class parameters will need to be supplied. The parameter of dictionary class is not checked by the compiler, which makes the caller more difficult and error-prone.
-
URL registration is not necessary for the implementation of componentized solutions, and componentized solutions formed through URL registration will reduce scalability and maintainability. To be specific:
- The URL registration process can be completely omitted by the Runtime. When a URL is invoked, the runtime instantiates the corresponding class.
- We need to constantly change the registry as services are added or removed. Using the Runtime instead of the registration process means that there is no need to maintain such a registration list.
Frameworks that use this approach include:
- URLNavigator 2.4K Star elegant URL routing framework
Scheme 3 (by code protocol) :
- Each module provides classes and methods with a convention name. Different class and method names provide different services.
- When a certain service is requested through the middleware, the middleware finds the corresponding class and method through the Runtime and invokes it.
- Extend middleware to provide methods and parameters for various services. That is, class names, method names, and parameter names are encapsulated and provided to the outside world through methods. So a module consists of two parts, one is a code protocol compliant classes and methods to provide specific services, and the other is an extension to middleware to facilitate the caller.
Frameworks that use this approach include:
- CTMediator 3.2K Star is a middleware that splits iOS projects into multiple projects without a registration process. It uses target-Action mode to enable sub-projects to communicate with each other without prior registration.
The realization of the CTMediator
Why does CTMediator avoid the problem in Scheme 1? Let’s break down the implementation logic of CTMediator.
In CTMediator, the modules call each other through target-action, which is the class name and method name that provide the service. The coupling between modules is solved by strings of names.
The name of the string to the specific class and method, is through the Runtime features, from the string to instantiate the class, from the string to get the method of the class, so as to call. (This avoids pre-use registration and registration issues.)
When using a service, if you need to use a string of service names, the code becomes ugly and difficult to maintain. At this time, the middleware can add classification or extension, the service corresponding class name, method name, parameter name encapsulation, external to provide a convenient method.
The resources
IOS application architecture: Componentization solutions