Android 应用架构指南

应用架构指南

* 移动应用用户体验

* 常见的架构原则
  * 分离关注点
  * 通过模型驱动界面

* 推荐应用架构
  * 概述
  * 构建界面
  * 获取数据
  * 连接 ViewModel 与存储区
  * 缓存数据
  * 保留数据
  * 显示正在执行的操作
  * 测试每个组件

* 最佳做法

* 附录:公开网络状态

一、移动应用用户体验

在大多数情况下,桌面应用将桌面或程序启动器当做单个入口点,然后作为单个整体流程运行。Android 应用则不然,它们的结构要复杂得多。典型的 Android 应用包含多个应用组件,包括 Activity、Fragment、Service、内容提供程序和广播接收器。

你需要在应用清单中声明其中的大多数应用组件。Android 操作系统随后使用此文件决定如何将你的应用集成到设备的整体用户体验中。鉴于正确编写的 Android 应用包含多个组件,并且用户经常会在短时间内与多个应用进行交互,因此应用需要适应不同类型的用户驱动型工作流和任务。

例如,思考一下当你在自己喜欢的社交网络应用中分享照片时会发生什么:

  1. 该应用将触发相机 intent。Android 操作系统随后启动相机应用来处理请求。

此时,用户已离开社交网络应用,但他们的体验仍然是无缝的。

  1. 相机应用可能会触发其他 intent,如启动文件选择器,而这可能会再启动一个应用。

  2. 最后,用户返回社交网络应用并分享照片。

在此过程中,用户随时可能会被电话或通知打断。对此中断采取行动后,用户希望能够返回并恢复此照片分享过程。这种应用跳跃行为在移动设备上很常见,因此你的应用必须正确处理这些流程。

请注意,移动设备的资源也很有限,因此操作系统可能随时终止某些应用进程以便为新的进程腾出空间。

鉴于这种环境条件,你的应用组件可以不按顺序地单独启动,并且操作系统或用户可以随时销毁它们。由于这些事件不受你的控制,因此你不应在应用组件中存储任何应用数据或状态,并且应用组件不应相互依赖。

二、常见的架构原则

如果你不应使用应用组件存储应用数据和状态,那么应如何设计应用?

2.1 分离关注点

要遵循的最重要的原则是分离关注点。一种常见的错误是在一个 ActivityFragment 中编写所有代码。这些基于界面的类应仅包含处理界面和操作系统交互的逻辑。应尽可能使这些类保持精简,这样可以避免许多与生命周期相关的问题。

请注意,你并不拥有 ActivityFragment 的实现,这些只是表示 Android 操作系统与应用之间关系的粘合类。操作系统可能会根据用户交互或因内存不足等系统条件随时销毁它们。为了提供令人满意的用户体验和更易于管理的应用维护体验,最好尽量减少对它们的依赖。

2.2 通过模型驱动界面

另一个重要原则是你应该通过模型驱动界面,最好是持久性模型。模型是负责处理应用数据的组件。它们独立于应用中的 View 对象和应用组件,因此不受应用的生命周期以及相关关注点的影响。

持久性是理想之选,原因如下:

  • 如果 Android 操作系统销毁应用以释放资源,用户不会丢失数据。

  • 当网络连接不稳定或不可用时,应用会继续工作。

应用所基于的模型类应明确定义数据管理职责,这样将使应用更可测试且更一致。

三、推荐应用架构

在本部分,我们将通过一个端到端用例演示如何使用架构组件构造应用。

注意:不可能有一种应用编写方式对每种情况都最为合适。话虽如此,但推荐的这个架构是适合大多数情况和工作流的一个很好的起点。如果你已经有编写 Android 应用的好方法(遵循常见的架构原则),则无需更改。

假设我们要构建一个用于显示用户个人资料的界面。我们将使用私有后端和 REST API 获取给定个人资料的数据。

3.1 概述

首先,请查看下图,该图显示了设计应用后所有模块应如何相互交互:

请注意,每个组件仅依赖于其下一级的组件。例如,Activity 和 Fragment 仅依赖于视图模型。存储区是唯一一个依赖于其他多个类的类;在本例中,存储区依赖于持久性数据模型和远程后端数据源。

这种设计打造了一致且愉快的用户体验。无论用户是在上次关闭应用几分钟后还是几天后回到应用,他们都会立即看到应用在本地保留的用户信息。如果此数据已过时,则应用的存储区模块将开始在后台更新数据。

3.2 构建界面

界面由 Fragment UserProfileFragment 及其对应的布局文件 user_profile_layout.xml 组成。

要驱动该界面,数据模型需要存储以下数据元素:

  • 用户 ID:用户的标识符。最好使用 Fragment 参数将该信息传递到相应 Fragment 中。如果 Android 操作系统销毁我们的进程,该信息将保留,以便下次重启应用时 ID 可用。

  • 用户对象:用于存储用户详细信息的数据类。

我们将使用 UserProfileViewModel(基于 ViewModel 架构组件)存储这些信息。

ViewModel 对象为特定的界面组件(如 Fragment 或 Activity)提供数据,并包含数据处理业务逻辑,以与模型进行通信。例如,ViewModel 可以调用其他组件来加载数据,还可以转发用户请求来修改数据。ViewModel 不了解界面组件,因此不受配置更改(如在旋转设备时重新创建 Activity)的影响。

我们现在定义了以下文件:

  • user_profile.xml:屏幕的界面布局定义。

  • UserProfileFragment:显示数据的界面控制器。

  • UserProfileViewModel:准备数据以便在 UserProfileFragment 中查看并对用户交互做出响应的类。

以下代码段显示了这些文件的起始内容。(为简单起见,省略了布局文件。)

UserProfileViewModel

public class UserProfileViewModel extends ViewModel {
    private String userId;
    private User user;

    public void init(String userId) {
        this.userId = userId;
    }
    public User getUser() {
        return user;
    }
}

UserProfileFragment

public class UserProfileFragment extends Fragment {
    private static final String UID_KEY = "uid";
    private UserProfileViewModel viewModel;

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        String userId = getArguments().getString(UID_KEY);
        viewModel = ViewModelProviders.of(this).get(UserProfileViewModel.class);
        viewModel.init(userId);
    }

    @Override
    public View onCreateView(LayoutInflater inflater,
                @Nullable ViewGroup container,
                @Nullable Bundle savedInstanceState) {
        return inflater.inflate(R.layout.user_profile, container, false);
    }
}

现在,我们有了这些代码模块,如何将它们连接起来?毕竟,在 UserProfileViewModel 类中设置 user 字段后,我们需要一种通知界面的方法。这就是 LiveData 架构组件的用处所在。

LiveData 是一种可观察的数据存储器。应用中的其他组件可以使用此存储器来监控对象的更改,而无需在它们之间创建明确且严格的依赖路径。LiveData 组件还遵循应用组件(如 Activity、Fragment 和 Service)的生命周期状态,并包括清理逻辑以防止对象泄漏和过多的内存消耗。

注意:如果你已在使用诸如 RxJava 或 Agera 之类的库,你可以继续使用这些库而不是 LiveData。但在使用诸如此类的库和方法时,请确保正确处理应用的生命周期。特别是,在相关的 LifecycleOwner 停止时暂停数据流,并在相关的 LifecycleOwner 销毁时销毁这些数据流。你还可以添加 android.arch.lifecycle:reactivestreams 软件工件,以将 LiveData 与其他响应流库(如 RxJava2)一起使用。

为了将 LiveData 组件纳入应用,我们将 UserProfileViewModel 中的字段类型更改为 LiveData<User>。现在,更新数据时,会通知 UserProfileFragment。此外,由于此 LiveData 字段具有生命周期感知能力,因此当不再需要引用时,会自动清理它们。

UserProfileViewModel

public class UserProfileViewModel extends ViewModel {
    ...
    private LiveData<User> user;
    public LiveData<User> getUser() {
        return user;
    }
}

现在,我们修改 UserProfileFragment 以观察数据并更新界面:

UserProfileFragment

@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);
    viewModel.getUser().observe(this, user -> {
      // Update UI.
    });
}

每次更新用户个人资料数据时,都会调用 onChanged() 回调并刷新界面。

如果你熟悉使用可观察回调的其他库,你可能已经意识到,我们并未替换 Fragment 的 onStop() 方法以停止观察数据。使用 LiveData 时没必要执行此步骤,因为它具有生命周期感知能力。这意味着,除非 Fragment 处于活跃状态(即,已接收 onStart() 但尚未接收 onStop()),否则它不会调用 onChanged() 回调。当调用 Fragment 的 onDestroy() 方法时,LiveData 还会自动移除观察者。

此外,我们也没有添加任何逻辑来处理配置更改,如用户旋转设备的屏幕。UserProfileViewModel 会在配置更改后自动恢复,所以一旦创建新的 Fragment,它就会接收相同的 ViewModel 实例,并且会立即使用当前的数据调用回调。鉴于 ViewModel 对象应该比它们更新的相应 View 对象存在的时间更长,因此 ViewModel 实现中不得包含对 View 对象的直接引用。要详细了解与界面组件生命周期对应的 ViewModel 生命周期,请参阅 ViewModel 的生命周期

3.3 获取数据

现在,我们已使用 LiveData 将 UserProfileViewModel 连接到 UserProfileFragment,那么如何获取用户个人资料数据?

在本例中,我们假定后端提供 REST API。我们使用 Retrofit 库访问后端,不过你可以随意使用起着相同作用的其他库。

下面是与后端进行通信的 Webservice 的定义:

Webservice

public interface Webservice {
    /**
     * @GET declares an HTTP GET request
     * @Path("user") annotation on the userId parameter marks it as a
     * replacement for the {user} placeholder in the @GET path
     */
    @GET("/users/{user}")
    Call<User> getUser(@Path("user") String userId);
}

实现 ViewModel 的第一个想法可能是直接调用 Webservice 来获取数据,然后将该数据分配给 LiveData 对象。这种设计行得通,但如果采用这种设计,随着应用的扩大,应用会变得越来越难以维护。这样会给 UserProfileViewModel 类太多的责任,这就违背了分离关注点原则。此外,ViewModel 的时间范围与 ActivityFragment 生命周期相关联,这意味着,当关联界面对象的生命周期结束时,会丢失 Webservice 的数据。此行为会带来不良的用户体验。

ViewModel 会将数据获取过程委派给一个新的模块,即存储区。

存储区模块会处理数据操作。它们会提供一个干净的 API,以便应用的其余部分可以轻松检索该数据。数据更新时,它们知道从何处获取数据以及进行哪些 API 调用。你可以将存储区视为不同数据源(如持久性模型、网络服务和缓存)之间的媒介。

UserRepository 类(如以下代码段中所示)使用 WebService 实例来获取用户的数据:

UserRepository

public class UserRepository {
    private Webservice webservice;
    // ...
    public LiveData<User> getUser(int userId) {
        // This isn't an optimal implementation. We'll fix it later.
        final MutableLiveData<User> data = new MutableLiveData<>();
        webservice.getUser(userId).enqueue(new Callback<User>() {
            @Override
            public void onResponse(Call<User> call, Response<User> response) {
                data.setValue(response.body());
            }

            // Error case is left out for brevity.
        });
        return data;
    }
}

虽然存储区模块看起来是不必要的,但它起着一项重要的作用:它会从应用的其余部分中提取数据源。现在,UserProfileViewModel 不知道如何获取数据,因此我们可以为视图模型提供从几个不同的数据获取实现获得的数据。

注意:为简单起见,我们省去了网络错误情况。有关公开错误和加载状态的替代实现,请参阅附录:公开网络状态。

管理组件之间的依赖关系

上方的 UserRepository 类需要一个 Webservice 实例才能获取用户的数据。它可以直接创建该实例,但要做到这一点,它还需要知道 Webservice 类的依赖项。此外,UserRepository 或许不是唯一一个需要 Webservice 的类。这种情况需要我们复制代码,因为每个需要引用 Webservice 的类都需要知道如何构造该实例及其依赖项。如果每个类都创建一个新的 WebService,应用会变得非常耗费资源。

你可以使用以下设计模式来解决此问题:

  • 依赖注入 (DI):依赖注入使类能够定义其依赖项而不构造它们。在运行时,另一个类负责提供这些依赖项。我们建议使用 Dagger 2 库在 Android 应用中实现依赖注入。Dagger 2 通过遍历依赖项树自动构造对象,并为依赖项提供编译时保证。

  • 服务定位器:服务定位器模式提供了一个注册表,类可以从中获取其依赖项而不构造它们。

实现服务注册表比使用 DI 更容易,因此,如果你不熟悉 DI,请改用服务定位器模式。

你可以借助这些模式来扩展代码,因为它们可提供清晰的依赖项管理模式(无需复制代码,也不会增添复杂性)。此外,你还可以借助这些模式在测试和生产数据获取实现之间快速切换。

我们的示例应用使用 Dagger 2 来管理 Webservice 对象的依赖项。

3.4 连接 ViewModel 与存储区

现在,我们修改 UserProfileViewModel 以使用 UserRepository 对象:

UserProfileViewModel

public class UserProfileViewModel extends ViewModel {
    private LiveData<User> user;
    private UserRepository userRepo;

    // Instructs Dagger 2 to provide the UserRepository parameter.
    @Inject
    public UserProfileViewModel(UserRepository userRepo) {
        this.userRepo = userRepo;
    }

    public void init(int userId) {
        if (this.user != null) {
            // ViewModel is created on a per-Fragment basis, so the userId
            // doesn't change.
            return;
        }
        user = userRepo.getUser(userId);
    }

    public LiveData<User> getUser() {
        return this.user;
    }
}

3.5 缓存数据

UserRepository 实现会抽象化对 Webservice 对象的调用,但由于它只依赖于一个数据源,因此不是很灵活。

UserRepository 实现的关键问题是,它从后端获取数据后,不会将该数据存储在任何位置。因此,如果用户在离开 UserProfileFragment 后再返回该类,则应用必须重新获取数据,即使数据未发生更改也是如此。

这种设计不够理想,原因如下:

  • 浪费了宝贵的网络带宽。

  • 迫使用户等待新的查询完成。

为了弥补这些缺点,我们向 UserRepository 添加了一个新的数据源,用于将 User 对象缓存在内存中:

UserRepository

// Informs Dagger that this class should be constructed only once.
@Singleton
public class UserRepository {
    private Webservice webservice;

    // Simple in-memory cache. Details omitted for brevity.
    private UserCache userCache;

    public LiveData<User> getUser(int userId) {
        LiveData<User> cached = userCache.get(userId);
        if (cached != null) {
            return cached;
        }

        final MutableLiveData<User> data = new MutableLiveData<>();
        userCache.put(userId, data);

        // This implementation is still suboptimal but better than before.
        // A complete implementation also handles error cases.
        webservice.getUser(userId).enqueue(new Callback<User>() {
            @Override
            public void onResponse(Call<User> call, Response<User> response) {
                data.setValue(response.body());
            }
        });
        return data;
    }
}

3.6 保留数据

使用我们当前的实现时,如果用户旋转设备或离开应用后立即返回应用,则现有界面将立即变为可见,因为存储区将从内存中的缓存检索数据。

不过,如果用户离开应用,数小时后当 Android 操作系统已终止进程后再回来,会发生什么?在这种情况下,如果依赖我们当前的实现,则需要再次从网络中获取数据。这一重新获取的过程不仅是一种糟糕的用户体验,而且很浪费资源,因为它会消耗宝贵的移动数据。

你可以通过缓存网络请求来解决此问题,但这样做会带来一个值得关注的新问题:如果相同的用户数据因另一种类型的请求(如获取好友列表)而显示出来,会发生什么?应用将会显示不一致的数据,这样比较容易让用户感到困惑。例如,如果用户在不同的时间发出好友列表请求和单一用户请求,应用可能会显示同一用户的数据的两个不同版本。应用将需要弄清楚如何合并这些不一致的数据。

处理这种情况的正确方法是使用持久性模型。这就是 Room 持久性库的用武之地。

Room 是一个对象映射库,可利用最少的样板代码实现本地数据持久性。在编译时,它会根据数据架构验证每个查询,这样损坏的 SQL 查询会导致编译时错误而不是运行时失败。Room 可以抽象化处理原始 SQL 表格和查询的一些底层实现细节。它还允许你观察对数据库数据(包括集合和连接查询)的更改,并使用 LiveData 对象公开这类更改。它甚至明确定义了解决一些常见线程问题(如访问主线程上的存储空间)的执行约束。

注意:如果你的应用已使用 SQLite 对象关系映射 (ORM) 等其他持久性解决方案,那么你无需将现有解决方案替换为 Room。不过,如果你正在编写新应用或重构现有应用,那么我们建议你使用 Room 保留应用数据。这样一来,你便可以利用该库的抽象和查询验证功能。

要使用 Room,我们需要定义本地架构。首先,我们向 User 数据模型类添加 @Entity 注解,并向该类的 id 字段添加 @PrimaryKey 注解。这些注解会将 User 标记为数据库中的表格,并将 id 标记为该表格的主键:

User

@Entity
class User {
  @PrimaryKey
  private int id;
  private String name;
  private String lastName;

  // Getters and setters for fields.
}

然后,我们通过为应用实现 RoomDatabase 来创建一个数据库类:

UserDatabase

@Database(entities = {User.class}, version = 1)
public abstract class UserDatabase extends RoomDatabase {
}

请注意,UserDatabase 是抽象类。Room 将自动提供它的实现。详情请参阅 Room 文档。

现在,我们需要一种将用户数据插入数据库的方法。为了完成此任务,我们创建一个数据访问对象 (DAO)

UserDao

@Dao
public interface UserDao {
    @Insert(onConflict = REPLACE)
    void save(User user);
    @Query("SELECT * FROM user WHERE id = :userId")
    LiveData<User> load(int userId);
}

请注意,load 方法将返回一个 LiveData<User> 类型的对象。Room 知道何时修改了数据库,并会自动在数据发生更改时通知所有活跃的观察者。由于 Room 使用 LiveData,因此此操作很高效;仅当至少有一个活跃的观察者时,它才会更新数据。

注意:Room 将根据表格的修改情况检查是否失效,这意味着它可能会发出误报通知。

定义 UserDao 类后,从我们的数据库类引用该 DAO:

UserDatabase

@Database(entities = {User.class}, version = 1)
public abstract class UserDatabase extends RoomDatabase {
    public abstract UserDao userDao();
}

现在,我们可以修改 UserRepository 以纳入 Room 数据源:

@Singleton
public class UserRepository {
    private final Webservice webservice;
    private final UserDao userDao;
    private final Executor executor;

    @Inject
    public UserRepository(Webservice webservice, UserDao userDao, Executor executor) {
        this.webservice = webservice;
        this.userDao = userDao;
        this.executor = executor;
    }

    public LiveData<User> getUser(String userId) {
        refreshUser(userId);
        // Returns a LiveData object directly from the database.
        return userDao.load(userId);
    }

    private void refreshUser(final String userId) {
        // Runs in a background thread.
        executor.execute(() -> {
            // Check if user data was fetched recently.
            boolean userExists = userDao.hasUser(FRESH_TIMEOUT);
            if (!userExists) {
                // Refreshes the data.
                Response<User> response = webservice.getUser(userId).execute();

                // Check for errors here.

                // Updates the database. The LiveData object automatically
                // refreshes, so we don't need to do anything else here.
                userDao.save(response.body());
            }
        });
    }
}

请注意,虽然我们在 UserRepository 中更改了数据的来源,但不需要更改 UserProfileViewModelUserProfileFragment。这种小范围的更新展示了我们的应用架构所具有的灵活性。这也很适合测试,因为我们可以提供虚假的 UserRepository,与此同时测试生产 UserProfileViewModel

如果用户等待几天后再返回使用此架构的应用,他们很可能会看到过时的信息,直到存储区可以获取更新的信息。根据你的用例,你可能不希望显示这些过时的信息。你可以显示占位符数据,此类数据将显示虚拟值并指示你的应用当前正在获取并加载最新信息。

单一可信来源

不同的 REST API 端点返回相同的数据是一种很常见的现象。例如,如果我们的后端有其他端点返回好友列表,则同一个用户对象可能来自两个不同的 API 端点,甚至可能使用不同的粒度级别。如果 UserRepository 按原样从 Webservice 请求返回响应,而不检查一致性,则界面可能会显示混乱的信息,因为来自存储区的数据的版本和格式将取决于最近调用的端点。

因此,我们的 UserRepository 实现会将网络服务响应保存在数据库中。这样一来,对数据库的更改将触发对活跃 LiveData 对象的回调。使用此模型时,数据库会充当单一可信来源,应用的其他部分则使用 UserRepository 对其进行访问。无论你是否使用磁盘缓存,我们都建议你的存储区将某个数据源指定为应用其余部分的单一可信来源。

3.7 显示正在执行的操作

在某些用例(如拉动刷新)中,界面务必要向用户显示当前正在执行某项网络操作。将界面操作与实际数据分离开来是一种很好的做法,因为数据可能会因各种原因而更新。例如,如果我们获取了好友列表,可能会以编程方式再次获取相同的用户,从而触发 LiveData<User> 更新。从界面的角度来看,传输中的请求只是另一个数据点,类似于 User 对象本身中的其他任何数据。

我们可以使用以下某个策略,在界面中显示一致的数据更新状态(无论更新数据的请求来自何处):

  • 更改 getUser() 以返回一个 LiveData 类型的对象。此对象将包含网络操作的状态。

有关示例,请参阅 Android 架构组件 GitHub 项目中的 NetworkBoundResource 实现。

  • UserRepository 类中再提供一个可以返回 User 刷新状态的公共函数。如果你只想在数据获取过程源自于显式用户操作(例如滑动以刷新)时在界面中显示网络状态,使用此策略效果会更好。

3.8 测试每个组件

在分离关注点部分中,我们已经提到,遵循此原则的一个主要好处是可测试性。

以下列表显示了如何测试扩展示例中的每个代码模块:

  • 界面和交互:使用 Android 界面插桩测试。创建此测试的最佳方法是使用 Espresso 库。你可以创建 Fragment 并为其提供模拟 UserProfileViewModel。由于 Fragment 只与 UserProfileViewModel 通信,因此模拟这一个类就足以完整测试应用的界面。

  • ViewModel:你可以使用 JUnit 测试来测试 UserProfileViewModel 类。你只需模拟一个类,即 UserRepository

  • UserRepository:你也可以使用 JUnit 测试来测试 UserRepository。你需要模拟 WebserviceUserDao。在这些测试中,请验证以下行为:

    • 存储区是否进行了正确的网络服务调用。

    • 存储区是否将结果保存到数据库中。

    • 在数据已缓存且保持最新状态时,存储区是否不会发出不必要的请求。

由于 WebserviceUserDao 都是接口,因此你可以模拟它们或者为更复杂的测试用例创建虚假实现。

  • UserDao:使用插桩测试来测试 DAO 类。由于这些插桩测试不需要任何界面组件,因此它们会快速运行。

对于每个测试,都请创建内存中数据库以确保测试没有任何副作用(例如更改磁盘上的数据库文件)。

注意:Room 允许指定数据库实现,因此可以通过提供 SupportSQLiteOpenHelper 的 JUnit 实现来测试 DAO。不过,不建议你使用这种方法,因为设备上运行的 SQLite 版本可能与开发设备上的 SQLite 版本不同。

  • Webservice:在这些测试中,请避免对后端进行网络调用。所有测试(尤其是基于网络的测试)都务必独立于外界。

有几个库(包括 MockWebServer)可帮助你为这些测试创建虚假的本地服务器。

  • 测试软件工件:架构组件提供了一个可控制其后台线程的 maven 软件工件。android.arch.core:core-testing 软件工件包含以下 JUnit 规则:

  • InstantTaskExecutorRule:使用此规则可立即在调用线程上执行任何后台操作。

  • CountingTaskExecutorRule:使用此规则可等待架构组件的后台操作。你还可以将此规则作为空闲资源与 Espresso 关联。

四、最佳做法

编程是一个创造性的领域,构建 Android 应用也不例外。无论是在多个 Activity 或 Fragment 之间传递数据,检索远程数据并将其保留在本地以在离线模式下使用,还是复杂应用遇到的任何其他常见情况,解决问题的方法都会有很多种。

虽然以下建议不是强制性的,但根据我们的经验,从长远来看,遵循这些建议会使你的代码库更强大、可测试性更高且更易维护:

  • 请避免将应用的入口点(如 Activity、Service 和广播接收器)指定为数据源。
    相反,你应只将其与其他组件协调,以检索与该入口点相关的数据子集。每个应用组件存在的时间都很短暂,具体取决于用户与其设备的交互情况以及系统当前的整体运行状况。

  • 在应用的各个模块之间设定明确定义的职责界限。
    例如,请勿在代码库中将从网络加载数据的代码散布到多个类或软件包中。同样,也不要将不相关的职责(如数据缓存和数据绑定)定义到同一个类中。

  • 尽量少公开每个模块中的代码。
    请勿试图创建“就是那一个”快捷方式来呈现一个模块的内部实现细节。短期内,你可能会省了点时间,但随着代码库的不断发展,你可能会反复陷入技术上的麻烦。

  • 考虑如何使每个模块可独立测试。
    例如,如果使用明确定义的 API 从网络获取数据,将会更容易测试在本地数据库中保留该数据的模块。如果你将这两个模块的逻辑混放在一处,或将网络代码分散在整个代码库中,那么即便能够进行测试,难度也会大很多。

  • 专注于应用的独特核心,以使其从其他应用中脱颖而出。
    不要一次又一次地编写相同的样板代码,这是在做无用功。相反,你应将时间和精力集中放在能让应用与众不同的方面上,并让 Android 架构组件以及建议的其他库处理重复的样板。

  • 保留尽可能多的相关数据和最新数据。
    这样,即使用户的设备处于离线模式,他们也可以使用你应用的功能。请注意,并非所有用户都能享受到稳定的高速连接。

  • 将一个数据源指定为单一可信来源。
    每当应用需要访问这部分数据时,这部分数据都应一律源于此单一可信来源。

五、附录:公开网络状态

在上文的推荐应用架构部分中,我们省略了网络错误和加载状态以简化代码段。

本部分将演示如何使用可封装数据及其状态的 Resource 类来公开网络状态。

以下代码段提供了 Resource 的实现示例:

// A generic class that contains data and status about loading this data.
public class Resource<T> {
    @NonNull public final Status status;
    @Nullable public final T data;
    @Nullable public final String message;
    private Resource(@NonNull Status status, @Nullable T data,
            @Nullable String message) {
        this.status = status;
        this.data = data;
        this.message = message;
    }

    public static <T> Resource<T> success(@NonNull T data) {
        return new Resource<>(Status.SUCCESS, data, null);
    }

    public static <T> Resource<T> error(String msg, @Nullable T data) {
        return new Resource<>(Status.ERROR, data, msg);
    }

    public static <T> Resource<T> loading(@Nullable T data) {
        return new Resource<>(Status.LOADING, data, null);
    }

    public enum Status { SUCCESS, ERROR, LOADING }
}

有一个很常见的情况是,一边从网络加载数据,一边显示这些数据在磁盘上的副本,因此建议你创建一个可在多个地方重复使用的辅助程序类。在本例中,我们创建一个名为 NetworkBoundResource 的类。

下图显示了 NetworkBoundResource 的决策树:

它首先观察资源的数据库。首次从数据库中加载条目时,NetworkBoundResource 会检查结果是好到足以分派,还是应从网络中重新获取。请注意,考虑到你可能会希望在通过网络更新数据的同时显示缓存的数据,这两种情况可能会同时发生。

如果网络调用成功完成,它会将响应保存到数据库中并重新初始化数据流。如果网络请求失败,NetworkBoundResource 会直接分派失败消息。

注意:在将新数据保存到磁盘后,我们会重新初始化来自数据库的数据流。不过,通常我们不需要这样做,因为数据库本身正好会分派更改。

请注意,依赖于数据库来分派更改将产生相关副作用,这样不太好,原因是,如果由于数据未更改而使得数据库最终未分派更改,就会出现这些副作用的未定义行为。

此外,不要分派来自网络的结果,因为这样将违背单一可信来源原则。毕竟,数据库可能包含在“保存”操作期间更改数据值的触发器。同样,不要在没有新数据的情况下分派 SUCCESS,因为如果这样做,客户端会接收错误版本的数据。

以下代码段显示了 NetworkBoundResource 类为其子级提供的公共 API:

// ResultType: Type for the Resource data.
// RequestType: Type for the API response.
public abstract class NetworkBoundResource<ResultType, RequestType> {
    // Called to save the result of the API response into the database.
    @WorkerThread
    protected abstract void saveCallResult(@NonNull RequestType item);

    // Called with the data in the database to decide whether to fetch
    // potentially updated data from the network.
    @MainThread
    protected abstract boolean shouldFetch(@Nullable ResultType data);

    // Called to get the cached data from the database.
    @NonNull @MainThread
    protected abstract LiveData<ResultType> loadFromDb();

    // Called to create the API call.
    @NonNull @MainThread
    protected abstract LiveData<ApiResponse<RequestType>> createCall();

    // Called when the fetch fails. The child class may want to reset components
    // like rate limiter.
    @MainThread
    protected void onFetchFailed();

    // Returns a LiveData object that represents the resource that's implemented
    // in the base class.
    public final LiveData<Resource<ResultType>> getAsLiveData();
}

请注意有关该类定义的以下重要细节:

  • 它定义了两个类型参数(ResultTypeRequestType),因为从 API 返回的数据类型可能与本地使用的数据类型不匹配。

  • 它对网络请求使用了一个名为 ApiResponse 的类。ApiResponseRetrofit2.Call 类的一个简单封装容器,可将响应转换为 LiveData 实例。

NetworkBoundResource 类的完整实现作为 Android 架构组件 GitHub 项目的一部分出现。

创建 NetworkBoundResource 后,我们可以使用它在 UserRepository 类中编写磁盘和网络绑定的 User 实现:

UserRepository

class UserRepository {
    Webservice webservice;
    UserDao userDao;

    public LiveData<Resource<User>> loadUser(final int userId) {
        return new NetworkBoundResource<User,User>() {
            @Override
            protected void saveCallResult(@NonNull User item) {
                userDao.insert(item);
            }

            @Override
            protected boolean shouldFetch(@Nullable User data) {
                return rateLimiter.canFetch(userId)
                        && (data == null || !isFresh(data));
            }

            @NonNull @Override
            protected LiveData<User> loadFromDb() {
                return userDao.load(userId);
            }

            @NonNull @Override
            protected LiveData<ApiResponse<User>> createCall() {
                return webservice.getUser(userId);
            }
        }.getAsLiveData();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352