Commit bc5f9705 by pye52

1、完成CommandViewModel的重构

2、部分业务逻辑转移到Repositoy
parent c18d21a7
......@@ -59,7 +59,7 @@ public class CommandHelper {
* true => 数据库已有"未完成"状态的该类指令,不应该重复插入 <br/>
* false => 数据库中该类指令都"已完成"或没有该类指令,可插入新的指令 <br/>
*/
public static boolean oneAtTime(Command command) {
public static boolean oncePerTime(Command command) {
return command.getAction().equals(APP_UPDATE);
}
......
......@@ -4,9 +4,12 @@ import androidx.lifecycle.LiveData;
import com.bgycc.smartcanteen.data.dao.CommandDao;
import com.bgycc.smartcanteen.entity.Command;
import com.blankj.utilcode.util.LogUtils;
import java.util.List;
import static com.bgycc.smartcanteen.utils.SmartCanteenUtils.TAG;
public class CommandRepository {
private CommandDao dao;
......@@ -26,6 +29,17 @@ public class CommandRepository {
return dao.insertCommand(command);
}
public long insertCommand(Command command, boolean oncePerTime) {
if (oncePerTime) {
int count = queryUndoneCommandCountByAction(command.getAction());
if (count > 0) {
LogUtils.d(TAG, "该指令已在数据库中记录并未执行完毕,不再插入重复指令");
return 0;
}
}
return insertCommand(command);
}
public int finishCommand(Command command) {
command.finish();
return updateCommand(command);
......
package com.bgycc.smartcanteen.viewModel;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.Observer;
import androidx.lifecycle.ViewModel;
import androidx.work.OneTimeWorkRequest;
import com.bgycc.smartcanteen.command.CommandHelper;
import com.bgycc.smartcanteen.entity.Command;
import com.bgycc.smartcanteen.executor.SCTaskExecutor;
import com.bgycc.smartcanteen.socket.SCWebSocketClient;
import com.bgycc.smartcanteen.socket.SCWebSocketListener;
import com.bgycc.smartcanteen.socket.SCWebSocketListenerAdapter;
import com.bgycc.smartcanteen.repository.CommandRepository;
import com.blankj.utilcode.util.LogUtils;
import com.google.gson.Gson;
import com.google.gson.JsonObject;
import java.util.List;
import static com.bgycc.smartcanteen.utils.SmartCanteenUtils.TAG;
/**
* 监听本地扫码、服务器下发的设置指令 <br/>
* 监听Command数据库的变动,按如下流程进行处理: <br/>
* 从数据库读取 -> 解析执行 -> 更新到数据库
*/
public class CommandViewModel extends ViewModel {
private CommandRepository commandRepository;
private Gson gson;
private String deviceSN;
private MutableLiveData<Command> commandWorker = new MutableLiveData<>();
private LiveData<List<Command>> dataLiveData;
public LiveData<Command> getCommandTask() {
return commandWorker;
}
public CommandViewModel(CommandRepository commandRepository, Gson gson, String deviceSN) {
this.commandRepository = commandRepository;
this.gson = gson;
this.deviceSN = deviceSN;
// 监听数据库的变动,并执行未完成的指令
this.dataLiveData = commandRepository.queryUndoneCommand();
this.dataLiveData.observeForever(dataObserver);
}
public void initialize() {
SCWebSocketClient.getInstance().addListener(listener);
}
@Override
protected void onCleared() {
super.onCleared();
if (dataLiveData != null) {
dataLiveData.removeObserver(dataObserver);
}
}
private Observer<List<Command>> dataObserver = commands -> {
if (commands == null || commands.isEmpty()) {
return;
}
Command first = commands.get(0);
commandWorker.postValue(first);
};
public OneTimeWorkRequest getCommandWorker(Command command) {
return CommandHelper.createWorker(gson, command, deviceSN);
}
public void commandFinish(Command command) {
UpdateDatabaseRunnable runnable = new UpdateDatabaseRunnable(command);
SCTaskExecutor.getInstance().executeOnDiskIO(runnable);
}
private SCWebSocketListener listener = new SCWebSocketListenerAdapter() {
@Override
public void onMessage(String action, JsonObject obj, String original) {
if (!CommandHelper.isCommand(action)) return;
LogUtils.d(TAG, "设备下发指令: " + original);
ResponseRunnable runnable = new ResponseRunnable(original, action);
SCTaskExecutor.getInstance().executeOnDiskIO(runnable);
}
};
private class UpdateDatabaseRunnable implements Runnable {
private Command command;
UpdateDatabaseRunnable(Command command) {
this.command = command;
}
@Override
public void run() {
command.finish();
LogUtils.d(TAG, "指令执行完毕: " + command.toString());
commandRepository.updateCommand(command);
}
}
private class ResponseRunnable implements Runnable {
private String response;
private String action;
ResponseRunnable(String response, String action) {
this.response = response;
this.action = action;
}
@Override
public void run() {
// 指令插入到数据库,则会触发dataObserver
Command command = new Command(response, action);
// 该action同一时间是否只能在数据库存在一条待执行的指令
if (CommandHelper.oneAtTime(command)) {
int count = commandRepository.queryUndoneCommandCountByAction(action);
if (count > 0) {
LogUtils.d(TAG, "该指令已在数据库中记录并未执行完毕,不再插入重复指令");
return;
}
}
long lastInsertId = commandRepository.insertCommand(command);
if (lastInsertId == -1) {
LogUtils.w(TAG, "指令插入到数据库失败: " + command.toString());
}
}
}
}
......@@ -27,8 +27,7 @@ class CommandViewModel(
if (commands == null || commands.isEmpty()) {
return@Observer
}
val first = commands[0]
commandWorker.postValue(first)
commandWorker.postValue(commands.first())
}
init {
......@@ -67,15 +66,9 @@ class CommandViewModel(
viewModelScope.launch(coroutinesDispatcher) {
// 指令插入到数据库,则会触发dataObserver
val command = Command(original, action)
val oncePerTime = CommandHelper.oncePerTime(command)
// 该action同一时间是否只能在数据库存在一条待执行的指令
if (CommandHelper.oneAtTime(command)) {
val count: Int = commandRepository.queryUndoneCommandCountByAction(action)
if (count > 0) {
LogUtils.d(SmartCanteenUtils.TAG, "该指令已在数据库中记录并未执行完毕,不再插入重复指令")
return@launch
}
}
val lastInsertId: Long = commandRepository.insertCommand(command)
val lastInsertId: Long = commandRepository.insertCommand(command, oncePerTime)
if (lastInsertId == -1L) {
LogUtils.w(SmartCanteenUtils.TAG, "指令插入到数据库失败: $command")
}
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or sign in to comment