首先我们来回顾一下Classic Bluetooth Mode(传统蓝牙模式):
是 Android 操作系统中最常见的蓝牙模式之一。它支持传统的蓝牙设备连接和数据传输,包括蓝牙耳机、蓝牙键盘、蓝牙音箱、蓝牙打印机等各种蓝牙设备。
在 Classic Bluetooth Mode 中,Android 设备可以同时扮演蓝牙主设备(Master)和从设备(Slave)的角色。作为蓝牙主设备,Android 设备可以搜索其他蓝牙设备并建立连接。作为从设备,Android 设备可以接受其他蓝牙设备的连接请求并进行通信。
Classic Bluetooth Mode 支持以下功能和协议:
-
数据传输:通过 Classic Bluetooth Mode,可以在 Android 设备和其他蓝牙设备之间传输数据。这包括文件传输、音频传输、图像传输等。例如,你可以使用 Classic Bluetooth Mode 在 Android 手机和蓝牙耳机之间进行音频传输。
-
音频通信:Classic Bluetooth Mode 支持蓝牙耳机、蓝牙音箱等音频设备的连接和通信。你可以将 Android 设备与蓝牙音箱配对,以便通过蓝牙播放音乐或接听电话。
-
串口通信:Classic Bluetooth Mode 支持使用串口配置文件(Serial Port Profile,SPP)进行串口通信。这使得 Android 设备可以与其他支持串口通信的设备进行连接和数据交换。
-
遥控器和输入设备:Classic Bluetooth Mode 支持蓝牙遥控器、蓝牙键盘、蓝牙鼠标等输入设备的连接和使用。你可以将蓝牙键盘连接到 Android 设备上,以便在键盘上输入文本或控制应用程序。
-
多点连接:Classic Bluetooth Mode 允许 Android 设备同时与多个蓝牙设备建立连接。这意味着你可以同时连接多个蓝牙设备,如耳机和音箱,并在它们之间进行切换或同时进行音频传输。
Classic Bluetooth Mode 提供了相应的 API 和框架供开发者使用,以实现各种蓝牙功能。开发者可以使用 Android 提供的蓝牙 API 进行设备发现、连接管理、数据传输等操作,以满足特定应用的需求。
总结来说,Classic Bluetooth Mode 是 Android 中用于传统蓝牙设备连接和通信的模式,它支持各种功能和协议,包括数据传输、音频通信、串口通信、遥控器和输入设备等。
接下来我们要用Classic Bluetooth Mode(传统蓝牙模式)实现一个简单蓝牙聊天示例:
首先,在 AndroidManifest.xml 文件中添加以下权限:
<uses-permission android:name="android.permission.BLUETOOTH"/>
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
首先,你需要创建一个用于扫描和显示附近蓝牙设备的设备列表界面。这个界面应该包括一个列表视图用于显示可用设备,并提供扫描按钮以扫描附近的设备。当用户选择一个设备时,你需要建立与该设备的蓝牙连接。
接下来,你需要创建一个聊天界面,用于显示收发的消息。这个界面应该包括一个消息列表视图和一个用于输入消息的文本框,以及一个发送按钮。当用户输入消息并点击发送按钮时,你需要将消息通过蓝牙发送给已连接的设备,并将收到的消息显示在消息列表中。
下面是一个简化的示例代码,用于演示这个蓝牙聊天应用的框架:
// 设备列表界面
public class DeviceListActivity extends AppCompatActivity {
// TODO: 实现设备扫描和连接逻辑
}
// 聊天界面
public class ChatActivity extends AppCompatActivity {
// TODO: 实现消息收发和蓝牙连接逻辑
}
DeviceListActivity
设备列表界面的代码:
import android.Manifest;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Bundle;
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 androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;
import java.util.ArrayList;
import java.util.Set;
public class DeviceListActivity extends AppCompatActivity {
private static final int REQUEST_ENABLE_BT = 1;
private static final int REQUEST_PERMISSION_LOCATION = 2;
private BluetoothAdapter mBluetoothAdapter;
private ArrayAdapter<String> mDeviceListAdapter;
private ListView mDeviceListView;
private Button mScanButton;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_device_list);
mDeviceListView = findViewById(R.id.device_list_view);
mScanButton = findViewById(R.id.scan_button);
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
if (mBluetoothAdapter == null) {
Toast.makeText(this, "Bluetooth is not supported on this device", Toast.LENGTH_SHORT).show();
finish();
return;
}
mDeviceListAdapter = new ArrayAdapter<>(this, android.R.layout.simple_list_item_1);
mDeviceListView.setAdapter(mDeviceListAdapter);
mDeviceListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
String deviceAddress = mDeviceListAdapter.getItem(position);
connectToDevice(deviceAddress);
}
});
mScanButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startBluetoothScan();
}
});
checkBluetoothPermissions();
}
private void checkBluetoothPermissions() {
if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) {
enableBluetooth();
} else {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_PERMISSION_LOCATION);
}
}
private void enableBluetooth() {
if (!mBluetoothAdapter.isEnabled()) {
Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
startActivityForResult(enableIntent, REQUEST_ENABLE_BT);
} else {
startBluetoothScan();
}
}
private void startBluetoothScan() {
mDeviceListAdapter.clear();
Set<BluetoothDevice> bondedDevices = mBluetoothAdapter.getBondedDevices();
for (BluetoothDevice device : bondedDevices) {
mDeviceListAdapter.add(device.getAddress());
}
if (mBluetoothAdapter.isDiscovering()) {
mBluetoothAdapter.cancelDiscovery();
}
mBluetoothAdapter.startDiscovery();
}
private void connectToDevice(String deviceAddress) {
// TODO: 实现蓝牙连接逻辑
try {
mSocket = mDevice.createRfcommSocketToServiceRecord(MY_UUID);
mSocket.connect(); mStatusText.setText("Connected");
mInputStream = mSocket.getInputStream();
mOutputStream = mSocket.getOutputStream();
} catch (IOException e) {
Log.e(TAG, "Failed to connect to device: " + e.getMessage()); mStatusText.setText("Connection failed"); }
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
if (requestCode == REQUEST_PERMISSION_LOCATION) {
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
enableBluetooth();
} else {
Toast.makeText(this, "Location permission is required for Bluetooth scanning", Toast.LENGTH_SHORT).show();
finish();
}
}
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == REQUEST_ENABLE_BT) {
if (resultCode == RESULT_OK) {
startBluetoothScan();
} else {
Toast.makeText(this, "Bluetooth must be enabled to use this app", Toast.LENGTH_SHORT).show();
finish();
}
}
}
}
activity_device_list.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".DeviceListActivity">
<Button
android:id="@+id/scan_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:layout_centerHorizontal="true"
android:layout_marginTop="16dp"
android:text="Scan" />
<ListView
android:id="@+id/device_list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_below="@id/scan_button" />
</RelativeLayout>
ChatActivity
聊天界面的代码:
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothSocket;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ListView;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.UUID;
public class ChatActivity extends AppCompatActivity {
private static final String TAG = "BluetoothChat";
private static final UUID MY_UUID = UUID.fromString("00001101-0000-1000-8000-00805F9B34FB"); // SPP UUID
private BluetoothDevice mDevice;
private BluetoothSocket mSocket;
private InputStream mInputStream;
private OutputStream mOutputStream;
private ListView mMessageListView;
private EditText mMessageEditText;
private Button mSendButton;
private ArrayList<String> mMessageList;
private MessageAdapter mMessageAdapter;
private Handler mHandler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_chat);
mMessageListView = findViewById(R.id.message_list_view);
mMessageEditText = findViewById(R.id.message_edit_text);
mSendButton = findViewById(R.id.send_button);
mMessageList = new ArrayList<>();
mMessageAdapter = new MessageAdapter(this, mMessageList);
mMessageListView.setAdapter(mMessageAdapter);
mDevice = getIntent().getParcelableExtra("device");
mHandler = new Handler();
mSendButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String message = mMessageEditText.getText().toString();
sendMessage(message);
}
});
connectToDevice();
}
private void connectToDevice() {
try {
BluetoothSocket socket = mDevice.createRfcommSocketToServiceRecord(MY_UUID);
socket.connect();
mSocket = socket;
mInputStream = mSocket.getInputStream();
mOutputStream = mSocket.getOutputStream();
startReceivingData();
} catch (IOException e) {
Log.e(TAG, "Failed to connect to device: " + e.getMessage());
Toast.makeText(this, "Connection failed", Toast.LENGTH_SHORT).show();
finish();
}
}
private void startReceivingData() {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
byte[] buffer = new byte[1024];
int bytes;
while (true) {
try {
bytes = mInputStream.read(buffer);
String receivedMessage = new String(buffer, 0, bytes);
mHandler.post(new Runnable() {
@Override
public void run() {
mMessageList.add(receivedMessage);
mMessageAdapter.notifyDataSetChanged();
}
});
} catch (IOException e) {
Log.e(TAG, "Failed to read data: " + e.getMessage());
break;
}
}
}
});
thread.start();
}
private void sendMessage(String message) {
try {
mOutputStream.write(message.getBytes());
mMessageList.add("Me: " + message);
mMessageAdapter.notifyDataSetChanged();
mMessageEditText.getText().clear();
} catch (IOException e) {
Log.e(TAG, "Failed to send data: " + e.getMessage());
Toast.makeText(this, "Failed to send message", Toast.LENGTH_SHORT).show();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
disconnect();
}
private void disconnect() {
try {
if (mInputStream != null) {
mInputStream.close();
}
if (mOutputStream != null) {
mOutputStream.close();
}
if (mSocket != null) {
mSocket.close();
}
} catch (IOException e) {
Log.e(TAG, "Error closing Bluetooth connection: " + e.getMessage());
}
}
}
activity_chat.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".ChatActivity">
<ListView
android:id="@+id/message_list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_above="@+id/message_layout"
android:divider="@android:color/darker_gray"
android:dividerHeight="1dp"
android:padding="8dp" />
<LinearLayout
android:id="@+id/message_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:orientation="horizontal"
android:padding="8dp">
<EditText
android:id="@+id/message_edit_text"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:hint="Enter message" />
<Button
android:id="@+id/send_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Send" />
</LinearLayout>
</RelativeLayout>
在这个示例演示了在 Android 中使用 Classic Bluetooth Mode(经典蓝牙模式)实现简单的蓝牙聊天应用。
在聊天界面,应用会与选择的设备建立蓝牙连接。通过 BluetoothSocket 进行数据传输,实现了消息的发送和接收功能。接收到的消息会显示在聊天界面的消息列表中,同时用户可以在输入框中输入消息并发送。
这个示例提供了一个基本的蓝牙聊天框架,可以作为开发更复杂蓝牙应用的起点。它展示了蓝牙设备的搜索、配对、连接和数据传输的流程。但需要注意的是,示例中并未处理连接错误和异常情况,也没有进行错误处理和优化。
在实际开发中,可以根据需求扩展这个示例,例如添加断开连接、错误处理、优化连接稳定性等功能。此外,还可以考虑增加加密、身份验证等安全性方面的功能。
希望能够对你有所帮助!