溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android中怎么實現(xiàn)藍(lán)牙設(shè)備檢測連接

發(fā)布時間:2021-07-12 11:22:29 來源:億速云 閱讀:245 作者:Leah 欄目:編程語言

Android中怎么實現(xiàn)藍(lán)牙設(shè)備檢測連接,針對這個問題,這篇文章詳細(xì)介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

Android為藍(lán)牙技術(shù)提供了4個工具類,分別是藍(lán)牙適配器BluetoothAdapter、藍(lán)牙設(shè)備BluetoothDevice、藍(lán)牙服務(wù)端套接字BluetoothServerSocket和藍(lán)牙客戶端套接字BluetoothSocket。

藍(lán)牙適配器BluetoothAdapter

BluetoothAdapter的作用其實跟其它的**Manger差不多,可以把它當(dāng)作藍(lán)牙管理器。下面是BluetoothAdapter的常用方法說明。

getDefaultAdapter:靜態(tài)方法,獲取默認(rèn)的藍(lán)牙適配器對象;enable:打開藍(lán)牙功能;disable:關(guān)閉藍(lán)牙功能;isEnable:判斷藍(lán)牙功能是否打開;startDiscovery:開始搜索周圍的藍(lán)牙設(shè)備;cancelDiscovery:取消搜索操作;isDiscovering:判斷當(dāng)前是否正在搜索設(shè)備;getBondedDevices:獲取已綁定的設(shè)備列表;setName:設(shè)置本機(jī)的藍(lán)牙名稱;getName:獲取本機(jī)的藍(lán)牙名稱;getAddress:獲取本機(jī)的藍(lán)牙地址;getRemoteDevice:根據(jù)藍(lán)牙地址獲取遠(yuǎn)程的藍(lán)牙設(shè)備;getState:獲取本地藍(lán)牙適配器的狀態(tài);listenUsingRfcommWithServiceRecord:根據(jù)名稱和UUID創(chuàng)建并返回BluetoothServiceSocket;listenUsingRfcommOn:根據(jù)渠道編號創(chuàng)建并返回BluetoothServiceSocket。

藍(lán)牙設(shè)備BluetoothDevice

BluetoothDevice用于指代某個藍(lán)牙設(shè)備,通常表示對方設(shè)備。BluetoothAdapter管理的是本機(jī)藍(lán)牙設(shè)備。下面是BluetoothDevice的常用方法說明。

getName:獲得該設(shè)備的名稱;  getAddress:獲得該設(shè)備的地址;  getBondState:獲得該設(shè)備的綁定狀態(tài);  createBond:創(chuàng)建匹配對象;  createRfcommSocketToServiceRecord:根據(jù)UUID創(chuàng)建并返回一個BluetoothSocket。

藍(lán)牙服務(wù)器套接字BluetoothServiceSocket

BluetoothServiceSocket是服務(wù)端的Socket,用來接收客戶端的Socket連接請求。下面是常用的方法說明。

accept:監(jiān)聽外部的藍(lán)牙連接請求;close:關(guān)閉服務(wù)端的藍(lán)牙監(jiān)聽。

藍(lán)牙客戶端套接字BluetoothSocket

BluetoothSocket是客戶端的Socket,用于與對方設(shè)備進(jìn)行數(shù)據(jù)通信。下面是常用的方法說明。

connect:建立藍(lán)牙的socket連接;  close:關(guān)閉藍(lán)牙的socket連接;  getInputStream:獲取socket連接的輸入流對象;  getOutputStream:獲取socket連接的輸出流對象;  getRemoteDevice:獲取遠(yuǎn)程設(shè)備信息。

layout\activity_bluetooth.xml界面布局代碼如下:界面布局代碼如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:orientation="vertical"  android:padding="5dp">  <LinearLayout    android:layout_width="match_parent"    android:layout_height="wrap_content"    android:orientation="horizontal">    <CheckBox      android:id="@+id/ck_bluetooth"      android:layout_width="wrap_content"      android:layout_height="wrap_content"      android:button="@null"      android:checked="false"      android:drawableLeft="@drawable/ck_status_selector"      android:text="藍(lán)牙"      android:textColor="#ff000000"      android:textSize="17sp" />    <TextView      android:id="@+id/tv_discovery"      android:layout_width="0dp"      android:layout_height="match_parent"      android:layout_weight="1"      android:gravity="right|center"      android:textColor="#ff000000"      android:textSize="17sp" />  </LinearLayout>  <LinearLayout    android:layout_width="match_parent"    android:layout_height="40dp"    android:orientation="horizontal">    <TextView      android:layout_width="0dp"      android:layout_height="match_parent"      android:layout_weight="4"      android:gravity="center"      android:text="名稱"      android:textColor="#ff000000"      android:textSize="17sp" />    <TextView      android:layout_width="0dp"      android:layout_height="match_parent"      android:layout_weight="5"      android:gravity="center"      android:text="地址"      android:textColor="#ff000000"      android:textSize="17sp" />    <TextView      android:layout_width="0dp"      android:layout_height="match_parent"      android:layout_weight="2"      android:gravity="center"      android:text="狀態(tài)"      android:textColor="#ff000000"      android:textSize="17sp" />  </LinearLayout>  <ListView    android:id="@+id/lv_bluetooth"    android:layout_width="match_parent"    android:layout_height="match_parent" /></LinearLayout>

BluetoothActivity.java邏輯代碼如下:

package com.fukaimei.bluetoothtest;import java.io.IOException;import java.lang.reflect.Method;import java.util.ArrayList;import android.app.AlertDialog;import android.bluetooth.BluetoothAdapter;import android.bluetooth.BluetoothDevice;import android.bluetooth.BluetoothSocket;import android.content.BroadcastReceiver;import android.content.Context;import android.content.Intent;import android.content.IntentFilter;import android.content.pm.PackageManager;import android.os.AsyncTask;import android.os.Bundle;import android.os.Handler;import android.os.Message;import android.support.annotation.NonNull;import android.support.v4.app.ActivityCompat;import android.support.v4.content.ContextCompat;import android.support.v7.app.AppCompatActivity;import android.util.Log;import android.view.View;import android.view.View.OnClickListener;import android.widget.AdapterView;import android.widget.AdapterView.OnItemClickListener;import android.widget.CheckBox;import android.widget.CompoundButton;import android.widget.ListView;import android.widget.CompoundButton.OnCheckedChangeListener;import android.widget.TextView;import android.widget.Toast;import com.fukaimei.bluetoothtest.adapter.BlueListAdapter;import com.fukaimei.bluetoothtest.bean.BlueDevice;import com.fukaimei.bluetoothtest.task.BlueAcceptTask;import com.fukaimei.bluetoothtest.task.BlueConnectTask;import com.fukaimei.bluetoothtest.task.BlueReceiveTask;import com.fukaimei.bluetoothtest.util.BluetoothUtil;import com.fukaimei.bluetoothtest.widget.InputDialogFragment;public class BluetoothActivity extends AppCompatActivity implements    OnClickListener, OnItemClickListener, OnCheckedChangeListener,    BlueConnectTask.BlueConnectListener, InputDialogFragment.InputCallbacks, BlueAcceptTask.BlueAcceptListener {  private static final String TAG = "BluetoothActivity";  private CheckBox ck_bluetooth;  private TextView tv_discovery;  private ListView lv_bluetooth;  private BluetoothAdapter mBluetooth;  private ArrayList<BlueDevice> mDeviceList = new ArrayList<BlueDevice>();  @Override  protected void onCreate(Bundle savedInstanceState) {    super.onCreate(savedInstanceState);    setContentView(R.layout.activity_bluetooth);    bluetoothPermissions();    ck_bluetooth = (CheckBox) findViewById(R.id.ck_bluetooth);    tv_discovery = (TextView) findViewById(R.id.tv_discovery);    lv_bluetooth = (ListView) findViewById(R.id.lv_bluetooth);    if (BluetoothUtil.getBlueToothStatus(this) == true) {      ck_bluetooth.setChecked(true);    }    ck_bluetooth.setOnCheckedChangeListener(this);    tv_discovery.setOnClickListener(this);    mBluetooth = BluetoothAdapter.getDefaultAdapter();    if (mBluetooth == null) {      Toast.makeText(this, "本機(jī)未找到藍(lán)牙功能", Toast.LENGTH_SHORT).show();      finish();    }  }  // 定義獲取基于地理位置的動態(tài)權(quán)限  private void bluetoothPermissions() {    if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_COARSE_LOCATION)        != PackageManager.PERMISSION_GRANTED) {      ActivityCompat.requestPermissions(this, new String[]{          android.Manifest.permission.ACCESS_COARSE_LOCATION}, 1);    }  }  /**   * 重寫onRequestPermissionsResult方法   * 獲取動態(tài)權(quán)限請求的結(jié)果,再開啟藍(lán)牙   */  @Override  public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {    if (requestCode == 1 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {      if (BluetoothUtil.getBlueToothStatus(this) == true) {        ck_bluetooth.setChecked(true);      }      ck_bluetooth.setOnCheckedChangeListener(this);      tv_discovery.setOnClickListener(this);      mBluetooth = BluetoothAdapter.getDefaultAdapter();      if (mBluetooth == null) {        Toast.makeText(this, "本機(jī)未找到藍(lán)牙功能", Toast.LENGTH_SHORT).show();        finish();      }    } else {      Toast.makeText(this, "用戶拒絕了權(quán)限", Toast.LENGTH_SHORT).show();    }    super.onRequestPermissionsResult(requestCode, permissions, grantResults);  }  @Override  public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {    if (buttonView.getId() == R.id.ck_bluetooth) {      if (isChecked == true) {        beginDiscovery();        Intent intent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);        startActivityForResult(intent, 1);        // 下面這行代碼為服務(wù)端需要,客戶端不需要        mHandler.postDelayed(mAccept, 1000);      } else {        cancelDiscovery();        BluetoothUtil.setBlueToothStatus(this, false);        mDeviceList.clear();        BlueListAdapter adapter = new BlueListAdapter(this, mDeviceList);        lv_bluetooth.setAdapter(adapter);      }    }  }  private Runnable mAccept = new Runnable() {    @Override    public void run() {      if (mBluetooth.getState() == BluetoothAdapter.STATE_ON) {        BlueAcceptTask acceptTask = new BlueAcceptTask(true);        acceptTask.setBlueAcceptListener(BluetoothActivity.this);        acceptTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);      } else {        mHandler.postDelayed(this, 1000);      }    }  };  @Override  public void onClick(View v) {    if (v.getId() == R.id.tv_discovery) {      beginDiscovery();    }  }  @Override  protected void onActivityResult(int requestCode, int resultCode, Intent intent) {    super.onActivityResult(requestCode, resultCode, intent);    if (requestCode == 1) {      if (resultCode == RESULT_OK) {        Toast.makeText(this, "允許本地藍(lán)牙被附近的其它藍(lán)牙設(shè)備發(fā)現(xiàn)", Toast.LENGTH_SHORT).show();      } else if (resultCode == RESULT_CANCELED) {        Toast.makeText(this, "不允許藍(lán)牙被附近的其它藍(lán)牙設(shè)備發(fā)現(xiàn)", Toast.LENGTH_SHORT).show();      }    }  }  private Runnable mRefresh = new Runnable() {    @Override    public void run() {      beginDiscovery();      mHandler.postDelayed(this, 2000);    }  };  private void beginDiscovery() {    if (mBluetooth.isDiscovering() != true) {      mDeviceList.clear();      BlueListAdapter adapter = new BlueListAdapter(BluetoothActivity.this, mDeviceList);      lv_bluetooth.setAdapter(adapter);      tv_discovery.setText("正在搜索藍(lán)牙設(shè)備");      mBluetooth.startDiscovery();    }  }  private void cancelDiscovery() {    mHandler.removeCallbacks(mRefresh);    tv_discovery.setText("取消搜索藍(lán)牙設(shè)備");    if (mBluetooth.isDiscovering() == true) {      mBluetooth.cancelDiscovery();    }  }  @Override  protected void onStart() {    super.onStart();    mHandler.postDelayed(mRefresh, 50);    blueReceiver = new BluetoothReceiver();    //需要過濾多個動作,則調(diào)用IntentFilter對象的addAction添加新動作    IntentFilter foundFilter = new IntentFilter(BluetoothDevice.ACTION_FOUND);    foundFilter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);    foundFilter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);    registerReceiver(blueReceiver, foundFilter);  }  @Override  protected void onStop() {    super.onStop();    cancelDiscovery();    unregisterReceiver(blueReceiver);  }  private BluetoothReceiver blueReceiver;  private class BluetoothReceiver extends BroadcastReceiver {    @Override    public void onReceive(Context context, Intent intent) {      String action = intent.getAction();      Log.d(TAG, "onReceive action=" + action);      // 獲得已經(jīng)搜索到的藍(lán)牙設(shè)備      if (action.equals(BluetoothDevice.ACTION_FOUND)) {        BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);        BlueDevice item = new BlueDevice(device.getName(), device.getAddress(), device.getBondState() - 10);        mDeviceList.add(item);        BlueListAdapter adapter = new BlueListAdapter(BluetoothActivity.this, mDeviceList);        lv_bluetooth.setAdapter(adapter);        lv_bluetooth.setOnItemClickListener(BluetoothActivity.this);      } else if (action.equals(BluetoothAdapter.ACTION_DISCOVERY_FINISHED)) {        mHandler.removeCallbacks(mRefresh);        tv_discovery.setText("藍(lán)牙設(shè)備搜索完成");      } else if (action.equals(BluetoothDevice.ACTION_BOND_STATE_CHANGED)) {        BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);        if (device.getBondState() == BluetoothDevice.BOND_BONDING) {          tv_discovery.setText("正在配對" + device.getName());        } else if (device.getBondState() == BluetoothDevice.BOND_BONDED) {          tv_discovery.setText("完成配對" + device.getName());          mHandler.postDelayed(mRefresh, 50);        } else if (device.getBondState() == BluetoothDevice.BOND_NONE) {          tv_discovery.setText("取消配對" + device.getName());        }      }    }  }  @Override  public void onItemClick(AdapterView<?> parent, View view, int position, long id) {    cancelDiscovery();    BlueDevice item = mDeviceList.get(position);    BluetoothDevice device = mBluetooth.getRemoteDevice(item.address);    try {      if (device.getBondState() == BluetoothDevice.BOND_NONE) {        Method createBondMethod = BluetoothDevice.class.getMethod("createBond");        Log.d(TAG, "開始配對");        Boolean result = (Boolean) createBondMethod.invoke(device);      } else if (device.getBondState() == BluetoothDevice.BOND_BONDED &&          item.state != BlueListAdapter.CONNECTED) {        tv_discovery.setText("開始連接");        BlueConnectTask connectTask = new BlueConnectTask(item.address);        connectTask.setBlueConnectListener(this);        connectTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, device);      } else if (device.getBondState() == BluetoothDevice.BOND_BONDED &&          item.state == BlueListAdapter.CONNECTED) {        tv_discovery.setText("正在發(fā)送消息");        InputDialogFragment dialog = InputDialogFragment.newInstance(            "", 0, "請輸入要發(fā)送的消息");        String fragTag = getResources().getString(R.string.app_name);        dialog.show(getFragmentManager(), fragTag);      }    } catch (Exception e) {      e.printStackTrace();      tv_discovery.setText("配對異常:" + e.getMessage());    }  }  //向?qū)Ψ桨l(fā)送消息  @Override  public void onInput(String title, String message, int type) {    Log.d(TAG, "onInput message=" + message);    Log.d(TAG, "mBlueSocket is " + (mBlueSocket == null ? "null" : "not null"));    BluetoothUtil.writeOutputStream(mBlueSocket, message);  }  private BluetoothSocket mBlueSocket;  //客戶端主動連接  @Override  public void onBlueConnect(String address, BluetoothSocket socket) {    mBlueSocket = socket;    tv_discovery.setText("連接成功");    refreshAddress(address);  }  //刷新已連接的狀態(tài)  private void refreshAddress(String address) {    for (int i = 0; i < mDeviceList.size(); i++) {      BlueDevice item = mDeviceList.get(i);      if (item.address.equals(address) == true) {        item.state = BlueListAdapter.CONNECTED;        mDeviceList.set(i, item);      }    }    BlueListAdapter adapter = new BlueListAdapter(this, mDeviceList);    lv_bluetooth.setAdapter(adapter);  }  //服務(wù)端偵聽到連接  @Override  public void onBlueAccept(BluetoothSocket socket) {    Log.d(TAG, "onBlueAccept socket is " + (socket == null ? "null" : "not null"));    if (socket != null) {      mBlueSocket = socket;      BluetoothDevice device = mBlueSocket.getRemoteDevice();      refreshAddress(device.getAddress());      BlueReceiveTask receive = new BlueReceiveTask(mBlueSocket, mHandler);      receive.start();    }  }  //收到對方發(fā)來的消息  private Handler mHandler = new Handler() {    @Override    public void handleMessage(Message msg) {      if (msg.what == 0) {        byte[] readBuf = (byte[]) msg.obj;        String readMessage = new String(readBuf, 0, msg.arg1);        Log.d(TAG, "handleMessage readMessage=" + readMessage);        AlertDialog.Builder builder = new AlertDialog.Builder(BluetoothActivity.this);        builder.setTitle("我收到消息啦").setMessage(readMessage).setPositiveButton("確定", null);        builder.create().show();      }    }  };  @Override  protected void onDestroy() {    super.onDestroy();    if (mBlueSocket != null) {      try {        mBlueSocket.close();      } catch (IOException e) {        e.printStackTrace();      }    }  }}

添加藍(lán)牙所需的相應(yīng)權(quán)限:

<!-- 藍(lán)牙 -->  <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />  <uses-permission android:name="android.permission.BLUETOOTH" />  <!--基于地理位置-->  <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

關(guān)于Android中怎么實現(xiàn)藍(lán)牙設(shè)備檢測連接問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI