该测试程序是根据网上代码更改的,用于向另一蓝牙设备发送一图片文件。本文截图测试的是向PC上发送一指定图片(如果与要连接的设备未配对,会提示配对的)。

需要注意以下几个方面:
1. 传统的UUID方法(也是网络上流行的)连接其它蓝牙设备的方式根本行不通,在网络上搜索了很久终于找到一个替代的方法是可以工作的(详细见代码)
2. 关于蓝牙设置的两个属性:“开启关闭”与“设置可见”,这是两个独立设置选项,但M9手机将它们设置成关联了,即打开了蓝牙设备就自动设置为可见了,而设 置为可见后蓝牙设备也就打开了(手机UI设置里面无法单独操作“设置可见”,但代码可以),所以这里也纠结了一段时间,相关代码部分有说明

先看程序截图:
1. M9手机截图:开启蓝牙并搜索设备后

android 发送蓝牙消息 发送到蓝牙设备_Android开发

2. PC端截图:手机端点击搜索到的设备"wzc-0"后,PC端会提示权限操作

android 发送蓝牙消息 发送到蓝牙设备_android 发送蓝牙消息_02

3. PC端截图:当PC端允许操作后,接收从M9传来的图片(ubuntu系统蓝牙接收到文件会默认存储到:/home/yourname/下载)

android 发送蓝牙消息 发送到蓝牙设备_搜索_03

下面看代码:
首先看AndroidManifest.xml文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.testBlueTooth"
      android:versionCode="1"
      android:versionName="1.0">
    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".testBlueTooth" android:label="@string/app_name" 
        		android:launchMode="singleTask" android:screenOrientation="portrait">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>             
        </activity>
</application>
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
</manifest>

需要注意两点:
1. 最后两行的权限申明
2. <activity>里面的强制竖屏属性设置:android:launchMode="singleTask" android:screenOrientation="portrait",避免纵横屏转换时程序异常退出!

然后看布局文件main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	android:orientation="vertical" android:layout_width="fill_parent"
	android:layout_height="fill_parent">
	<LinearLayout android:id="@+id/LinearLayout01" 	android:layout_width="wrap_content" android:layout_height="wrap_content">
		<TextView android:id="@+id/TextView01" android:layout_height="wrap_content" 
				android:text="手机蓝牙开关" android:layout_width="100dip"></TextView>
		<ToggleButton android:layout_height="wrap_content" android:text="蓝牙开关" 
				android:layout_width="wrap_content" android:id="@+id/tbtnSwitch"></ToggleButton>
		<Button android:layout_height="wrap_content" android:text="本机蓝牙可见" 
				android:id="@+id/btnDis" android:layout_width="160dip"></Button>
	</LinearLayout>
	<LinearLayout android:id="@+id/LinearLayout02" android:layout_width="wrap_content" android:layout_height="wrap_content">
		<Button android:layout_height="wrap_content" android:id="@+id/btnSearch"
			android:text="搜索设备" android:layout_width="160dip"></Button>
		<Button android:layout_height="wrap_content" android:id="@+id/btnExit"
			android:text="退出程序" android:layout_width="160dip"></Button>
	</LinearLayout>
	<ListView android:layout_width="fill_parent" android:id="@+id/lvDevices"
		android:layout_height="fill_parent">
	</ListView>
</LinearLayout>

主要就是两个嵌套<LinearLayout>和一个<ListView>。

最后是主测试程序testBlueTooth.java:

package com.testBlueTooth;
 
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
//import java.util.UUID;
import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothSocket;
import android.content.BroadcastReceiver;
import android.content.ContentValues;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ListView;
import android.widget.Toast;
import android.widget.ToggleButton;
 
public class testBlueTooth extends Activity {
//static final String SPP_UUID = "00001101-0000-1000-8000-00805F9B34FB";
static final String FilePath = "file:///sdcard/test.jpg";
	Button btnSearch, btnDis, btnExit;
	ToggleButton tbtnSwitch;
	ListView lvBTDevices;
	ArrayAdapter<String> adtDevices;
	List<String> lstDevices = new ArrayList<String>();
	BluetoothAdapter btAdapt;
public static BluetoothSocket btSocket;
 
public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
// Button 设置
		btnSearch = (Button) this.findViewById(R.id.btnSearch);
		btnSearch.setOnClickListener(new ClickEvent());
		btnExit = (Button) this.findViewById(R.id.btnExit);
		btnExit.setOnClickListener(new ClickEvent());
		btnDis = (Button) this.findViewById(R.id.btnDis);
		btnDis.setOnClickListener(new ClickEvent());
 
// ToogleButton设置
		tbtnSwitch = (ToggleButton) this.findViewById(R.id.tbtnSwitch);
		tbtnSwitch.setOnClickListener(new ClickEvent());
 
// ListView及其数据源 适配器
		lvBTDevices = (ListView) this.findViewById(R.id.lvDevices);
		adtDevices = new ArrayAdapter<String>(testBlueTooth.this, android.R.layout.simple_list_item_1, lstDevices);
		lvBTDevices.setAdapter(adtDevices);
		lvBTDevices.setOnItemClickListener(new ItemClickEvent());
 
// 初始化本机蓝牙功能
		btAdapt = BluetoothAdapter.getDefaultAdapter();
 
/*
		 * 关于开关指示按钮,就是两装状态之间的切换:
		 *   打开: 表示当前服务状态为开启,而不是在点击之后才开启
		 *   关闭: 表示当前服务状态为关闭,而不是在点击之后才关闭
		 * 如果理解反了,则下面的false/true设置也会反的!
		 */
if (btAdapt.getState() == BluetoothAdapter.STATE_OFF)// 读取蓝牙状态并显示
			tbtnSwitch.setChecked(false);
else if (btAdapt.getState() == BluetoothAdapter.STATE_ON)
			tbtnSwitch.setChecked(true);
 
// 注册Receiver来获取蓝牙设备相关的结果
		IntentFilter intent = new IntentFilter();
		intent.addAction(BluetoothDevice.ACTION_FOUND);// 用BroadcastReceiver来取得搜索结果
		intent.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);
		intent.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED);
		intent.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
		registerReceiver(searchDevices, intent);
}
 
/*
	 * 搜索蓝牙设备列表
	 */
private BroadcastReceiver searchDevices = new BroadcastReceiver() {
public void onReceive(Context context, Intent intent) {
			String action = intent.getAction();
			Bundle b = intent.getExtras();
			Object[] lstName = b.keySet().toArray();
 
// 显示所有收到的消息及其细节
for (int i = 0; i < lstName.length; i++) {
				String keyName = lstName[i].toString();
Log.e(keyName, String.valueOf(b.get(keyName)));
}
//搜索设备时,取得设备的MAC地址
if (BluetoothDevice.ACTION_FOUND.equals(action)) {
				BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
				String str= device.getName() + "|" + device.getAddress();
// 防止重复添加
if (lstDevices.indexOf(str) == -1){
// 获取设备名称和mac地址
					lstDevices.add(str); 
}
				adtDevices.notifyDataSetChanged();
}
}
};
 
/*
	 * 退出程序
	 */
	protected void onDestroy() {
	    this.unregisterReceiver(searchDevices);
		super.onDestroy();
// 自毁进程
		android.os.Process.killProcess(android.os.Process.myPid());
}
 
/*
	 * 想其它蓝牙设备发送一图片文件
	 */
class ItemClickEvent implements AdapterView.OnItemClickListener {
public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
			btAdapt.cancelDiscovery();
			String str = lstDevices.get(arg2);
			String[] values = str.split("\\|");
			String address=values[1];
Log.e("address: ",values[1]);
 
			BluetoothDevice btDev = btAdapt.getRemoteDevice(address);
 
			try {
/*
				 * 网络流传的通过UUID连接的方式是不工作的,经过了好一阵搜索终于找到了下面的替换方法
				 */
//UUID uuid = UUID.fromString(SPP_UUID);
//btSocket = btDev.createRfcommSocketToServiceRecord(uuid);
 
				Method m = null;
				try {
					m = btDev.getClass().getMethod("createRfcommSocket", new Class[] {int.class});
} catch (SecurityException e) {
					e.printStackTrace();
} catch (NoSuchMethodException e) {
					e.printStackTrace();
}
				try {
					btSocket = (BluetoothSocket) m.invoke(btDev, 1);
} catch (IllegalArgumentException e) {
					e.printStackTrace();
} catch (IllegalAccessException e) {
					e.printStackTrace();
} catch (InvocationTargetException e) {
					e.printStackTrace();
}
 
				btSocket.connect();
 
/*
				 * 该段落是通过btsocket发送字符串给其它蓝牙设备,接收端应该需要做相应接收处理才能收到
				 * 所以尚未测试,暂且留着吧
				 */
//				OutputStream outStream = null;
//				outStream = btSocket.getOutputStream();
//                String message = "Hello message from client to server.";
//                byte[] msgBuffer = message.getBytes();
//                try {
//                        outStream.write(msgBuffer);
//                } catch (IOException e) {
//                	e.printStackTrace();
//                }
//                btSocket.close();
 
 
/*
				 * 发送一指定的文件到其它蓝牙设备
				 */
				ContentValues cv = new ContentValues();
				cv.put("uri", FilePath);
				cv.put("destination", address);
				cv.put("direction", 0);
				Long ts = System.currentTimeMillis();
				cv.put("timestamp", ts);
				getContentResolver().insert(Uri.parse("content://com.android.bluetooth.opp/btopp"), cv);
                btSocket.close();
 
/*
                 * 下面一段是通过intent的方式发送文件
                 * 与上面一段的不同在于,该方式会打开一个数据分享方式列表,如蓝牙,短信,Email 等
                 * 选择蓝牙方式后也是可以发送到其它蓝牙设备的
                 * 只不过偶尔也会抛出一个ioException异常,所以健壮性还有待加强,如添加try/catch模块
                 */
//				Intent intent = new Intent();
//				intent.setAction(Intent.ACTION_SEND);
//				intent.setType("image/jpg");
//				intent.putExtra(Intent.EXTRA_STREAM, Uri.fromFile(new File("/sdcard/test.jpg")) );
//				startActivity(intent);
 
 
} catch (IOException e) {
				e.printStackTrace();
}
}
}
 
 
class ClickEvent implements View.OnClickListener {
public void onClick(View v) {
// 搜索蓝牙设备,在BroadcastReceiver显示结果
if (v == btnSearch)
{
// 如果蓝牙还没开启,这提示开启
if (btAdapt.getState() == BluetoothAdapter.STATE_OFF) {
					Toast.makeText(testBlueTooth.this, "请先打开蓝牙", 1000).show();
return;
}
				setTitle("本机蓝牙地址:" + btAdapt.getAddress());
				lstDevices.clear();
				btAdapt.startDiscovery();
// 本机蓝牙启动/关闭
} else if (v == tbtnSwitch) {
if (tbtnSwitch.isChecked() == true)
					btAdapt.enable();
else if (tbtnSwitch.isChecked() == false)
					btAdapt.disable();
/*
			 * 设置本机蓝牙可检测性,即可被他人搜索到。
			 * 这里稍微说明一下:蓝牙设备一般都会支持“开启关闭”和“设置可见”两种行为:
			 *   开启关闭:当然是打开/关闭蓝牙设备了
			 *   设置可见:开启可检测性,以便其它蓝牙设备可以搜索到本机
			 * 本人使用M9测试的,M9的“开启关闭”和“设置可见”默认被绑定了,即开启后可见属性也启用了,
			 * 同样启用可见后(UI上不允许这么操作,通过下面的代码可以)蓝牙也开启了
			 * 
			 * 所以为了避免重复操作,下面做了些判断:
			 *   1. 如果是蓝牙开启,则不再需要启用可见
			 *   2. 如果蓝牙未开启,这启用可见后,将蓝牙开启的指示按钮设置为开启状态
			 *   
			 * 用于自己测试时应根据自己的手机情况稍做改动!
			 * 如有的手机在设置蓝牙时需要对“开启关闭”和“设置可见“两个选项分别单独操作,
			 * 这时只需要else里面部分就行了,即每次都执行设置可检测性操作。
			 */
} else if (v == btnDis)
{
if(tbtnSwitch.isChecked() == true){
					Toast.makeText(testBlueTooth.this, "蓝牙已经可见", 1000).show();
}
else{
					Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
					discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
					startActivity(discoverableIntent);
					tbtnSwitch.setChecked(true);
}
} else if (v == btnExit) {
				try {
if (btSocket != null)
						btSocket.close();
} catch (IOException e) {
					e.printStackTrace();
}
				testBlueTooth.this.finish();
}
}
}
}

里面都有详细的说明。

注意:用户测试时需更改一下发送文件的路径,及FilePath变量的值!