做程序员苦逼的地方就在于,当公司决定做什么的时候,是不会跟你商量的,只会跟你说,xxx,这个可能需要你来实现一下。fuck,想好实现思路了吗?(这是我司的程序员提出,我们来做整理完善的)

Android双屏显示,可能会和别的双屏机制不同,大多数情况下是一个android系统,分主副屏而已。我司的硬件是两个android系统,两个屏幕,内部通过一根usb直连(这根usb连接线很稳定,代工厂和我们讲的,坑~)。双屏运行两个独立的android系统,相互通过一条底层的通道传输数据,主屏通常可以运行业务软件。副屏可以显示宣传图片,视频,购物清单等信息,但不限于这些,实际上副屏也是个可以触摸交互的系统。

在这样的硬件前提下,我们开发需要实现这两个屏幕的通信,涉及到usb的驱动开发(由代工厂搞定),我们只需要调用jni的一些方法即可。上层应用之间的通信,类似于广播,主副屏可以相互发送接收,提供公共的api,可供其他的app调用,使之能实现自己的业务逻辑。

Android 双目 android 双屏_Android 双目

本片文章主要讲底层的Service的实现(也就是驱动的上一层)

思路

双屏通信,主屏会要求副屏显示一些文字(命令),图片(发送文件+命令),图文混合,甚至会发送一些音频,视频等大文件,几百M到几G不等。因为是双向通信,主屏发送指令过后,需要等待副屏的回调。通常如果是命令或者是小文件,毫秒级别内就能被处理掉。但如果是几个G的大文件呢?时间就被延时了,如果这个时候再有命令发送过来了,就会等待(需要维护一个任务列表),这样肯定是不好的。所以我们切割文件,将文件分包,一个一个的发送,最后拼装还原,这样即使中途了命令或者小文件,也能立马被处理掉。

我们有一个任务队列,service不断的去任务队列去轮询,取到任务,根据Task信息,区分是任务类别,做相应的处理。如果是文件的话,我们进行分包处理,这里,我暂定义的最大单个文件包为512kb,然后发送。副屏接收,拼装,还原(每个包有相应的头信息),在给主屏反馈结果,主屏做相应处理。

大致的一个流程图:

Android 双目 android 双屏_android_02

协议与机制

其实在整个流程中,我们主要要区分任务的来源,以及之后要反馈的源头,分包与还原包不能错乱,不然会产生脏数据。

那我们定义一下任务的类型:FileTask(文件任务),MemoryTask(内存任务,字符串之类的),ControlTask(控制任务)。对不同的任务类型处理是不同的,有的直接是内存传输,有的是写本地文件。

之前在usb通道还没有连通的时候,我们是用UDP协议来写demo实现的,在usb通来以后,直接改用就可以来。所以说,即使没有这个usb通道,你也可以用udp连接来测试两个手机直接的传输,只是这个速度就依赖于网络了。

具体实现

我们有一个底层service,CoreService,所有的发送、接收,回调都是靠它来实现的。那我们就具体围绕它来展开。

@Override
    public void onCreate() {
        mSerialPort = new SerialPort();
        connectRunable = new Connect();
        new Thread(connectRunable).start();
    }

SerialPort类里面是一些native的方法,通过jni来调用底层的usb驱动的,这个就略过来,各家的都是不一样的。我们只要知道它有读写的方法即可。

public native int read(int fd, byte[] data, int offset, int len);
    public native int write(int fd, byte[] data, int offset, int len);

Connect类是连接操作,副屏向主屏发起连接的请求。

class Connect implements Runnable {

        @Override
        public void run() {
            id = 0;
            // 在建立连接之前,断开之前所有的任务
            if (mapSend != null) {
                for (Integer i : mapSend.keySet()) {
                    for (SendTask sendTask : mapSend.get(i).tasksFile) {
                        try {
                            sendTask.setTaskState(TaskState.error_sendData);
                            sendTask.getCallback().sendError(sendTask.getID(), sendTask.getTaskState().get_code(),
                                    sendTask.getTaskState().get_message());
                        } catch (Exception e) {
                        }
                    }
                    for (SendTask sendTask : mapSend.get(i).tasksMemory) {
                        try {
                            sendTask.setTaskState(TaskState.error_sendData);
                            sendTask.getCallback().sendError(sendTask.getID(), sendTask.getTaskState().get_code(),
                                    sendTask.getTaskState().get_message());
                        } catch (Exception e) {
                        }
                    }
                }
            }

            // 重新初始化参数
            mapSend = new ConcurrentHashMap<Integer, SendProcess>();
            mapRecv = new ConcurrentHashMap<Integer, RecvTask>();
            controlQueue = new ConcurrentLinkedQueue<ControlTask>();
            finshAndWait = new HashMap<Integer, Task>();
            errorSendTaskID = new HashSet<Integer>();
            errorRecvTaskID = new HashSet<Integer>();
            // 对象锁
            lock = new Object();
            // 线程控制锁
            controlThreadLock = new Object();
            controlProcess = new SendProcess(errorSendTaskID, controlThread, finshAndWait);
            mapSend.put(-1, controlProcess);
            controlThread = new ControlThread(controlQueue, controlProcess, lock, controlThreadLock, finshAndWait,
                    errorSendTaskID);
            controlProcess.setControlThread(controlThread);
            controlThread.start();

            while (true) {
                // usb端口打开
                if (mSerialPort.tryOpen(isMain)) {
                    // isMain true:表示主屏 false:表示副屏
                    if (isMain) {
                        byte[] temp = new byte[1];
                        boolean flag = false;
                        do {
                            // 读取建立请求连接的数据 -1
                            if (mSerialPort.read(mSerialPort.mFd, temp, 0, 1) <= 0) {
                                flag = true;
                                break;
                            } else if (temp[0] != -1 && temp[0] != -2) {
                                flag = true;
                                break;
                            } else if (temp[0] == -2) {
                                flag = false;
                                break;
                            }
                            // 向副屏发送建立请求连接的数据 -1
                            if (mSerialPort.write(mSerialPort.mFd, temp, 0, 1) <= 0) {
                                flag = true;
                                break;
                            }
                        } while (temp[0] == -1);
                        if (flag) {
                            continue;
                        }
                    } else {
                        // 向主屏发送建立请求连接的数据 -1
                        sendsyn = new sendSYN();
                        sendsynThread = new Thread(sendsyn);
                        sendsynThread.start();
                        try {
                            Thread.sleep(1000);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        byte[] temp = new byte[1];
                        mSerialPort.clear(mSerialPort.mFd);
                        if (mSerialPort.read(mSerialPort.mFd, temp, 0, 1) <= 0) {
                            sendsyn.gh = false;
                            continue;
                        } else if (temp[0] != -1) {
                            sendsyn.gh = false;
                            continue;
                        }
                        sendsyn.gh = false;
                        try {
                            sendsynThread.join();
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        temp[0] = -2;
                        // 读取建立请求连接的数据 -1
                        if (mSerialPort.write(mSerialPort.mFd, temp, 0, 1) <= 0) {
                            sendsyn.gh = false;
                            continue;
                        }
                    }
                    // 发送处理
                    transferSend = new TransferSend(lock, mapSend, mSerialPort, handler);
                    // 接收处理
                    transferRecv = new TransferRecv(mapRecv, mSerialPort, controlThread, errorRecvTaskID, handler);
                    transferRecv.start();
                    transferSend.start();
                    // 发送handler表示连接成功
                    handler.sendMessage(handler.obtainMessage(-2));
                    break;
                } else {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }

            isConnecting = false;
        }

    }

代码有点多,来分析一下初始化的整个连接过程。

// 发送task的map列表
    private ConcurrentHashMap<Integer, SendProcess> mapSend;
    // 接收task的map列表
    private ConcurrentHashMap<Integer, RecvTask> mapRecv;

1.首先,在建立连接之前,断开之前所有的任务,Task的TaskState.error_sendData就是表示usb数据通道断开,取消所有发送接收的任务。

2.初始化参数。包括发送,接收的列表,锁对象,任务id等,ControlThread类是一个线程控制类,来处理task的。SendProcess类是发送task的包装类,把它丢到ControlThread类去处理,然后开启Thread,让它不停的去轮询任务队列。

3.开始建立发送与接收的连接。isMain这个字段,true表示主屏,false表示副屏。先来看副屏的代码,初始化一个sendSYN类。它向主屏发送一个为-1字节的数据,然后如果主屏收到一个为-1的数据,就表示这是副屏发起的连接请求(正常的数据请求是不可能为-1 的)。主屏收到以后,也向副屏发送一个-1字节的数据,如果副屏也收到来这个数据,表示双屏建立连接成功。

连接通信成功,就可以相互发送数据了。来看下定义的Task这个类。

Android 双目 android 双屏_架构_03

这里面是一些数据信息,hasSendedLength,hasRecvLength,fileLength用来分包,还原包的,sender表示发送者,成功或失败要反馈发送者…

TaskType是一个枚举

Android 双目 android 双屏_Android 双目_04

分为文件任务,内存任务,控制任务三种。

那我们是如何向副屏发送任务的呢?是通过服务的aidl来发送的和接收回调信息的。

// SendService.aidl
interface SendService {

    int sendFileToFile(in String recvPackageName,in String path,boolean isReport, long userFlag,in SendServiceCallback callback);

    int sendByteToMemory(in String recvPackageName,in byte [] data,in SendServiceCallback callback);

}

在CoreService的onBind()方法中,返回了此对象。现在来看下SendService的具体实现。

@Override
    public IBinder onBind(Intent intent) {
        if (callback == null) {
            callback = new CallBack();
        }
        return callback;
    }

    private class CallBack extends SendService.Stub {
        @Override
        public int sendFileToFile(String recvPackageName, String path, boolean isReport, long userFlag,
                SendServiceCallback callback) throws RemoteException {
            SendProcess process = mapSend.get(Binder.getCallingUid());
            if (process == null) {
                process = new SendProcess(errorSendTaskID, controlThread, finshAndWait);
                mapSend.put(Binder.getCallingUid(), process);
            }
            int tem_id = getID();
            synchronized (lock) {
                process.addTask(new FileTask(tem_id,
                        getApplicationContext().getPackageManager().getNameForUid(Binder.getCallingUid()),
                        recvPackageName, path, isReport, userFlag, callback));
                lock.notify();
            }
            return tem_id;
        }

        @Override
        public int sendByteToMemory(String recvPackageName, byte[] data, SendServiceCallback callback)
                throws RemoteException {
            SendProcess process = mapSend.get(Binder.getCallingUid());
            if (process == null) {
                process = new SendProcess(errorSendTaskID, controlThread, finshAndWait);
                mapSend.put(Binder.getCallingUid(), process);
            }
            int tem_id = getID();
            synchronized (lock) {
                process.addTask(new MemoryTask(tem_id,
                        getApplicationContext().getPackageManager().getNameForUid(Binder.getCallingUid()),
                        recvPackageName, data, callback));
                lock.notify();
            }
            return tem_id;
        }
    }

主要是从map队列中取出客户端任务,放进任务列表,唤醒处理线程,执行任务。主要发送任务是SendProcess类,其中我们定义了sendM()和sendF()方法,来发送字符串命令和文件。

Android 双目 android 双屏_架构_05

Android 双目 android 双屏_android_06

最终都通过task的send()方法来发送,其实也就是前面所说的SerialPort中的write()这个jni方法。

Android 双目 android 双屏_架构_07

其中fillData()方法,也就是前面所有的给Task填充数据,包括请求头,大小,描述等。

下面我们再来看一下接收的方法,也是前面所说的SerialPort的read()来读取发送过来的数据,通过aidl回调主屏。其中的回调callback在任务创建的时候传入的,在控制线程和发送线程中对其作出相应的回调处理。

Android 双目 android 双屏_架构_08

// SendServiceCallback.aidl
interface SendServiceCallback {

    oneway void sendSuccess(int id);

    oneway void sendError(int id,int  errorId, String errorInfo);

    oneway void sendProcess(int id, long totle, long sended);

}

获取到数据,进行拼装还原,取出任务的携带信息,进行分类处理。

byte flag = data[4];
            int taskId = ByteUtil.bytes2Int(data, 5);
            fileTaskRecv = (RecvTask) mapRecv.get(taskId);
            int sendPackNameLength = ByteUtil.bytes2Int(data, 9);
            int recvPackNameLength = ByteUtil.bytes2Int(data, 13 + sendPackNameLength);
            if (fileTaskRecv == null) {
                if (errorTaskID.contains(taskId)) {
                    return;
                }
                String sendPackName = new String(data, 13, sendPackNameLength, "utf-8");
                String recvPackName = new String(data, 17 + sendPackNameLength, recvPackNameLength, "utf-8");
                int timeOut = ByteUtil.bytes2Int(data, 17 + recvPackNameLength + sendPackNameLength);
                long fileLength = ByteUtil.bytes2long(data, 21 + recvPackNameLength + sendPackNameLength);
                long userFlag = ByteUtil.bytes2long(data, 29 + recvPackNameLength + sendPackNameLength);

最后通过Service中的Handler来回调处理,其实是通过广播发送出去的,所以需要接收双屏通信信息的app都要注册该广播。

Android 双目 android 双屏_android_09

存在问题与改进空间

这样,双屏通信的大致流程就已经说完了,其中有一些需要补充和完善的地方,因为项目紧急,所以第一版上线的也比较粗糙,后续会陆续改进的。但我想,不管是怎样改,分包,发送,接收,回调,原包这样逻辑应该是通用的。

问题:1.接收到的文件没有处理,因为sdcard的内存是有限的。
2.如果连接断开(像强行关机之类),失败的任务是不能复原的。在初始化的时候,我们把所有以前的任务都当作失败任务放弃掉了。这里其实可以做一些缓存处理,在拿出未完成的任务,在重连的时候继续处理。

还有一些东西可能还没有考虑到,在后面无尽的需求中,也许会增加上去。

Beating Heart !