这是我的代码:
public class FloatWifiManager implements IWifiManager {
private WifiManager wifiManager;
private BroadcastReceiver wifiScanReceiver;
public FloatWifiManager(Context context) {
...
wifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
// Registering Wifi Receiver
wifiScanReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context c, Intent intent) {
if (intent.getAction().equals(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION)) {
// not getting called, only after running app and manually going to the wifi settings in android
}
}
};
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION);
context.registerReceiver(wifiScanReceiver, intentFilter);
wifiManager.startScan();
}
我完全按照我在所有示例中看到的那样注册了BroadcastReceiver
,并startScan
。
会发生什么, wifi列表正在发生变化(当然,我已经测试过),但如果我只是留在应用中,则不会调用onReceive
。
最终调用onReceive
的原因是启动应用程序,让它保持运行,然后进入Android手机设置 - > Wifi设置。去那里时,List突然更新,并且onReceive
被调用。
这里有什么问题?
wifiManager.startScan();
只运行一次扫描吗?或者它是一个持续收听传入“扫描结果”的功能?
显然,为什么接收器不会被调用?
答案 0 :(得分:8)
是的,startScan()
只请求一次扫描。
您可以摆脱if (intent.getAction().equals(..))
条件。其他任何事情似乎都没问题。
只是为了说清楚 - 我的目标是拥有一个接收器 每次Wifi网络列表都在变化时调用,而不是 单击“开始扫描”按钮。
AFAIK无论何时任何wifi网络发生变化,都无法收到通知。您只能使用startScan
请求扫描 - 当然,您可以使用线程或处理程序重复调用startScan
。
docs表示在“接入点扫描完成时调用了SCAN_RESULTS_AVAILABLE_ACTION
,并且可以从请求者”获得结果。如何以及何时进行扫描取决于请求者的实施。 Elenkov写道,“Android设备很少包含原始的wpa_supplicant代码;所包含的实现经常被修改以更好地与底层SoC兼容”。
扫描接入点
此示例扫描可用的接入点和ad hoc网络。 btnScan
激活WifiManager.startScan()
方法启动的扫描。扫描完成后,WifiManager
调用SCAN_RESULTS_AVAILABLE_ACTION
意图,WifiScanReceiver
类处理扫描结果。结果显示在TextView
。
public class MainActivity extends AppCompatActivity {
private final static String TAG = "MainActivity";
TextView txtWifiInfo;
WifiManager wifi;
WifiScanReceiver wifiReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
wifi=(WifiManager)getSystemService(Context.WIFI_SERVICE);
wifiReceiver = new WifiScanReceiver();
txtWifiInfo = (TextView)findViewById(R.id.txtWifiInfo);
Button btnScan = (Button)findViewById(R.id.btnScan);
btnScan.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Log.i(TAG, "Start scan...");
wifi.startScan();
}
});
}
protected void onPause() {
unregisterReceiver(wifiReceiver);
super.onPause();
}
protected void onResume() {
registerReceiver(
wifiReceiver,
new IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION)
);
super.onResume();
}
private class WifiScanReceiver extends BroadcastReceiver {
public void onReceive(Context c, Intent intent) {
List<ScanResult> wifiScanList = wifi.getScanResults();
txtWifiInfo.setText("");
for(int i = 0; i < wifiScanList.size(); i++){
String info = ((wifiScanList.get(i)).toString());
txtWifiInfo.append(info+"\n\n");
}
}
}
}
<强>权限强>
需要在 AndroidManifest.xml 中定义以下权限:
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
调用android.permission.ACCESS_WIFI_STATE
需要 WifiManager.getScanResults()
。如果没有android.permission.CHANGE_WIFI_STATE
,则无法使用WifiManager.startScan()
启动扫描。
在为api级别23或更高级别(Android 6.0及更高版本)编译项目时,必须插入android.permission.ACCESS_FINE_LOCATION
或android.permission.ACCESS_COARSE_LOCATION
。此外,需要申请许可,例如在您的主要活动的onCreate
方法中:
@Override
protected void onCreate(Bundle savedInstanceState) {
...
String[] PERMS_INITIAL={
Manifest.permission.ACCESS_FINE_LOCATION,
};
ActivityCompat.requestPermissions(this, PERMS_INITIAL, 127);
}