以编程方式连接到蓝牙

Posted

技术标签:

【中文标题】以编程方式连接到蓝牙【英文标题】:Connect to Bluetooth programmatically 【发布时间】:2015-08-01 11:41:26 【问题描述】:

我正在尝试以编程方式将我的设备连接到例如我的耳机...我有 KitKat 版本并且一切正常(Bluetooth 总是自动连接没有问题)但是因为我已经更新到 @ 987654325@ 没有。我想知道是否有任何方法可以将我的android phone 的任何配对设备连接到Bluetooth

从现在开始,我有了这段代码(获取设备名称和设备地址),因为我认为使用它可以连接像 device.connect(MAC-Address); 这样的东西,但它不起作用......

    BluetoothAdapter bluetoothAdapter
    = BluetoothAdapter.getDefaultAdapter();
Set < BluetoothDevice > pairedDevices = bluetoothAdapter.getBondedDevices();
if (pairedDevices.size() > 0) 
    for (BluetoothDevice device: pairedDevices) 
        mDeviceName.add(device.getName());
        mDeviceMAC.add(device.getAddress());

    

bluetoothClass.setDeviceName(mDeviceName);
bluetoothClass.setDeviceMac(mDeviceMAC);

问题

在我的 MotoG (KitKat) 上,如果我打开我的 Bluetooth,它会自动连接到设备(如果它在附近并已配对...)但在我的 LG G3 上,我必须转到配置/蓝牙/配对设备/那里点击设备进行连接...我想避免这种情况...应该可以吗?


我想知道是否有可能连接到特定的蓝牙,只需添加Device nameDevice MAC... 或多或少就像Android 当我点击我的设备进行连接时它会自动连接...我只想得到那个 CLICK 事件。 我知道Android 应该自动连接到已配对的设备,但有任何例外情况不会...配对它的唯一方法是点击...这就是为什么我想知道是否有办法做到这一点... 我已经阅读并测试了kcoppock answer,但它仍然不起作用..

有什么建议吗?

编辑

我想做的主要事情是自动连接我的Bluetooth,但因为我已经阅读了Hey you 的答案......我想通了,我知道这是一个Android 错误,所以事情我想做的是选择paired devices,然后点击我要连接的设备(不做任何Intent并连接它,而不是去Configuration/Bluetooth/...。 顺便说一句,我已经阅读了*** 上的任何答案,我发现Sockets 的一些内容是他们用来连接Bluetooth 的吗?这可能是一个解决方案吗?

【问题讨论】:

@Heyyou 是的,对不起...我编辑了问题 当我说我升级时,我的意思是当我使用另一个设备时,女巫 > 5.0,所以即使我有 BT,连接也需要很长时间,而且有时无法连接仅当我打开已配对的 BT 列表设备并单击我要加入的当前 BT 设备时才连接,你知道吗? 欢迎回答我会测试它并让你知道它是否有效:) 我的 Lg g3 有 5.0 棒棒糖 【参考方案1】:

编辑以回答最新问题

您可以避免使用意图搜索配对设备。当连接到未配对的设备时,会弹出一条通知,要求配对设备。配对后,这些设备不会再显示此消息,连接应该是自动的(根据您编写程序的方式)。

我使用 Intent 来启用蓝牙并让我的设备可被发现,然后我设置我的代码以进行连接,然后按下按钮进行连接。在您的情况下,您需要确保您的配件也可以被发现。在我的情况下,我使用唯一的 UUID,并且两个设备都必须识别它才能连接。仅当您对两种设备都进行编程时才能使用此功能,无论两者都是 android 还是一种 android 和另一种设备类型。

试试这个,看看能不能解决你的问题。


这个答案是对原始问题的回答,然后才被编辑为另一个问题。

为了清楚起见,我已经编辑了我的答案,正如我从 cmets 中看到的那样,它具有误导性。你的问题有两个部分。

在我的 MotoG (KitKat) 上,如果我打开蓝牙,它会自动连接 到设备(如果它在附近并已配对...)但在我的 LG G3 上我必须去 到配置/蓝牙/配对设备/,然后点击设备以 连接...我想避免这种情况...应该可以吗?

这不是编程问题,而是平台问题。Android 5.0 中存在一个有据可查的错误,蓝牙无法自动连接以及许多其他 BT 问题。这些问题在 5.0 上的所有更新中继续存在。版本,直到 5.1 才修复。升级。

http://www.digitaltrends.com/mobile/android-lollipop-problems/11/

http://forums.androidcentral.com/lg-g3/473064-bluetooth-streaming-choppy-lg-3-lollipop.html

首先要更新到 5.1

这些问题已在 Lollipop 更新 5.1 中得到解决

http://www.reddit.com/r/Android/comments/306m3y/lollipop_51_bluetooth/


编辑: 我不相信这会解决你的自动配对问题,你想知道如何使用 BTGatt。

如果我输入设备,我已经看到了。检查我能做什么让我 connectGatt() 表示 /.../ 但我无法弄清楚如何做到这一点......

使用 BluetoothGatt

https://developer.android.com/reference/android/bluetooth/BluetoothGatt.html

此类提供蓝牙 GATT 功能以启用 与蓝牙智能或智能就绪设备通信。 /.../ 可以使用蓝牙设备发现支持 GATT 的设备 发现或 BLE 扫描过程。

https://developer.android.com/reference/android/bluetooth/BluetoothGattCallback.html

这是一个很好的例子,说明如何使用 BluetoothGatt(它使用听力速率):https://github.com/googlesamples/android-BluetoothLeGatt/blob/master/Application/src/main/java/com/example/android/bluetoothlegatt/BluetoothLeService.java

我在这里复制了一些代码,以防链接失效。

它基本上遵循与常规蓝牙连接类似的线路。您需要发现并找到支持的设备。 监控状态等 这是与 gatt 最相关的两个特性。

回调:

// Implements callback methods for GATT events that the app cares about.  For example,
// connection change and services discovered.
private final BluetoothGattCallback mGattCallback = new BluetoothGattCallback() 
    @Override
    public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState) 
        String intentAction;
        if (newState == BluetoothProfile.STATE_CONNECTED) 
            intentAction = ACTION_GATT_CONNECTED;
            mConnectionState = STATE_CONNECTED;
            broadcastUpdate(intentAction);
            Log.i(TAG, "Connected to GATT server.");
            // Attempts to discover services after successful connection.
            Log.i(TAG, "Attempting to start service discovery:" +
                    mBluetoothGatt.discoverServices());

         else if (newState == BluetoothProfile.STATE_DISCONNECTED) 
            intentAction = ACTION_GATT_DISCONNECTED;
            mConnectionState = STATE_DISCONNECTED;
            Log.i(TAG, "Disconnected from GATT server.");
            broadcastUpdate(intentAction);
        
    

    @Override
    public void onServicesDiscovered(BluetoothGatt gatt, int status) 
        if (status == BluetoothGatt.GATT_SUCCESS) 
            broadcastUpdate(ACTION_GATT_SERVICES_DISCOVERED);
         else 
            Log.w(TAG, "onServicesDiscovered received: " + status);
        
    

    @Override
    public void onCharacteristicRead(BluetoothGatt gatt,
                                     BluetoothGattCharacteristic characteristic,
                                     int status) 
        if (status == BluetoothGatt.GATT_SUCCESS) 
            broadcastUpdate(ACTION_DATA_AVAILABLE, characteristic);
        
    

    @Override
    public void onCharacteristicChanged(BluetoothGatt gatt,
                                        BluetoothGattCharacteristic characteristic) 
        broadcastUpdate(ACTION_DATA_AVAILABLE, characteristic);
    
;

广播:

private void broadcastUpdate(final String action,
                             final BluetoothGattCharacteristic characteristic) 
    final Intent intent = new Intent(action);

    // This is special handling for the Heart Rate Measurement profile.  Data parsing is
    // carried out as per profile specifications:
    // http://developer.bluetooth.org/gatt/characteristics/Pages/CharacteristicViewer.aspx?u=org.bluetooth.characteristic.heart_rate_measurement.xml
    if (UUID_HEART_RATE_MEASUREMENT.equals(characteristic.getUuid())) 
        int flag = characteristic.getProperties();
        int format = -1;
        if ((flag & 0x01) != 0) 
            format = BluetoothGattCharacteristic.FORMAT_UINT16;
            Log.d(TAG, "Heart rate format UINT16.");
         else 
            format = BluetoothGattCharacteristic.FORMAT_UINT8;
            Log.d(TAG, "Heart rate format UINT8.");
        
        final int heartRate = characteristic.getIntValue(format, 1);
        Log.d(TAG, String.format("Received heart rate: %d", heartRate));
        intent.putExtra(EXTRA_DATA, String.valueOf(heartRate));
     else 
        // For all other profiles, writes the data formatted in HEX.
        final byte[] data = characteristic.getValue();
        if (data != null && data.length > 0) 
            final StringBuilder stringBuilder = new StringBuilder(data.length);
            for(byte byteChar : data)
                stringBuilder.append(String.format("%02X ", byteChar));
            intent.putExtra(EXTRA_DATA, new String(data) + "\n" + stringBuilder.toString());
        
    
    sendBroadcast(intent);

这个问题也有一些相关代码,学习的时候可能会有所帮助:BLuetooth Gatt Callback not working with new API for Lollipop

现在问题来了。您的设备是否支持智能蓝牙或智能设备?

此链接提供了一个很棒的智能设备列表。当你实施你的程序时,你也会发现。

http://www.bluetooth.com/Pages/Bluetooth-Smart-Devices-List.aspx

【讨论】:

我明天会测试它,因为现在我没有Bluetooth device 也没有LG g3 所以明天我会尽快让你知道:) 谢谢你的回答。顺便说一句BluetoothGatt 是用来连接配对设备的? 嗯,我会阅读有关“智能设备”的信息,看看它是否可以工作:P 我现在要测试它是否有效,有没有其他方法可以在不使用BluetoothCatt的情况下连接BT? 我必须在服务上创建回调吗? 好吧,让我发布另一个问题,我会在它准备好时通知你:) 我会更好地解释我的新问题..【参考方案2】:

这就是我使用 Java Reflection 和 BluetoothProfile 完成这项工作的方式:

属性:

private boolean mIsConnect = true;
private BluetoothDevice mDevice;
private BluetoothA2dp mBluetoothA2DP;
private BluetoothHeadset mBluetoothHeadset;
private BluetoothHealth mBluetoothHealth;

呼叫:

mBluetoothAdapter.getProfileProxy(getApplicationContext() , mProfileListener, BluetoothProfile.A2DP);
mBluetoothAdapter.getProfileProxy(getApplicationContext() , mProfileListener, BluetoothProfile.HEADSET);
mBluetoothAdapter.getProfileProxy(getApplicationContext() , mProfileListener, BluetoothProfile.HEALTH);

听众:

private BluetoothProfile.ServiceListener mProfileListener = new BluetoothProfile.ServiceListener() 
        public void onServiceConnected(int profile, BluetoothProfile proxy) 
            if (profile == BluetoothProfile.A2DP) 
                mBluetoothA2DP = (BluetoothA2dp) proxy;
                try 
                    if (mIsConnect) 
                        Method connect = BluetoothA2dp.class.getDeclaredMethod("connect", BluetoothDevice.class);
                        connect.setAccessible(true);
                        connect.invoke(mBluetoothA2DP, mDevice);
                     else 
                        Method disconnect = BluetoothA2dp.class.getDeclaredMethod("disconnect", BluetoothDevice.class);
                        disconnect.setAccessible(true);
                        disconnect.invoke(mBluetoothA2DP, mDevice);
                    
                catch (Exception e)
                 finally 
                
             else if (profile == BluetoothProfile.HEADSET) 
                mBluetoothHeadset = (BluetoothHeadset) proxy;
                try 
                    if (mIsConnect) 
                        Method connect = BluetoothHeadset.class.getDeclaredMethod("connect", BluetoothDevice.class);
                        connect.setAccessible(true);
                        connect.invoke(mBluetoothHeadset, mDevice);
                     else 
                        Method disconnect = BluetoothHeadset.class.getDeclaredMethod("disconnect", BluetoothDevice.class);
                        disconnect.setAccessible(true);
                        disconnect.invoke(mBluetoothHeadset, mDevice);
                    
                catch (Exception e)
                 finally 
                
             else if (profile == BluetoothProfile.HEALTH) 
                mBluetoothHealth = (BluetoothHealth) proxy;
                try 
                    if (mIsConnect) 
                        Method connect = BluetoothHealth.class.getDeclaredMethod("connect", BluetoothDevice.class);
                        connect.setAccessible(true);
                        connect.invoke(mBluetoothHealth, mDevice);
                     else 
                        Method disconnect = BluetoothHealth.class.getDeclaredMethod("disconnect", BluetoothDevice.class);
                        disconnect.setAccessible(true);
                        disconnect.invoke(mBluetoothHealth, mDevice);
                    
                catch (Exception e)
                 finally 
                
            
        
        public void onServiceDisconnected(int profile) 
        
    ;

我希望这对尝试连接蓝牙音频设备和耳机的人有所帮助。

【讨论】:

有时间我会看看谢谢你的回答 hexan :) 如果我需要更多代码来让它工作,我会告诉你:D 没问题! :) 你应该只需要设置 mDevice 和 mBluetoothAdapter (BluetoothAdapter mBluetoothAdapter)。如果您需要更多帮助,请不要犹豫。 非常好。谢谢。你知道我们在哪里可以读到为什么你需要“getDeclaredMethod”而不是直接调用方法吗?

以上是关于以编程方式连接到蓝牙的主要内容,如果未能解决你的问题,请参考以下文章

是否可以使用 CBPeripheral 通过蓝牙从 iOS 连接到 macOS

可以连接到 iPhone/iPad 的蓝牙功能的照片相机

Bluez 编程

C# 以编程方式连接到 SQL 服务器 CE

以编程方式更改控制面板网络选项以连接到 *** [重复]

如何以编程方式连接到 ***?