• Android蓝牙测试—发送一文件到另一蓝牙设备


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

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

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


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


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


    下面看代码:
    首先看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变量的值!

    参考链接:
    [1] stackoverflow.com/questions/3397071/andr...ery-failed-exception
    [2] stackoverflow.com/questions/4921384/how-...roid-programatically
    [3] android.tgbus.com/Android/tutorial/201103/346657.shtml
    [4] www.mikenimer.com/?p=373

  • 相关阅读:
    安全实践鬼手诀 杂志
    Android 应用资源随笔
    Android 构架
    Andorid杂笔 深入理解Activity,Intenthe IntentFilter
    Android杂笔 事件处理
    创建9-Patch自定义伸缩图片
    重拾C之语句,操作符和表达式
    CSS float属性
    最近最少使用队列算法
    java常见面试题
  • 原文地址:https://www.cnblogs.com/wzc0066/p/2948233.html
Copyright © 2020-2023  润新知