Android学习之蓝牙操作

Android

蓝牙是一种支持设备短距离传输数据的无线技术。android在2.0以后提供了这方面的支持。 从查找蓝牙设备到能够相互通信要经过几个基本步骤(本机做为服务器): 1.设置权限 在manifest中配置

Xml代码  

  1. <uses-permission android:name="android.permission.BLUETOOTH"/>
  2. <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
<uses-permission android:name="android.permission.BLUETOOTH"/>
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>

2.启动蓝牙 首先要查看本机是否支持蓝牙,获取BluetoothAdapter蓝牙适配器对象

Java代码  

  1. BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
  2. if(mBluetoothAdapter == null){
  3. //表明此手机不支持蓝牙
  4. return;
  5. }
  6. if(!mBluetoothAdapter.isEnabled()){ //蓝牙未开启,则开启蓝牙
  7. Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
  8. startActivityForResult(enableIntent, REQUEST_ENABLE_BT);
  9. }
  10. //......
  11. public void onActivityResult(int requestCode, int resultCode, Intent data){
  12. if(requestCode == REQUEST_ENABLE_BT){
  13. if(requestCode == RESULT_OK){
  14. //蓝牙已经开启
  15. }
  16. }
  17. }
BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
if(mBluetoothAdapter == null){
        //表明此手机不支持蓝牙
        return;
}
if(!mBluetoothAdapter.isEnabled()){	//蓝牙未开启,则开启蓝牙
			Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
			startActivityForResult(enableIntent, REQUEST_ENABLE_BT);
}
//......
public void onActivityResult(int requestCode, int resultCode, Intent data){
       if(requestCode == REQUEST_ENABLE_BT){
              if(requestCode == RESULT_OK){
                   //蓝牙已经开启
              }
       }
}

3。发现蓝牙设备 这里可以细分为几个方面 (1)使本机蓝牙处于可见(即处于易被搜索到状态),便于其他设备发现本机蓝牙

Java代码  

  1. //使本机蓝牙在300秒内可被搜索
  2. private void ensureDiscoverable() {
  3. if (mBluetoothAdapter.getScanMode() !=
  4. BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE) {
  5. Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
  6. discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
  7. startActivity(discoverableIntent);
  8. }
  9. }
//使本机蓝牙在300秒内可被搜索
private void ensureDiscoverable() {
        if (mBluetoothAdapter.getScanMode() !=
            BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE) {
            Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
            discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
            startActivity(discoverableIntent);
        }
}

(2)查找已经配对的蓝牙设备,即以前已经配对过的设备

Java代码  

  1. Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
  2. if (pairedDevices.size() > 0) {
  3. findViewById(R.id.title_paired_devices).setVisibility(View.VISIBLE);
  4. for (BluetoothDevice device : pairedDevices) {
  5. //device.getName() +" "+ device.getAddress());
  6. }
  7. } else {
  8. mPairedDevicesArrayAdapter.add("没有找到已匹对的设备");
  9. }
		Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
		if (pairedDevices.size() > 0) {
			findViewById(R.id.title_paired_devices).setVisibility(View.VISIBLE);
			for (BluetoothDevice device : pairedDevices) {
				//device.getName() +" "+ device.getAddress());
			}
		} else {
			mPairedDevicesArrayAdapter.add("没有找到已匹对的设备");
		}

(3)通过mBluetoothAdapter.startDiscovery();搜索设备,要获得此搜索的结果需要注册 一个BroadcastReceiver来获取。先注册再获取信息,然后处理

Java代码  

  1. //注册,当一个设备被发现时调用onReceive
  2. IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
  3. this.registerReceiver(mReceiver, filter);
  4. //当搜索结束后调用onReceive
  5. filter = new IntentFilter(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
  6. this.registerReceiver(mReceiver, filter);
  7. //.......
  8. private BroadcastReceiver mReceiver = new BroadcastReceiver() {
  9. @Override
  10. public void onReceive(Context context, Intent intent) {
  11. String action = intent.getAction();
  12. if(BluetoothDevice.ACTION_FOUND.equals(action)){
  13. BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
  14. // 已经配对的则跳过
  15. if (device.getBondState() != BluetoothDevice.BOND_BONDED) {
  16. mNewDevicesArrayAdapter.add(device.getName() + "\n" + device.getAddress());  //保存设备地址与名字
  17. }
  18. }else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(action)) {  //搜索结束
  19. if (mNewDevicesArrayAdapter.getCount() == 0) {
  20. mNewDevicesArrayAdapter.add("没有搜索到设备");
  21. }
  22. }
  23. }
  24. };
//注册,当一个设备被发现时调用onReceive
IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
		this.registerReceiver(mReceiver, filter);

//当搜索结束后调用onReceive
filter = new IntentFilter(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
		this.registerReceiver(mReceiver, filter);
//.......
private BroadcastReceiver mReceiver = new BroadcastReceiver() {
		@Override
		public void onReceive(Context context, Intent intent) {
			String action = intent.getAction();
			if(BluetoothDevice.ACTION_FOUND.equals(action)){
				 BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
	              // 已经配对的则跳过
	             if (device.getBondState() != BluetoothDevice.BOND_BONDED) {
	                  mNewDevicesArrayAdapter.add(device.getName() + "\n" + device.getAddress());  //保存设备地址与名字
	             }
			}else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(action)) {  //搜索结束
                if (mNewDevicesArrayAdapter.getCount() == 0) {
                    mNewDevicesArrayAdapter.add("没有搜索到设备");
                }
			}

		}
};

4.建立连接 查找到设备 后,则需要建立本机与其他设备之间的连接。 一般用本机搜索其他蓝牙设备时,本机可以作为一个服务端,接收其他设备的连接。 启动一个服务器端的线程,死循环等待客户端的连接,这与ServerSocket极为相似。 这个线程在准备连接之前启动

Java代码  

  1. //UUID可以看做一个端口号
  2. private static final UUID MY_UUID =
  3. UUID.fromString("fa87c0d0-afac-11de-8a39-0800200c9a66");
  4. //像一个服务器一样时刻监听是否有连接建立
  5. private class AcceptThread extends Thread{
  6. private BluetoothServerSocket serverSocket;
  7. public AcceptThread(boolean secure){
  8. BluetoothServerSocket temp = null;
  9. try {
  10. temp = mBluetoothAdapter.listenUsingRfcommWithServiceRecord(
  11. NAME_INSECURE, MY_UUID);
  12. } catch (IOException e) {
  13. Log.e("app", "listen() failed", e);
  14. }
  15. serverSocket = temp;
  16. }
  17. public void run(){
  18. BluetoothSocket socket=null;
  19. while(true){
  20. try {
  21. socket = serverSocket.accept();
  22. } catch (IOException e) {
  23. Log.e("app", "accept() failed", e);
  24. break;
  25. }
  26. }
  27. if(socket!=null){
  28. //此时可以新建一个数据交换线程,把此socket传进去
  29. }
  30. }
  31. //取消监听
  32. public void cancel(){
  33. try {
  34. serverSocket.close();
  35. } catch (IOException e) {
  36. Log.e("app", "Socket Type" + socketType + "close() of server failed", e);
  37. }
  38. }
  39. }
//UUID可以看做一个端口号
private static final UUID MY_UUID =
        UUID.fromString("fa87c0d0-afac-11de-8a39-0800200c9a66");
   //像一个服务器一样时刻监听是否有连接建立
    private class AcceptThread extends Thread{
    	private BluetoothServerSocket serverSocket;

    	public AcceptThread(boolean secure){
    		BluetoothServerSocket temp = null;
    		try {
	    		temp = mBluetoothAdapter.listenUsingRfcommWithServiceRecord(
	    					NAME_INSECURE, MY_UUID);
    		} catch (IOException e) {
				  Log.e("app", "listen() failed", e);
			}
    		serverSocket = temp;
    	}

    	public void run(){
    		BluetoothSocket socket=null;
    		while(true){
    			try {
					socket = serverSocket.accept();
				} catch (IOException e) {
					 Log.e("app", "accept() failed", e);
	                 break;
				}
    		}
    		if(socket!=null){
    			//此时可以新建一个数据交换线程,把此socket传进去
    		}
    	}

    	//取消监听
    	public void cancel(){
	        try {
	            serverSocket.close();
	        } catch (IOException e) {
	            Log.e("app", "Socket Type" + socketType + "close() of server failed", e);
	        }
    	}

}

搜索到设备后可以获取设备的地址,通过此地址获取一个BluetoothDeviced对象,可以看做客户端,通过此对象device.createRfcommSocketToServiceRecord(MY_UUID);同一个UUID可与服务器建立连接获取另一个socket对象,由此服务端与客户端各有一个socket对象,此时 他们可以互相交换数据了。 创立客户端socket可建立线程

Java代码  

  1. //另一个设备去连接本机,相当于客户端
  2. private class ConnectThread extends Thread{
  3. private BluetoothSocket socket;
  4. private BluetoothDevice device;
  5. public ConnectThread(BluetoothDevice device,boolean secure){
  6. this.device = device;
  7. BluetoothSocket tmp = null;
  8. try {
  9. tmp = device.createRfcommSocketToServiceRecord(MY_UUID_SECURE);
  10. } catch (IOException e) {
  11. Log.e("app", "create() failed", e);
  12. }
  13. }
  14. public void run(){
  15. mBluetoothAdapter.cancelDiscovery();    //取消设备查找
  16. try {
  17. socket.connect();
  18. } catch (IOException e) {
  19. try {
  20. socket.close();
  21. } catch (IOException e1) {
  22. Log.e("app", "unable to close() "+
  23. " socket during connection failure", e1);
  24. }
  25. connetionFailed();  //连接失败
  26. return;
  27. }
  28. //此时可以新建一个数据交换线程,把此socket传进去
  29. }
  30. public void cancel() {
  31. try {
  32. socket.close();
  33. } catch (IOException e) {
  34. Log.e("app", "close() of connect  socket failed", e);
  35. }
  36. }
  37. }
    //另一个设备去连接本机,相当于客户端
    private class ConnectThread extends Thread{
    	private BluetoothSocket socket;
    	private BluetoothDevice device;
    	public ConnectThread(BluetoothDevice device,boolean secure){
    		this.device = device;
    		BluetoothSocket tmp = null;
    		try {
				tmp = device.createRfcommSocketToServiceRecord(MY_UUID_SECURE);
			} catch (IOException e) {
				 Log.e("app", "create() failed", e);
			}
    	}

    	public void run(){
    		mBluetoothAdapter.cancelDiscovery();	//取消设备查找
    		try {
				socket.connect();
			} catch (IOException e) {
				try {
					socket.close();
				} catch (IOException e1) {
					 Log.e("app", "unable to close() "+
	                            " socket during connection failure", e1);
				}
				connetionFailed();	//连接失败
				return;
			}
	        //此时可以新建一个数据交换线程,把此socket传进去
    	}

    	  public void cancel() {
              try {
                  socket.close();
              } catch (IOException e) {
                  Log.e("app", "close() of connect  socket failed", e);
              }
          }
    }

5.建立数据通信线程,进行读取数据

Java代码  

  1. //建立连接后,进行数据通信的线程
  2. private class ConnectedThread extends Thread{
  3. private BluetoothSocket socket;
  4. private InputStream inStream;
  5. private OutputStream outStream;
  6. public ConnectedThread(BluetoothSocket socket){
  7. this.socket = socket;
  8. try {
  9. //获得输入输出流
  10. inStream = socket.getInputStream();
  11. outStream = socket.getOutputStream();
  12. } catch (IOException e) {
  13. Log.e("app", "temp sockets not created", e);
  14. }
  15. }
  16. public void run(){
  17. byte[] buff = new byte[1024];
  18. int len=0;
  19. //读数据需不断监听,写不需要
  20. while(true){
  21. try {
  22. len = inStream.read(buff);
  23. //把读取到的数据发送给UI进行显示
  24. Message msg = handler.obtainMessage(BluetoothChat.MESSAGE_READ,
  25. len, -1, buff);
  26. msg.sendToTarget();
  27. } catch (IOException e) {
  28. Log.e("app", "disconnected", e);
  29. connectionLost();   //失去连接
  30. start();    //重新启动服务器
  31. break;
  32. }
  33. }
  34. }
  35. public void write(byte[] buffer) {
  36. try {
  37. outStream.write(buffer);
  38. // Share the sent message back to the UI Activity
  39. handler.obtainMessage(BluetoothChat.MESSAGE_WRITE, -1, -1, buffer)
  40. .sendToTarget();
  41. } catch (IOException e) {
  42. Log.e("app", "Exception during write", e);
  43. }
  44. }
  45. public void cancel() {
  46. try {
  47. socket.close();
  48. } catch (IOException e) {
  49. Log.e("app", "close() of connect socket failed", e);
  50. }
  51. }
  52. }
//建立连接后,进行数据通信的线程
    private class ConnectedThread extends Thread{
    	private BluetoothSocket socket;
    	private InputStream inStream;
    	private OutputStream outStream;

    	public ConnectedThread(BluetoothSocket socket){

    		this.socket = socket;
    		try {
    			//获得输入输出流
				inStream = socket.getInputStream();
				outStream = socket.getOutputStream();
			} catch (IOException e) {
				Log.e("app", "temp sockets not created", e);
			}
    	}

    	public void run(){
    		byte[] buff = new byte[1024];
    		int len=0;
    		//读数据需不断监听,写不需要
    		while(true){
    			try {
					len = inStream.read(buff);
					//把读取到的数据发送给UI进行显示
					Message msg = handler.obtainMessage(BluetoothChat.MESSAGE_READ,
							len, -1, buff);
					msg.sendToTarget();
				} catch (IOException e) {
					Log.e("app", "disconnected", e);
                    connectionLost();	//失去连接
                    start();	//重新启动服务器
                    break;
				}
    		}
    	}

    	public void write(byte[] buffer) {
            try {
                outStream.write(buffer);

                // Share the sent message back to the UI Activity
                handler.obtainMessage(BluetoothChat.MESSAGE_WRITE, -1, -1, buffer)
                        .sendToTarget();
            } catch (IOException e) {
                Log.e("app", "Exception during write", e);
            }
        }

        public void cancel() {
            try {
                socket.close();
            } catch (IOException e) {
                Log.e("app", "close() of connect socket failed", e);
            }
        }
    }

到这里,蓝牙通信的基本操作已经全部完成。

转自:http://zhouyunan2010.iteye.com/blog/1186021

时间: 2024-10-07 08:04:50

Android学习之蓝牙操作的相关文章

Android学习笔记-ContentProvider操作

---恢复内容开始--- 之前写了一个用SQlite来实现增删改查的应用,今天又新学了一个用ContentProvider来操作的增删改查 首先ContentProvider是用来共享数据的,那么咱们先来建立一个数据源,之后用其他程序获得共享的ContentProvider,来实现CRUD   数据源结构,一共三个Java文件 SQLDatabaseHelper.java 1 package com.example.sqlcz; 2 3 import android.content.Contex

Android学习之Image操作及时间日期选择器

一.基础学习 1.ImageView是图片容器,就相当于RadioGroup是RadioButton的容器一样,是View的直接子类. 1 <imageview android:id="@+id/img" android:layout_width="fill_parent" android:layout_height="wrap_content" android:src="@drawable/logo"></

【Android学习】蓝牙扫描

activity_main.xml <?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/re

Android学习--------实现增删改查数据库操作以及实现类似微信好友对话管理操作

最近的一个实验用到东西挺多,特地总结一下. 要求功能: 1.第一个页面添加歌曲名和歌手,跳到第二个页面列表显示所有记录,使用sqlite数据库 2.对于第二个页面显示的信息可以进行删除和修改操作,并自动刷新 最后我做的效果: 长按列表项弹出单个管理的菜单,像微信好友对话管理一样. 删除时可以直接删除这一条并在列表中直接显示,更新时弹出对话框更新内容提交后在列表中重新显示. 做此例遇到的困难: 1.菜单获取上下文 2.获取对话框中的内容 3.对话框按钮添加监听事件-----注意包不要导错:impo

九、Android学习第八天——广播机制与WIFI网络操作(转)

(转自:http://wenku.baidu.com/view/af39b3164431b90d6c85c72f.html) 九.Android学习第八天——广播机制与WIFI网络操作 今天熟悉了Android中的广播机制与WIFI网络的一些基本操作,总结如下: Android的广播机制 我们知道广播机制中,发送方不会关心接收方时候接收到数据或者如何去处理数据. 这里总结下Android中BroadcastReceiver的注册方法: (一)在应用程序中进行注册 (二)在Manifest.xml

Android蓝牙操作笔记(转)

蓝牙是一种支持设备短距离传输数据的无线技术.android在2.0以后提供了这方面的支持. 从查找蓝牙设备到能够相互通信要经过几个基本步骤(本机做为服务器): 1.设置权限 在manifest中配置 <uses-permission android:name="android.permission.BLUETOOTH"/> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN&qu

Android学习---通过内容提供者(ContentProvider)操作另外一个应用私有数据库的内容

一.什么是ContentProvider? ContentProvider直译过来就是内容提供者,主要作用就是A应用提供接口给B应用调用数据,和之前介绍的sharedPreference和直接开放文件访问类似,都是共享应用程序数据,不同的是之前的两种文件格式可能完全不同,如可能为xml,txt,sql等等,这里ContentProvider返回的数据格式是统一的,因此应用的更为广泛一点. 二.实例 这里使用的是A应用通过ContentProvider共享数据给B应用.这里A应用用的是前文中的an

Android——4.2 蓝牙 BlueTooth 初探入门 (一)

蓝牙,wifi,3g 作为移动端平台最常见的dongle,蓝牙现在应用越来越广泛,特别是现在流行智能终端,多点互控,wifi,3g等资源消耗大,红外又有距离和障碍物的限制,所以说蓝牙通信在未来其它的不说,单在智能TV上应用的情景将会越来越广泛.这里只是初步理解学习,之后将会一层一层从kernel直到application 撰写不易,转载需注明出处:http://blog.csdn.net/jscese/article/details/41283879 结构层次: 先看一张层次图: 可以看到大体分

Android端简易蓝牙聊天通讯App(原创)

欢迎转载,但请注明出处!谢谢.http://www.cnblogs.com/weizhxa/p/5792775.html 最近公司在做一个蓝牙串口通讯的App,有一个固定的蓝牙设备,需要实现手机连接相互交换数据.以前没怎么做过蓝牙开发,故查看Android App Guide的蓝牙篇,发现有个chat示例,故此做了点研究.在研究的基础上进行了此App的实现. 1.App特点: 1.1 App中同时存在服务器与客户端,任意手机可以作为服务器或者客户端: 1.2 客户端可以进行蓝牙环境扫描: 1.3