首页 > 代码库 > 项目中的“蓝牙”

项目中的“蓝牙”

           之前写过一篇关于蓝牙的文章,我把它称为译文,就是把官方的BluetoothChat那个程序拿过来说android中的蓝牙的操作。因为项目要用到蓝牙,所以这方面的文章看得也不少,不过果然文章都是大同小异,基本都是那个蓝牙聊天程序的照抄,然后修改一下而已。所以当我做这个项目的时候,就很头痛,因为没有多少可以参考的资料,所以一直研究那个聊天程序,现在算是,彻底弄懂了蓝牙的所有操作部分。当然我这篇文章不是想说那个程序,我是想说说我那个项目中的蓝牙操作,希望能给点不一样的蓝牙资料,供做项目的学习,也算是一种思路吧。

    直接说我那个项目对蓝牙的要求。这是一个轨道磨耗检测上位机程序,和下位机通信用的蓝牙。协议部分是这样的,首先在登陆进去的activity里面建立蓝牙连接,重点和核心是在整个程序的运行中要保证连接一直存在,也许你们不懂,这么说吧,就是我这里面有很多操作的activity,拿测量这一项来说,首先是用户选择线路的这个activity,然后才是测量的10个值的显示activity,还有测量值的绘图activity,这只是测量,还有数据库查询等等。为什么说这是重点呢?想想那个蓝牙聊天程序,它其实只是在主activity那里实现通信的,当然我相信你不会提议每个activity都建立一次连接这么傻的问题。协议还没说完,建立连接只是第一步,建立连接之后,在测量activity,我们要给下位机发送开始测量的指令码,然后下位机把这段指令码回发,当接收到这段指令码,才开始发送测量数据指令,然后返回的就是数据。这里有两点,第一,上位机是一直在发送测量数据指令,(隔100ms,下位机接收和发送有延时);第二,接收到的数据要CRC校验,校验通过才能显示在界面上,错误数据要抛弃。

    上面说了很多,表达能力不行,直接简单的上图说吧。

    第一副图是登陆界面进去的第一个activity,蓝牙是在这里面建立连接,并一直保持,在整个程序的运行中。第二幅图是线路检测里面的第二个界面,要把下位机上的这些数据都要读出来,因为下位机是一直在测量,所以显示也是动态变化的。之前我模仿蓝牙聊天程序在第二幅图的界面做了蓝牙建立,很简单就成功了,可以和下位机通信,要求都满足,但是问题在于,每次用户换一个地点测量新数据,就要重现连接一次设备,然后才能开始测量。这么麻烦,作为成品是不能忍受的,所以也就要求:一次连接,保证不退出程序就可以一直使用蓝牙,这是这篇文章的重点。

    首先在第一副图建立蓝牙连接不难做到,参考蓝牙聊天程序都可以实现。在以后的过程中都要保持连接,我把它理解为,我们在第一副图那个界面实例化的bluetoothservice的一个实例在以后的任何一个activity中都要用到,所以我用的方法是把这个实例变成“全局变量”,也就是保存在application中的变量,然后我在测量这个activity中要用到蓝牙,所以拿出这个全局变量,就保证了这个连接是一次连接,以后每个activity都是可用的。这里不深说全局变量,其实也没什么特别,也算是android数据传递的一种方法。给出部分程序:

public class MyApp extends Application {

	private String correct13 = "0";
	private String correct25 = "0";
	private String correct48 = "0";
	private String correct65 = "0";
	private BluetoothService mChatService = null;
	
	public String[] getcorrect(){
		String[] str = new String[5];
		str[0] = correct13;
		str[1] = correct25;
		str[3] = correct48;
		str[4] = correct65;
		return str;
	};
	
	public void setcorrect(String[] str){
		this.correct13 = str[0];
		this.correct25 = str[1];
		this.correct48 = str[3];
		this.correct65 = str[4];
	}
	
	public BluetoothService getservice(){
		return mChatService;
	}
	
	public void setservice(BluetoothService service){
		this.mChatService = service;
	}
	

	
}

    第一副图那个activity中设置这个service,在测量那个activity中获取这个service。

MyApp app = (MyApp)getApplicationContext();
		app.setservice(mChatService);

MyApp app = (MyApp)getApplicationContext();
		mChatService = app.getservice();

    接下来说的是数据传输。上面的设置成全局变量只能保证发送指令是没有问题的,但是接收读取数据,这个是有问题的。因为handler传递的数据是传到第一副图的那个activity中的,而不是测量那个activity中。但是显示却是在测量那个activity中显示,这就是问题所在,这么说应该能懂。至于为什么传递的下位机的数据是到第一个activity而不是测量那个activity,这个其实很简单,因为是在第一个activity中实例化的service,看蓝牙聊天程序就知道实例化时是要传入context和handler的,这个handler是在第一个activity中申明的,所以,只能传给第一个activity。

    那能不能也用全局变量把第一个activity里面的handler的数据变成全局的呢?其实是不行的,因为这是一个动态的数据显示,就是下位机一直在发送数据,而不是发一次,handler更新了数据,但是不能更新全局变量里面的数据,所以做不到在测量activity里面获取到动态数据。

    对于上面分析,其实可以明白重点就是怎么动态传递数据这个问题,何不考虑broadcastreceiver呢?是的,我用的就是广播机制,把handler的数据广播出去,接收到下位机数据就广播一次,然后在测量activity里面接收广播,然后把数据处理显示出来,思路就是这样。部分代码如下:

public void send(byte[] readBuf){
			
		    Intent intent = new Intent("android.intent.action.MY_BROADCAST");  
		    intent.putExtra("data", readBuf);  
		    sendBroadcast(intent);  
		    Log.i("chz", "发了");

		}

    这是发送广播。

//广播
  	 public class myReceiver extends BroadcastReceiver{
  		 
		public void onReceive(Context context, Intent intent) {
			        	
			readBuf = intent.getByteArrayExtra("data");			        	
			Log.i("chz", "收到了");
			String kaishicl = new String(bytesHexTobytes(readBuf));
			if(kaishicl.equals("01060001FFFFD9BA000000000000000000000000000000")){
			            	mHander.post(mRunnable);
			}else{
			//construct a string from the valid bytes in the buffer
			 byte[] newbytes = new byte[readBuf.length - 2];//获取不含校验位的字节数组
			 System.arraycopy(readBuf, 0, newbytes, 0, readBuf.length - 2);
			 String read = new String(bytesHexTobytes(newbytes));
			 String checkMessage = CRC16M.getBufHexStr(CRC16M.getSendBuf(read));//得到生成的校验后字符串
			 String readMessage = new String(bytesHexTobytes(readBuf));
			          		
			 if(checkMessage.equals(readMessage)){//比较两个字符串是否相等

			String guangliangdaichoice = guangliangdai.getSelectedItem().toString();
			if(guangliangdaichoice.equals("光亮带位置(mm)")){
			      //光亮带位置
			     String guangliangdaiweizhiets = showlight(readBuf[17],readBuf[18]);
			     guangliangdaiet.setText(guangliangdaiweizhiets);
			}else{
			      //光亮带宽度
			     Cursor cursor = sqlitedb.rawQuery("select * from data", null);
			     int id = 1;
			     while(cursor.moveToNext()){
			     id++;
			    };
			    Cursor cur = sqlitedb.rawQuery("select * from data where id = " + Integer.toString(id-1), null);
			    String temp = "";
			    while(cur.moveToNext()){
			    	   temp += cur.getString(13);
			    }
			    	    			
			    String value = http://www.mamicode.com/showlight(readBuf[17],readBuf[18]);>

     这是广播的操作,还好这个操作不是耗时的,所以无所谓。

	IntentFilter filter = new IntentFilter();
        filter.addAction("android.intent.action.MY_BROADCAST");
        this.registerReceiver(myreceiver, filter);

     这个添加一个过滤器获取确定的广播。
   

     嗯,其实我这里只是提供一个思路来解决遇到的问题,相信大家也会有自己的方法和思路。