- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我英语不好。敬请谅解。
我想知道蓝牙设置何时完成。我想在蓝牙连接和设置完成后将数据发送到我的设备。我如何知道流打开何时完成?
当我尝试所有 BLE 设置时,我只是睡了几秒钟,但我想将其更改为顺序代码。
private class GattClientCallback extends BluetoothGattCallback {
@Override
public void onServicesDiscovered( BluetoothGatt _gatt, int _status ) {
super.onServicesDiscovered( _gatt, _status );
// check if the discovery failed
if( _status != BluetoothGatt.GATT_SUCCESS ) {
Log.e( TAG, "Device service discovery failed, status: " + _status );
return;
}
// find discovered characteristics
List<BluetoothGattCharacteristic> matching_characteristics= BluetoothUtils.findBLECharacteristics(_gatt);
if( matching_characteristics.isEmpty() ) {
Log.e( TAG, "Unable to find characteristics" );
return;
}
// log for successful discovery
Log.d( TAG, "Services discovery is successful" );
// find command characteristics from the GATT server
cmd_characteristic= BluetoothUtils.findCommandCharacteristic( ble_gatt_ );
//setCharacteristicNotification
ble_gatt_.setCharacteristicNotification(cmd_characteristic, true);
BluetoothGattDescriptor descriptor = cmd_characteristic.getDescriptor(CHARACTERISTIC_UPDATE_NOTIFICATION_DESCRIPTOR_UUID);
descriptor.setValue(BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE);
ble_gatt_.writeDescriptor(descriptor); //descriptor write operation successfully started?
// stream open complete.
// update the connection status message
msg_handler(SET_TV_STATUS,"connection complete");
}
}
String result = pref.getString("first_device", "0");
if(result.equals("0")) {
ble_info.startScan();
} else {
ble_info.connectSavedDevice();
}
packet_sleep(5000); //I have to send data after ble connecting finished.
if(!ble_info.connected_) {
return;
}
while(ble_info.connected_ && state_check != ACK_COMM_START) {
ble_info.sendData(SET_COMM_START);
packet_sleep(ACK_TERM);
}
ble_info 类
public class BluetoothInfo {
protected static final int SET_TV_STATUS = 0;
protected static final int SET_TV_CONNECTION = 1;
protected static final UUID CHARACTERISTIC_UPDATE_NOTIFICATION_DESCRIPTOR_UUID = UUID.fromString("00002902-0000-1000-8000-00805f9b34fb");
protected static PacketReceive packetReceive = new PacketReceive();
//-------------------------------
//mac address
public String devAddress;
// ble adapter
public BluetoothAdapter ble_adapter_;
// flag for scanning
public boolean is_scanning_= false;
// flag for connection
public boolean connected_= false;
// scan results
public Map<String, BluetoothDevice> scan_results_;
// scan callback
public ScanCallback scan_cb_;
// ble scanner
public BluetoothLeScanner ble_scanner_;
// scan handler
public Handler scan_handler_;
// BLE Gatt
public BluetoothGatt ble_gatt_;
BluetoothGattCharacteristic cmd_characteristic;
// to save data
SharedPreferences pref;
SharedPreferences.Editor editor;
//constructor
BluetoothInfo(BluetoothManager ble_manager, SharedPreferences pref) {
ble_adapter_= ble_manager.getAdapter();
ble_scanner_ = ble_adapter_.getBluetoothLeScanner();
//to save data
this.pref = pref;
}
@RequiresApi(api = Build.VERSION_CODES.M)
public void startScan() {
// check if location permission
if (context.checkSelfPermission(Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
requestLocationPermission();
Log.d(TAG, "Scanning Failed: no fine location permission");
return;
}
// disconnect gatt server
disconnectGattServer();
scan_handler_= new Handler();
scan_handler_.postDelayed( this::stopScan, SCAN_PERIOD );
scan_results_= new HashMap<>();
scan_cb_= new BLEScanCallback( scan_results_ );
Log.d(TAG,"Scanning...");
// check ble adapter and ble enabled
if (ble_adapter_ == null || !ble_adapter_.isEnabled()) {
requestEnableBLE();
Log.d(TAG, "Scanning Failed: ble not enabled");
return;
}
// now ready to scan
ble_scanner_.startScan(scan_cb_); //scan filter 사용x
// set scanning flag
is_scanning_= true;
}
/*
Stop scanning
*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
private void stopScan() {
// check pre-conditions
if( is_scanning_ && ble_adapter_ != null && ble_adapter_.isEnabled() && ble_scanner_ != null ) {
// stop scanning
ble_scanner_.stopScan( scan_cb_ );
scanComplete();
}
// reset flags
scan_cb_= null;
is_scanning_= false;
scan_handler_= null;
Log.d(TAG,"scanning stopped");
}
/*
Handle scan results after scan stopped
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
private void scanComplete() {
Toast.makeText(context, "scan finished", Toast.LENGTH_LONG).show();
// check if nothing found
if( scan_results_.isEmpty() ) {
Log.d( TAG, "scan results is empty" );
return;
}
ArrayList<BluetoothDevice> found_devices = new ArrayList<>();
ArrayList<String> found_devices_names = new ArrayList<>();
// loop over the scan results and connect to them
for( String device_name : scan_results_.keySet() ) {
Log.d( TAG, "Found device: " + device_name );
if(device_name==null)
continue;
BluetoothDevice device= scan_results_.get( device_name );
if( device_name.contains( BLE_NAME) ) {
found_devices.add(device);
found_devices_names.add(device_name);
}
}
// exist device
if(found_devices.size() >= 1) {
CharSequence[] items = found_devices_names.toArray(new String[found_devices_names.size()]);
AlertDialog.Builder builder = new AlertDialog.Builder(context);
builder.setTitle("select device");
builder.setItems(items, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int pos) {
String selectedText = items[pos].toString();
connectDevice(found_devices.get(pos));
Toast.makeText(context, selectedText, Toast.LENGTH_SHORT).show();
}
});
builder.show();
} else {
Toast.makeText(context, "not found", Toast.LENGTH_SHORT).show();
}
}
/*
Connect to the ble device
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
private void connectDevice(BluetoothDevice _device ) {
// update the status
msg_handler(SET_TV_CONNECTION, "connecting");
devAddress = _device.getAddress();
editor = pref.edit();
editor.putString("first_device",devAddress);
editor.apply();
Log.d(TAG,"Connecting to " + devAddress);
GattClientCallback gatt_client_cb= new GattClientCallback();
ble_gatt_= _device.connectGatt( context, false, gatt_client_cb );// %CONNECT%
}
/*
Disconnect Gatt Server
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
public void disconnectGattServer() {
Log.d( TAG, "Closing Gatt connection" );
msg_handler(SET_TV_STATUS,"disconnect");
// reset the connection flag
connected_= false;
// disconnect and close the gatt
if( ble_gatt_ != null ) {
ble_gatt_.disconnect();
ble_gatt_.close();
}
}
/*
Request BLE enable
*/
private void requestEnableBLE() {
Intent ble_enable_intent= new Intent( BluetoothAdapter.ACTION_REQUEST_ENABLE );
((MainActivity)context).startActivityForResult( ble_enable_intent, REQUEST_ENABLE_BT );
}
/*
Request Fine Location permission
*/
@RequiresApi(api = Build.VERSION_CODES.M)
private void requestLocationPermission() {
((MainActivity)context).requestPermissions( new String[]{ Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_FINE_LOCATION );
}
/*
connect to the saved device
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
public boolean connectSavedDevice() {
String result = pref.getString("first_device", "0");
if(result.equals("0")) {
Log.d(TAG,"no saved device");
return false;
}
BluetoothDevice device = ble_adapter_.getRemoteDevice(result);
GattClientCallback gatt_client_cb= new GattClientCallback();
ble_gatt_= device.connectGatt( context, false, gatt_client_cb ); // %connect%뜨는거
return true;
}
/*
Send Data
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
public void sendData(byte[] strByte) {
// check connection
if( !connected_ )
{
Log.e( TAG, "Failed to sendData due to no connection" );
return;
}
// disconnect if the characteristic is not found
if( cmd_characteristic == null ) {
Log.e( TAG, "Unable to find cmd characteristic" );
disconnectGattServer();
return;
}
System.out.print("len : " + strByte.length + "->");
for(int i=0;i<strByte.length;i++) {
System.out.print((int)strByte[i] + " ");
}
System.out.println();
// start stimulation
startStimulation( cmd_characteristic, strByte );
}
/*
Start stimulation
@param cmd_characteristic command characteristic instance
@param program_id stimulation program id
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
private void startStimulation(BluetoothGattCharacteristic _cmd_characteristic, byte[] strByte ) {
_cmd_characteristic.setValue( strByte );
boolean success= ble_gatt_.writeCharacteristic( _cmd_characteristic );
if( success ) {
Log.d( TAG, "send data complete" );
}
else
{
Log.e( TAG, "Failed to write command" );
}
}
final Handler handler = new Handler() {
public void handleMessage(Message msg) {
switch(msg.what) {
case SET_TV_STATUS:
((MainActivity)context).tv_status_handler(msg.obj.toString());
break;
case SET_TV_CONNECTION:
((MainActivity)context).tv_connection_handler(msg.obj.toString());
break;
}
}
};
public void msg_handler(int set_tv, String text) {
Message msg = handler.obtainMessage();
msg.what = set_tv;
msg.obj = text;
handler.sendMessage(msg);
return;
}
/*
BLE Scan Callback class
*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
private class BLEScanCallback extends ScanCallback {
private Map<String, BluetoothDevice> cb_scan_results_;
/*
Constructor
*/
BLEScanCallback( Map<String, BluetoothDevice> _scan_results ) {
cb_scan_results_= _scan_results;
}
@Override
public void onScanResult( int _callback_type, ScanResult _result ) {
Log.d( TAG, "onScanResult" );
addScanResult( _result );
}
@Override
public void onBatchScanResults( List<ScanResult> _results ) {
for( ScanResult result: _results ) {
addScanResult( result );
}
}
@Override
public void onScanFailed( int _error ) {
Log.e( TAG, "BLE scan failed with code " +_error );
}
/*
Add scan result
*/
private void addScanResult( ScanResult _result ) {
// get scanned device
BluetoothDevice device= _result.getDevice();
// save devices for name
String device_name = device.getName();
// add the device to the result list
cb_scan_results_.put( device_name, device );
// log
Log.d( TAG, "scan results device: " + device_name );
msg_handler(SET_TV_STATUS,"scanned device : " + device_name);
}
}
/*
Gatt Client Callback class
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
private class GattClientCallback extends BluetoothGattCallback {
// @Override
// public void onDescriptorWrite(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status) {
// super.onDescriptorWrite(gatt, descriptor, status);
// }
@Override
public void onConnectionStateChange(BluetoothGatt _gatt, int _status, int _new_state ) {
super.onConnectionStateChange( _gatt, _status, _new_state );
if( _status == BluetoothGatt.GATT_FAILURE ) {
disconnectGattServer();
return;
} else if( _status != BluetoothGatt.GATT_SUCCESS ) {
disconnectGattServer();
return;
}
if( _new_state == BluetoothProfile.STATE_CONNECTED ) {
// // 연결완료 되었을 시 1. set communication, 2. get standby state, 3. get basic data 동기화하기.
// sendData(SET_COMM_START);
// MainActivity.packet_sleep(200);
// sendData(GET_STANDBY_STATE); //get airtop standby state(1.3)
// MainActivity.packet_sleep(200);
// if(airtopDevice.standby_state == ACTIVE_STATE) //device active 상태이면 basic data 확인
// sendData(GET_BASIC_DATA); //get airtop basic data(1.1)
// // update the connection status message
// msg_handler(SET_TV_STATUS,"연결 완료");
// msg_handler(SET_TV_CONNECTION,"연결됨");
// set the connection flag
connected_= true;
Log.d( TAG, "Connected to the GATT server" );
_gatt.discoverServices();
} else if ( _new_state == BluetoothProfile.STATE_DISCONNECTED ) {
disconnectGattServer();
}
}
@Override
public void onServicesDiscovered( BluetoothGatt _gatt, int _status ) {
super.onServicesDiscovered( _gatt, _status );
// check if the discovery failed
if( _status != BluetoothGatt.GATT_SUCCESS ) {
Log.e( TAG, "Device service discovery failed, status: " + _status );
return;
}
// find discovered characteristics
List<BluetoothGattCharacteristic> matching_characteristics= BluetoothUtils.findBLECharacteristics(_gatt);
if( matching_characteristics.isEmpty() ) {
Log.e( TAG, "Unable to find characteristics" );
return;
}
// log for successful discovery
Log.d( TAG, "Services discovery is successful" );
// find command characteristics from the GATT server
cmd_characteristic= BluetoothUtils.findCommandCharacteristic( ble_gatt_ );
//setCharacteristicNotification
ble_gatt_.setCharacteristicNotification(cmd_characteristic, true);
BluetoothGattDescriptor descriptor = cmd_characteristic.getDescriptor(CHARACTERISTIC_UPDATE_NOTIFICATION_DESCRIPTOR_UUID);
descriptor.setValue(BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE);
ble_gatt_.writeDescriptor(descriptor); //descriptor write operation successfully started?
// stream open complete.
// update the connection status message
msg_handler(SET_TV_STATUS,"connect complete");
// stream open 이후 send data -> main에서 sleep 이후 보내줌.
// sendData(SET_COMM_START);
// MainActivity.packet_sleep(200);
// sendData(GET_STANDBY_STATE); //get airtop standby state(1.3)
// MainActivity.packet_sleep(200);
// if(airtopDevice.standby_state == ACTIVE_STATE) //device active 상태이면 basic data 확인
// sendData(GET_BASIC_DATA); //get airtop basic data(1.1)
}
@Override
public void onCharacteristicChanged( BluetoothGatt _gatt, BluetoothGattCharacteristic _characteristic ) {
super.onCharacteristicChanged( _gatt, _characteristic );
Log.d( TAG, "characteristic changed: " + _characteristic.getUuid().toString() );
readCharacteristic( _characteristic );
}
@Override
public void onCharacteristicWrite( BluetoothGatt _gatt, BluetoothGattCharacteristic _characteristic, int _status ) {
super.onCharacteristicWrite( _gatt, _characteristic, _status );
if( _status == BluetoothGatt.GATT_SUCCESS ) {
Log.d( TAG, "Characteristic written successfully" );
} else {
Log.e( TAG, "Characteristic write unsuccessful, status: " + _status) ;
disconnectGattServer();
}
}
@Override
public void onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
super.onCharacteristicRead(gatt, characteristic, status);
if (status == BluetoothGatt.GATT_SUCCESS) {
Log.d (TAG, "Characteristic read successfully" );
readCharacteristic(characteristic);
} else {
Log.e( TAG, "Characteristic read unsuccessful, status: " + status);
// Trying to read from the Time Characteristic? It doesnt have the property or permissions
// set to allow this. Normally this would be an error and you would want to:
// disconnectGattServer();
}
}
/*
Log the value of the characteristic
@param characteristic
*/
private void readCharacteristic( BluetoothGattCharacteristic _characteristic ) {
byte[] msg = _characteristic.getValue();
String str = new String(msg);
for(int i=0;i<str.length();i++) {
packetReceive.input_byte(str.charAt(i));
}
String prt = new String();
for(int i=0;i<msg.length;i++) {
prt += " "+(int)msg[i];
}
Log.d(TAG,"msg len"+msg.length + ": " +prt);
}
}
}
最佳答案
您可以实现onConnectionStateChange。这会让您知道连接是否成功。
@Override
public void onConnectionStateChange(BluetoothGatt gatt, int status,int newState) {
if (newState == BluetoothProfile.STATE_CONNECTED) {
// you can send data here
} else if (newState == BluetoothProfile.STATE_DISCONNECTED) {
}
}
关于java - 我想知道BLE gatt设置的完成时间(如回调),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58091211/
您好,我是使用 xampp 的 PHPmyadmin 新手,没有 MYSQL 背景。当我喜欢研究它是如何工作的时,我的脑海中浮现出一个想法,它让我一周都无法休眠,因为我似乎无法弄清楚如何使用 MIN(
Go docs say (强调): Programs using times should typically store and pass them as values, not pointers.
我有一组用户在 8 月 1 日有一个条目。我想找到在 8 月 1 日有条目但在 8 月 2 日没有做任何事情的用户。 现在是 10 月,所以事件已经过去很久了。 我有限的知识说: SELECT * F
我有以下代码,主要编码和取消编码时间结构。这是代码 package main import ( "fmt" "time" "encoding/json" ) type chec
您能详细解释一下“用户 CPU 时间”和“系统 CPU 时间”吗?我读了很多,但我不太理解。 最佳答案 区别在于时间花在用户空间还是内核空间。用户 CPU 时间是处理器运行程序代码(或库中的代码)所花
应用程序不计算东西,但做输入/输出、读取文件、使用网络。我希望探查器显示它。 我希望像 callgrind 中的东西一样,在每个问题中调用 clock_gettime。 或者像 oprofile 那样
目前我的 web 应用程序接收 websocket 数据来触发操作。 这会在页面重新加载时中断,因此我需要一个能够触发特定事件的客户端解决方案。 这个想法可行吗? 假设你有 TimeX = curre
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我有一个 Instant (org.joda.time.Instant) 的实例,我在一些 api 响应中得到它。我有另一个来自 (java.time.Instant) 的实例,这是我从其他调用中获得
如何集成功能 f(y) w.r.t 时间;即 'y'是一个包含 3000 个值和值 time(t) 的数组从 1 到 3000 不等。所以,在整合 f(y) 后我需要 3000 个值. 积分将是不确定
可以通过 CLI 创建命名空间,但是如何使用 Java SDK 来创建命名空间? 最佳答案 它以编程方式通过 gRPC API 完成由服务公开。 在 Java 中,生成的 gRPC 客户端可以通过 W
我有一个函数,它接受 2 组日期(开始日期和结束日期),这些日期将用于我的匹配引擎 我必须知道start_date1和end_date1是否在start_date2和end_date2内 快进:当我在
我想从 Python 脚本运行“time”unix 命令,以计算非 Python 应用程序的执行时间。我会使用 os.system 方法。有什么方法可以在Python中保存这个输出吗?我的目标是多次运
我正在寻找一种“漂亮的数字”算法来确定日期/时间值轴上的标签。我熟悉 Paul Heckbert's Nice Numbers algorithm . 我有一个在 X 轴上显示时间/日期的图,用户可以
在 PowerShell 中,您可以格式化日期以返回当前小时,如下所示: Get-Date -UFormat %H 您可以像这样在 UTC 中获取日期字符串: $dateNow = Get-Date
我正在尝试使用 Javascript 向父子窗口添加一些页面加载检查功能。 我的目标是“从父窗口”检测,每次子窗口完全加载然后执行一些代码。 我在父窗口中使用以下代码示例: childPage=wi
我正在尝试设置此 FFmpeg 命令的 drawtext 何时开始,我尝试使用 start_number 但看起来它不会成功。 ffmpeg -i 1.mp4 -acodec aac -keyint_
我收到了一个 Excel (2010) 电子表格,它基本上是一个文本转储。 单元格 - J8 具有以下信息 2014 年 2 月 4 日星期二 00:08:06 EST 单元格 - L8 具有以下信息
我收到的原始数据包含一列具有以下日期和时间戳格式的数据: 2014 年 3 月 31 日凌晨 3:38 单元格的格式并不一致,因为有些单元格有单个空格,而另一些单元格中有两个或三个字符之间的空格。所以
我想知道是否有办法在我的 Grails 应用程序顶部显示版本和构建日期。 编辑:我应该说我正在寻找构建应用程序的日期/时间。 最佳答案 在您的主模板中,或任何地方。 Server version:
我是一名优秀的程序员,十分优秀!