上周参照网上代码,做了USB的初步探测程序,工作正常 。今天从硬件部拿到了一段例程,原本打算参考它来完善自己的程序。但运行之后总是报错,逐步跟进错误,进而发现了一个匪疑所思的问题。调试一天也未发现原因以及解决办法。在此做记录,留待以后解决或从网上得到帮助。

问题程序:

UsbHid.java:

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

public class UsbHid extends Activity{
    private static UsbHid usbHid = null;
    Context context;
    private static final String TAG = "UsbHid";
    private UsbManager myUsbManager;
    private UsbDevice myUsbDevice;
    private boolean HasUsbDevice;
    ...

    private UsbHid(Context context) {
        this.context = context;
        myUsbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
        HasUsbDevice = enumerateDevice();
        Log.d(TAG, "UsbHid init");
    }

    public bollean enumerateDevice() {
        if (myUsbManager == null)
            return -1;
        HashMap<String, UsbDevice> deviceList = myUsbManager.getDeviceList();
        if (!deviceList.isEmpty()) {
            ...
            return  true;
        }
        return false;
    }

    public static UsbHid getInstance(Context context) {
        if (usbHid == null) {
            usbHid = new UsbHid(context);
        }
        return usbHid;
    }
    public boolean getUsbDevice() {
        return HasUsbDevice;
    }
    ...
}
 

MainActivity.java:

public class MainActivity extends ActionBarActivity {
 UsbHid usbHid = null;

 UsbManager myUsbManager;
 UsbDevice myUsbDevice;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

  usbHid = UsbHid.getInstance(this);

        if (usbHid.getUsbDevice()) {
            usbHid.ConnectUsbHid();
            this.setTitle("已连接USB设备");
        } else {
            this.setTitle("末检测到USB设备");
        }
        if (usbHid.getUsbDevice()) {
            UsbReadThread thread = new UsbReadThread();
            thread.start();
        }
  }
  ...
}

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

以上程序从原理上分析应该没有问题,但实际中deviceList.isEmpty()总是返回null。

更为奇怪的是,若将UsbHid.java中的

myUsbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
HasUsbDevice = enumerateDevice();
移到MainActivity.java中,则没有问题,可以正常枚举设备。

前一天写的以上内容,今天原本打算暂时绕过以上问题,继续进行后续开发,毕竟这个问题不是不可避免的。但还是觉得应该再基于这个问题研究一下,因为这样才能够提高水平,丰富经验。

进行反复修改尝试,根据现象判断,最终发现了问题所在:

UsbHid.java:

public static UsbHid getInstance(Context context) {
        if (usbHid == null) {
            usbHid = new UsbHid(context);
        }
        return usbHid;
    }

就是这句导致的问题!将这句程序封掉后,可以枚举到设备,程序可以正常工作了。

具体原因尚不清楚,但初步分析应该是由于usbHid是静态变量(private static UsbHid usbHid = null;),放在全局存储区,因此回收的时候并没有马上释放掉。下次进程再启动时,它并不是null,而是上次一分配过的值。基于这个想法,做了实验,将上面这段代码改写如下:

public static UsbHid getInstance(Context context) {
        if (usbHid == null) {
            usbHid = new UsbHid(context);

return usbHid;
        }
        return null;
    }
改写程序的目的是测试是否每次都返回usbHid而非null。宏观上从程序运行现象来看应该是如果每次都返回usbHid,则每次程序都正常;如果有返回null的时候,则程序会停止运行。

测试结果印证了 猜测,一次成功接一次失败,即确实usbHid在前一次进程结束后并未彻底释放,而直到出错被停止运行后才被释放掉。

另外一个测试也验证了以上想法:将手机重启,之后运行以上程序,由于是第一此运行,因此完全正常,而后每次即使不插入usb设备,也不报错。这就说明了usbHid这个变量并没有被释放干净。

至于为何进程结束后,仍不能是放干净,则有待进一步深入,应该是与Java的回收机制有关了……

 

调试Android USB遇到的令人费解的问题的更多相关文章

  1. Eclipse中调试Android技巧

    Android eclipse中程序调试 一:断点调试 用eclipse开发android程序的时,跟VS一样是可以断点单步调试的. 步骤如下. 1 设置断点:在编码窗体的左边框上用鼠标双击,或者右键 ...

  2. 在 Chrome 中调试 Android 浏览器

    最近需要使用 Chrome Developer Tools 调试 Android 浏览器,但是官方指南并不是很好使,经过一番折腾,终于调试成功了,在此把经验分享给需要的朋友. Chrome Devel ...

  3. Web程序员开发App系列 - 调试Android和IOS手机代码(补图)

    Web程序员开发App系列 Web程序员开发App系列 - 认识HBuilder Web程序员开发App系列 - 申请苹果开发者账号 Web程序员开发App系列 - 调试Android和iOS手机代码 ...

  4. 在ubuntu下真机调试android程序出现设备没有访问权限

    今天把android的开发环境从windows平台切换到了ubuntu上. java jdk android-adt android-ndk都下好,环境变量都配好之后, 在调试程序的时候,出现设备没有 ...

  5. 理解WebKit和Chromium: 调试Android系统上的Chromium

    转载请注明原文地址:http://blog.csdn.net/milado_nju 1. Android上的调试技术 在Android系统上,开发人员能够使用两种不同的语言来开发应用程序,一种是Jav ...

  6. I.MX6 Android USB Touch eGTouchA.ini文件存放

    /******************************************************************** * I.MX6 Android USB Touch eGTo ...

  7. 通过Wifi调试Android应用

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  8. android usb Host模式下与usb Hid 设备的通信

    做android 与USB HID设备的通信有段时间了,总结一下遇到的问题和解决方法: 1,第一次遇到的问题:android 版本低不支持usb hid, 被要求做相关项目的时候,就从mUsbMana ...

  9. 如何通过Chrome远程调试android设备上的Web网站

    网上的帖子很多,但很多都是老版本的,试过了,根本不管用,花了一天时间,终于在本机试验通过了,特记录下来,以备用.有需要的朋友也可以参考.先上一张图,看看PC端chrome上调试的效果: 左边是手机的模 ...

随机推荐

  1. hadoop命令报错:权限问题

    root用户执行hadoop命令报错: [root@vmocdp125 conf]# hadoop fs -ls /user/ [INFO] 17:50:42 main [RetryInvocatio ...

  2. java多线程总结三:sleep()、join()、interrupt()示例

    这是一个来自<java编程思想上的示例> package demo.thread; /** *sleep()是静态方法,是属于类的,作用是让当前线程阻塞 *join()是使线程同步,如在某 ...

  3. iOS开发——手机号,密码,邮箱,身份证号,中文判断

    目前这些方面的判断主要是用了正则表达式 手机号的判断,目前主要是长度.均是数字,支持的号段由于第三方通讯比如京东通讯,小米通讯等支持的号段挺多, 有171,170,135,147等等,所以号段限制简单 ...

  4. 关于Ionic的安装

    Ionic是一个前端的框架,帮助开发者使用HTML5, CSS3和JavaScript做出原生应用. http://ionicframework.com/getting-started/ 这里介绍了如 ...

  5. NOIP201501&&02

    NOIP201501金币 试题描述 国王将金币作为工资,发放给忠诚的骑士.第一天,骑士收到一枚金币:之后两天(第二天和第三天) ,每天收到两枚金币:之后三天(第四.五.六天) ,每天收到三枚金币:之后 ...

  6. mysql 相关环境配置

    1. mysql 版本要求 5.5 以上, 否则不支持 emoji 表情. $ sudo apt-get install mysql-server $ sudo vim /etc/mysql/my.c ...

  7. 建造者模式(Builder Pattern)

    建造者模式:使用多个简单对象一步步构建成一个复杂的对象. 有时候,我们会创建一个“复杂”的对象,这个对象的由很多子对象构成,由于需求的变化,这个对象的各个部分经常面临剧烈的变化. 继续工厂模式的披萨店 ...

  8. PHOTOSHOP 半透明方格

    1.新建60*60的透明文档,在左方和上方用直线工具画白边,存储为图案(编辑/定义图案) 2.新建图层,用油漆桶填充图案 3. 选择若干小方格,填充白色后设置不透明度50%

  9. Oracle 自连接 / 外连接 / 子查询

    --连接查询的三种格式 select ename, deptno,dname from emp natural join dept; select ename, deptno,dname from e ...

  10. HTML5课程大纲/学习路线

    HTML5课程大纲/学习路线 这是什么? 这个一个HTML技术路线的课程大纲/学习大纲. 你能用它做什么? 如果你是找工作的人, 利用本大纲, 你可以学习HTML5语言, 做一个HTML前端工程师, ...