I have done SDK development in the company for more than two years. I will share some EXPERIENCE of SDK development with my knowledge and learning.

1. What is SDK

I believe friends who do Android development must have used third-party SDK, such as push SDK, share SDK and so on. The full name of SDK is Software Development Kit, which translates as “Software Development Kit”. An SDK is usually a specific set of software packages, frameworks, etc., written to aid in the development of some kind of software.

SDK can be divided into system SDK and application SDK. The system SDK is a collection of tools developed for using specific software frameworks, hardware platforms, and so on. The application SDK is a set of tools developed based on the system SDK that are independent of specific business and have specific functions.

SDK users are mainly B-end customers, and the final deliverable products are codes, examples and documents. Customers’ access to SDK is also a process of communication with SDK providers. The cost of external communication is higher than that of internal communication, and more problems will be encountered. So SDK development is more demanding than app development. If you can develop a good SDK, you can develop a good app, but if you can develop a good app, you can not develop a good SDK.

2. SDK achieves goals

The goal of SDK is to be simple, stable and efficient.

concise

For users, a good product should be simple and easy to use, and should not take them too long to learn. The SAME should be true of the SDK, which should not involve a lot of tedious docking. Users can read the code and documentation and do it in a very short amount of time.

For example, when a developer needs to use an SDK service, they just need to add a new line to the code. It only takes one line of code to initialize the SDK in a project, the developer doesn’t care about the GLContext, it’s already handled internally, and it doesn’t care about synchronization or asynchronous issues.

public class FURenderer {
    / / define
    public static void setup(Context context) {
        / /...}}// a line of code call
FURenderer.setup(context);
Copy the code

stable

From the perspective of SDK users, we expect the services of the third-party SDK to be stable and efficient, which is reflected in providing stable and reliable services and efficient runtime performance. This requires us to do the following as much as possible when designing and implementing the SDK:

  • Provides a stable API externally. Once the API of the SDK is determined, it cannot be changed except in special circumstances, and it is very costly for the provider to change the API.
  • To provide stable external business. After providing a stable API, there must be a stable business to support it.
  • Runtime stability. Ensure that the SDK itself runs stably and the host application cannot be unstable due to access to the SDK.
  • Version updates steadily. SDK versions iterate very slowly, so try to shield the iterative process from users as much as possible to avoid unnecessary adaptation costs.

efficient

Performance issues should be considered in both general application development and SDK development. SDK designers should consider the following:

  • Less memory footprint. Generally, THE SDK and App run in the same process. In this case, the SDK needs to manage the memory occupied by itself, allocate it reasonably, and release it carefully.
  • Less memory jitter. SDK designers must reduce memory jitter caused by frequent GC in order to use less memory.
  • Less power consumption. The tradeoff between low battery consumption and high performance can be considered in terms of CPU computation, screen refresh frame rate, etc.

3. SDK architecture design

The architectural implementation of the SDK determines the subsequent maintenance difficulty, so it is best to determine the appropriate solution based on the actual business. Using modular development in a project as an example, explain the principles of architectural design.

Following the principles of object-oriented development, the goal is to achieve three goals: maintainability, reusability, and extensibility. Specifically speaking:

  • According to the principle of single responsibility, the system is divided into several small modules, each module is relatively independent, reducing the complexity of implementation classes.
  • According to the principle of interface isolation, a contract interface is defined for each module. The granularity and function of the interface should be small, and the smaller the interface, the easier it is to maintain.
  • Modules communicate with each other through protocols or interfaces, avoiding direct interdependence to reduce coupling and minimize mutual understanding, which embodies Demeter’s law.
  • According to the open and close principle, the common behavior of each module is defined, and the skeleton implementation is provided through the template design mode, which is easy to expand functions.
  • According to the principle of composition over inheritance, the combination of classes is used to ensure the flexibility of design when the functions of multiple modules are superimposed.

For example, the functional modules of the third-party Demo of the project refer to the framework of Java collection framework, which is divided into three parts: contract interface, abstract class and concrete implementation.

  • The IEffectModule is defined as the contract interface of the effect, including common operations for creating, setting parameters, destroying and so on.
  • Abstract AbstractEffectModule, as a framework for IEffectModule, implements common methods and defines common member variables.
  • Define the IFaceBeautyModule interface, which inherits the IEffectModule interface, including the additional operation of setting parameters. FaceBeautyModule as its implementation class, and inherit the AbstractEffectModule, reuse the base class code.
  • Similar to the beauty makeup and body module, the contract interface is defined first, and then the concrete implementation is defined. The interfaces are isolated from each other and highly cohesive within the interfaces.
  • FURenderer implements the IFURenderer rendering interface and the IModuleManager module management interface, combining various functional modules.

4. SDK design specification

API design is very important in any development, and many times the quality of software is reflected in API design. In normal application development, apis only circulate among developers and are not exposed to other people who are not developing the application. However, as a service, SDK needs to expose part of the API to developers in order to use SDK services.

Here are some principles to focus on.

  1. The method name indicates its purpose

A good method name is the most intuitive indication of what it does. The name is self-explanatory and does not require additional documentation, which reduces unnecessary communication costs. What could be more intuitive for developers than reading code directly? According to Refactoring, naming each variable as if it were your child is not too much to ask.

  1. Validation of parameters

If the program runs abnormally, it will ruin the user experience, which is very bad. We use the idea of “defensive programming” to avoid damaging the system with illegal input.

When the validity check fails, different processing policies are used for different method permissions:

  • Explicitly check for the public method to throw an exception, and use@throwTo explain why the exception was thrown
  • For private methods, the validation of parameters is checked by assertions.
  • Check the validity of constructor arguments to keep objects in a uniform state.

It is important to note that if the cost of inspection is too high, it needs to be considered comprehensively.

  1. Methods implement a single function

A method should have a single function and do as few, specialized things as possible, which is also the embodiment of the single responsibility principle. The “Alibaba code code” states that a method should have no more than 80 lines, and large methods should be broken down into smaller ones.

Also note that it is better to offer small but beautiful approaches than big and complete approaches, which are often subject to change and have a higher risk. So instead of providing smaller methods that can be combined, small and beautiful methods make it easier to reuse code.

  1. Access control

Including class method permissions and variable permissions, can declare private do not public, external knowledge as little as the better. Static methods are naturally thread-safe. Protect methods and variables that are exposed by inheritance.

  1. Avoid too long parameters

Too long parameters will cause memory difficulties, and call parameter passing error prone, should be avoided. If too long parameters cannot be avoided, consider other solutions:

  • Through the use of Builder mode to achieve
  • By encapsulating multiple parameters as class objects

For example, there is a method in the project that has a lot of parameters.

int onDrawFrameSingleInput(byte[] img, int w, int h, int format, byte[] readBackImg, int readBackW, int readBackH);
Copy the code

After reconstruction, parameters are encapsulated as objects, and only one object is passed in to call methods to avoid the bad experience caused by a large number of parameters.

public class VideoFrame {
    private int width;
    private int height;
    private byte[] data;
    private byte[] readback;
    private int readbackWidth;
    private int readbackHeight;
    private int pixelFormat;
    // ...
}

int onDrawFrameSingleInput(VideoFrame videoFrame);
Copy the code
  1. Be careful with method overloading

Overloading often confuses developers, and when a method needs to be overridden, different method names can be used instead. For constructors, you can use static factories instead of overloading.

The ObjectOutputStream class provided in Java is a good example: its write has a variant for each primitive type, such as write characters, write Boolean, and so on. Instead of using overloading as Write (Long L), write(Boolean b), the designers designed it as writeLong(L), writeBoolean(b).

For example, the project external processing methods are all overloaded, can only be distinguished according to parameters, very confusing. Change to a different method name, see the name to know the method to call, a lot of clarity.

/ / before the refactoring
int onDrawFrame(byte[] img, int tex, int w, int h);
int onDrawFrame(byte[] img, int w, int h);/ / after the refactoring
int onDrawFrameDualInput(byte[] img, int tex, int w, int h);
int onDrawFrameSingleInput(byte[] img, int w, int h, int format);
Copy the code
  1. Avoid methods that return NULL directly

Do not return NULL for methods that need to return an array or collection. For example, if we go to a pastry shop to buy bread and run out of bread is a normal state, we should not return null. Instead, we should return an array or collection of length zero. Java provides collections.emptyxxx () for empty Collections.

  1. Avoid introducing third-party libraries

GitHub has many open source third-party libraries, such as OkHttp, Image Loading Glide, etc., but the basic principles followed in SDK development are:

  • The principle of minimum availability is to use the least amount of code and not add entities unnecessarily.
  • The principle of least dependence, that is, the minimum external dependence, if not necessary do not increase dependence.

The introduction of third-party libraries may bring the following problems:

  • The third-party libraries of the host application and the SDK depend on different versions, which may cause conflicts and increase the interconnection cost.
  • Open source libraries are constantly updated, and SDKS need to be updated in a timely manner, adding additional maintenance work.
  • Due to the introduction of open source libraries, problems are difficult to troubleshoot.
  1. Ensure compatibility

The SDK is iterative, with new features and bug fixes coming with each release. For the user, the upgrade version should not change too much. It is usually enough to simply replace the library file or change the version number of the remote dependent library. Avoid direct renaming of public interfaces. If the old interface is Deprecated, indicate it with the @deprecated keyword and give an alternative and a Deprecated date.

  1. Less invasive

To ensure that there is less code intrusion mainly when providing services externally, design a good API with full consideration of the developer’s usage scenarios. A good API needs to be defined the way most developers expect it to be – semantically easy to understand, and simple and reliable to use. The specific performance is that it can run stably and reliably under normal circumstances and feedback error information in time under abnormal circumstances.

For example, when using Gradle to download dependent libraries, the AAR package contains unnecessary bundle resources. We provide the build configuration for packaging APK, and freely choose the bundles to be packaged, reducing the intrusion to the host application.

    applicationVariants.all { variant ->
        variant.mergeAssetsProvider.configure {
            doLast {
                delete(fileTree(dir: outputDir,
                        // Delete unnecessary bundle files
                        includes: ['model/ai_face_processor_lite.bundle'.'model/ai_gesture.bundle'.'graphics/controller.bundle'.'graphics/tongue.bundle'))}}}Copy the code

5. The SDK for delivery

Packaging bag

The Android platform typically publishes SDKS using a JAR and an AAR, the difference being that the JAR contains only code, and an AAR can contain code, resources, and dynamic libraries. In general, an AAR is the most appropriate delivery method, and by uploading it to the Maven server, consumers can integrate it in one line of code. For customers who need flexible customization, we will also provide SDK source code, the drawback is difficult to upgrade, to change a lot of code.

For code obfuscation, don’t mix the public and native interfaces. Internal implementation details can be confused to reduce the size of SDK packages.

Access to the document

Access documentation is used to tell SDK users how to use the SDK, detailed steps to use it, and possible problems. The document contains update records, basic information, API description, integration procedure, and FAQs. The standard of good documentation is clarity and easy to understand. A developer who doesn’t know anything about the SDK can look at the documentation and make a list of the most frequently encountered problems, and the terminology should be explained accordingly.

The Demo sample

An integration Demo is usually a simple App that shows how to quickly access the SDK. The source code of Demo is hosted on GitHub for users’ reference. The version change policy is consistent with the SDK version change. Even though it is a Demo, its development principles should be consistent with the SDK to ensure high quality delivery.

Reference article:

  • Random Thoughts: Developing a first-class Android SDK
  • Compile Android SDK development specifications
  • Android SDK development experience