Android / Java线程同步:while(true){}导致阻塞

时间:2016-06-01 04:05:03

标签: java android multithreading while-loop

我正在尝试更好地了解我的Android应用中线程的行为。出于某种原因,当我在我的一个工作线程中使用while(true)时,该线程的run方法中的代码在while(true)循环之前顺序存在,从不执行。为了清楚起见,我不确定代码(Toast消息)是否实际上没有执行,或者Android OS处理线程同步的方式是否导致我的Toast消息不显示。这种行为似乎是某种阻塞,但我无法弄清楚为什么会发生这种情况。

我的应用程序使用3个线程:UI线程(Android应用程序中的默认/主线程),在运行时从设备的USB端口无限读取数据的线程,以及通过来自USB的消息处理此数据的线程读线程。问题似乎发生在我的USBController类中。当我注释掉我的无限循环时,循环开始之前的所有Toast消息都显示正常。 当我没有评论我的时候(真的),没有任何东西消息显示!我对此非常困惑,我认为我误解了Android操作系统处理线程的一些基本信息。即使while循环导致阻塞,我不认为它因为它驻留在工作线程中,为什么不会触发在while循环之前发生的Toast消息?这是同步问题?我是否滥用了Android的Handler-Looper系统?

以下代码。注意:我已经包含了主要活动的相关部分和整个USBController类。我对这个类的实现很大程度上依赖于mik3y/usb-serial-for-android中的USB到串行库。我不认为这是必要的,但我已经包含了包含我的第三个线程SensorDataBuffer的类,它接收来自线程UsbController的消息。

UsbController.java

    public class UsbController extends Thread{
    ...
    @Override
    public void run() {
        android.os.Process.setThreadPriority(android.os.Process.THREAD_PRIORITY_DEFAULT); //sets thread to default queing priority
        Looper.prepare();
        Toast.makeText(mContext.getApplicationContext(), "Hello from UsbController's run method!", Toast.LENGTH_SHORT).show();

        // **********************USB otg*******************************
        //Obtain permission to use Android device's USB intent
        PendingIntent mPermissionIntent;
        mPermissionIntent = PendingIntent.getBroadcast(mContext, 0, new Intent(ACTION_USB_PERMISSION), 0);

        // Find all available drivers from attached devices.
        ProbeTable customTable = new ProbeTable();
        customTable.addProduct(0x03EB, 0x2044, CdcAcmSerialDriver.class);                 
        UsbManager manager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
        UsbSerialProber prober = new UsbSerialProber(customTable);
        List<UsbSerialDriver> availableDrivers = prober.findAllDrivers(manager);

        if (availableDrivers.isEmpty()) {
            Toast.makeText(mContext.getApplicationContext(), "No available USB drivers found",Toast.LENGTH_SHORT).show(); // Toast message for debugging
        }
        else {                                                  // open connection to first avail. driver
            UsbSerialDriver driver = availableDrivers.get(0);
            Toast.makeText(mContext.getApplicationContext(), "Driver found",Toast.LENGTH_SHORT).show(); // Toast message for debugging
            UsbDeviceConnection connection = manager.openDevice(driver.getDevice());
            Toast.makeText(mContext.getApplicationContext(), "Device Driver Opened",Toast.LENGTH_SHORT).show(); // Toast message for debugging
            if (connection == null) {           // You probably need to call UsbManager.requestPermission(driver.getDevice(), ..)
                Toast.makeText(mContext.getApplicationContext(),"Connection to device not allowed, need permissions",Toast.LENGTH_LONG).show();
                manager.requestPermission(driver.getDevice(),mPermissionIntent);  //conn test
                if (manager.hasPermission(driver.getDevice())==true){
                    Toast.makeText(mContext.getApplicationContext(),"Permissions granted",Toast.LENGTH_SHORT).show();
                }
            }
            else {                      // Read some data! Most have just one port (port 0).
                List<UsbSerialPort> myPortList = driver.getPorts();
                UsbSerialPort port = myPortList.get(0);
                Toast.makeText(mContext.getApplicationContext(),"USB OTG Connection Established",Toast.LENGTH_SHORT).show();
                try {
                    port.open(connection);
                    port.setParameters(9600, 8, UsbSerialPort.STOPBITS_1, UsbSerialPort.PARITY_NONE); // sets baud rate,databits, stopbits, & parity
                    port.setDTR(true);                 //necessary to make Arduino Micro begin running it's program
                    Toast.makeText(mContext.getApplicationContext(),"port opened, parameters set, DTR set",Toast.LENGTH_SHORT).show();
                    byte buffer[] = new byte[16];      
                    String incompPacket = "";
                    Toast.makeText(mContext.getApplicationContext(), "hi again!"), Toast.LENGTH_LONG).show();
                    while (true){                  //continuous loop to read data
                        numBytesRead = port.read(buffer, 100);          
                        arduinoData = new String(buffer, "US-ASCII");
                        String raw = arduinoData.substring(0, numBytesRead);
                        if (numBytesRead > 0) {
                            ...
                        }
                    }
                } catch (IOException e) {
                    Toast.makeText(mContext, e.getMessage(), Toast.LENGTH_SHORT).show();
                }
            }
        }
        Looper.loop(); 
    }
}

MainActivity.java

...
    @Override
    protected void onCreate(Bundle savedInstanceState) {

        //Multi-threading
        //Create thread to handle incoming data from USB Controller thread
        SensorDataBuffer pressureDataBuffer = new SensorDataBuffer(MainActivity.this);
        Thread bufferThread = new Thread(pressureDataBuffer);
        bufferThread.start();

        //Create USB Serial Worker thread which will continuously receive data
        UsbController serialDataLink = new UsbController(PlayFrets.this);
        Thread sensorMonitorThread = new Thread(serialDataLink);
        sensorMonitorThread.start();
        //Toast.makeText(this, "USB Controller thread started", Toast.LENGTH_SHORT).show();

        //Build GUI
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);           //Removes action bar from display
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);  //Removes status bar from display

        //Create AsyncTask to load the note files. A splash screen will be displayed while task is executing
        new AsyncTask_NoteFileLoader(this).execute();
        }
...

SensorDataBuffer.java

public class SensorDataBuffer extends Thread{

    //Handler subclass which accepts messages one by one in
    //the main activitiy's FIFO message que called a "Looper"
    //The worker thread, sensorMonitor, runs UsbController in parallel
    //with the UI thread and continuously formats and sends pressure sensor
    //values read from the microcontroller to the Handler which updates the
    //corresponding pressure state logic variables in the UI thread.
    public void run(){
        android.os.Process.setThreadPriority(Process.THREAD_PRIORITY_URGENT_AUDIO); //TODO:priority was previously more favorable, test this to ensure UI doesn't lag
        Looper.prepare(); //create MessageQue to receive messages from USB Controller thread
        UsbController.setHandler(bufferHandler);

        bufferHandler = new Handler(Looper.myLooper()) {
                //do stuff
        };
        Looper.loop();
    }
}

1 个答案:

答案 0 :(得分:0)

如何使用HandlerThreads,Handlers和Runnables呢?使您的代码更清晰,更易于维护。

在你的onCreate()中只需创建几个:

HandlerThread usbThread = new HandlerThread("USBController");
usbThread.start();
usbHandler = new Handler(usbThread.getLooper());

HandlerThread sensorThread = new HandlerThread("SensorDataBuffer");
sensorThread.start();
sensorHandler = new Handler(sensorThread.getLooper());

然后创建Runnables并将它们发布到Handlers

usbHandler.post(new Runnable(){
    run(){
        //....
        numBytesRead = port.read(buffer, 100);
            if (numBytesRead > 0) {
               sensorHandler.post(new Runnable(){run(){//doSomething}});
            }
        //....
        if(isStillRunning)
            usbHandler.post(this);
    }
});

你可以让runnable帖子本身,它将永远运行。从内部可以将runnable发布到其他处理程序(如主线程处理程序)以显示您的Toasts。