我有两个Android设备,我想使用蓝牙连接,并通过RFCOMM频道传输数据。我只有一台设备接收数据,而另一台设备发送它......
使用此代码,我可以连接到其他设备并开始收听RFCOMM频道:
Method m = device.getClass().getMethod("createRfcommSocket", new Class[] { int.class });
socket = (BluetoothSocket) m.invoke(device, 2);
socket.connect();
class BasicThread implements Runnable{
public void run() {
try {
InputStream stream = socket.getInputStream();
BufferedReader r = new BufferedReader(new InputStreamReader(stream));
while (true){
Log.d("myapp", "now listening...");
latestLine = r.readLine();
Log.d("myapp", latestLine);
}
} catch (IOException e) {
}
}
}
new Thread(new BasicThread()).run();
使用其他设备,我已经实现了这样的监听套接字:
Method m = blue.getClass().getMethod("listenUsingRfcommOn", new Class[] { int.class });
BluetoothServerSocket socket = (BluetoothServerSocket) m.invoke(blue, 2);
BluetoothSocket sock = socket.accept();
Log.d("myapp", "Connected...\n\n\n\n\n\n\n\n");
OutputStream s = sock.getOutputStream();
final PrintWriter out = new PrintWriter(s);
它们都在RFCOMM频道2上连接,并且两者都相互联系,但是,第二个设备始终在BluetoothSocket sock = socket.accept();
任何帮助?
答案 0 :(得分:3)
好的,我是新手,但我可以尝试帮忙。所以这是我的经验,我设法使用反射连接两个设备。我的Android手机使用方法listenUsingInsecureRfcommOn
接收数据,而其他设备是通信主机并通过BT SPP发送数据。我有这个方法的问题,因为它没有可见的SDP记录,所以我无法用其他设备检测到它。因此,我使用Bluecove和Java SE制作小型嗅探器,尝试连接到给定范围内的每个端口。这是代码:
package application.test;
import static java.lang.System.out;
import java.io.InputStream;
import java.io.PrintStream;
import java.text.SimpleDateFormat;
import javax.microedition.io.Connector;
import javax.microedition.io.StreamConnection;
public class RfCommClient {
public static void main(String args[]) {
try {
Thread.sleep(1000);
} catch (InterruptedException e1) {
e1.printStackTrace();
}
String add = "btspp://8C71F894A36D:";
String par = ";authenticate=false;encrypt=false;master=true";
String url = null;
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd;HH-mm-ss-SSS");
for (int i = 1; i < 15; ++i) {
try {
url = add + i + par;
out.format("Time: %s, port = %d\n", sdf.format(System.currentTimeMillis()), i);
StreamConnection conn = (StreamConnection) Connector.open(url);
PrintStream ops = new PrintStream(conn.openOutputStream());
ops.println("Hi there...");
// response
Thread.sleep(1000);
InputStream is = conn.openInputStream();
byte[] resp = new byte[5];
int r = is.read(resp);
out.println("r = " + r + ", response = " + new String(resp, "US-ASCII"));
Thread.sleep(10 * 1000);
conn.close();
} catch (Exception e) {
out.println("Exception occured, time = " + sdf.format(System.currentTimeMillis()) + ", i = " + i);
//e.printStackTrace();
}
}
}
}
我已经了解到一些端口被占用,并且某些端口无法使用(如文档所述,例如端口0)。例如,我认为端口2被采用,因为当我向它发送一些数据时,我从ERR开始收到5个字符:)。
而另一方面,我的线程仍在等待?! :)
这导致我们注意到另一件事,端口(或通道)并不总是映射到所需的数字。例如,对我来说经常发生我想要在端口15上发送一些东西,但在Android上,等待端口9的线程收到了数据:)
所以我建议,检查哪个端口真正分配了!
您可以使用我发布的代码实现这一目标。
另一件事,这是一个link到channelPicker
函数,它在使用普通API时选择通道,如果我没有弄错,在一些常量中应该代表保留通道。
我刚注意到一些事情,我注册端口的代码略有不同,我的方法就是这样:
Method m = cba.getDeclaredMethod("listenUsingInsecureRfcommOn", int.class);
ss = (BluetoothServerSocket) m.invoke(BluetoothAdapter.getDefaultAdapter(), port);
无论如何,我知道这可能为时已晚,但是,将来也许会有类似的问题。