Socket.io 및 Android 네트워크 검색을 사용하여 앱을 만들었습니다. 서버가 시작되고 안드로이드 장치가 서비스를 해결하면 소켓을 만들려고 시도하지만 첫 번째 시도에서는 작동하지 않습니다. 앱을 종료하고 다시 시작하면 제대로 작동합니다. 서버가 액세스 포인트에서 실행 중입니다. 네트워크 연결은 정상이지만 서버에 ping을 할 수 없습니다. 그러나 이전에 언급했듯이 앱을 종료하고 다시 실행하면 정상적으로 작동합니다.
Eidt 2
추가 디버깅 후 내 앱이 Wi-Fi 연결을 설정할 때마다 충돌 범위를 좁혔습니다. ssid 및 암호는 qrcode에서 가져오고 앱은 Wi-Fi 연결을 설정합니다.
qr 코드 데이터 읽기
@Override
public void receiveDetections(Detector.Detections<Barcode> detections) {
final SparseArray<Barcode> barCodes = detections.getDetectedItems();
if (!receivedDetection) {
if (barCodes.size() != 0) {
Log.i(TAG, "received a Barcode");
receivedDetection = true;
Gson g = new Gson();
try {
barCodeData = g.fromJson(barCodes.valueAt(0).rawValue, AccessPointCredentials.class);
} catch (Exception e) {
barCodeData = new AccessPointCredentials();
barCodeData.setSsid(barCodes.valueAt(0).rawValue);
barCodeData.setPass(null);
e.printStackTrace();
}
connectToWifi(barCodeData);
}
}
}
connectToWifi ()
private void connectToWifi(final AccessPointCredentials credentials) {
new AsyncTask<Object, Object, Object>() {
@Override
protected Object doInBackground(Object... objects) {
WifiManager wifiManager = (WifiManager) mActivity.getSystemService(Context.WIFI_SERVICE);
wifiManager.setWifiEnabled(true);
int netId;
try {
Thread.sleep(100);
} catch (InterruptedException ie) {
ie.printStackTrace();
}
for (WifiConfiguration tmp : wifiManager.getConfiguredNetworks()) {
if (tmp.SSID.equals("\"" + credentials.getSsid() + "\"")) {
netId = tmp.networkId;
boolean enabled = wifiManager.enableNetwork(netId, true);
Log.i(TAG, "doInBackground: Enabled? " + enabled);
mActivity.runOnUiThread(new Runnable() {
@Override
public void run() {
mActivity.onBackPressed();
}
});
return null;
}
}
final String[] apSettings = {credentials.getSsid(), credentials.getPass()};
final WifiConnect WifiTask = new WifiConnect(mActivity);
mActivity.runOnUiThread(new Runnable() {
@Override
public void run() {
WifiTask.execute(apSettings);
}
});
this.cancel(true);
return null;
}
@TargetApi(23)
@Override
protected void onPostExecute(Object aVoid) {
super.onPostExecute(aVoid);
if (Build.VERSION.RELEASE.equalsIgnoreCase("6.0")) {
if (!Settings.System.canWrite(mActivity)) {
Intent goToSettings = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
goToSettings.setData(Uri.parse("package:" + mActivity.getPackageName()));
mActivity.startActivity(goToSettings);
} else {
bindToNetwork();
}
} else {
bindToNetwork();
}
}
}.execute();
}
bindToNetwork ()
private void bindToNetwork() {
final ConnectivityManager connectivityManager = (ConnectivityManager) mActivity.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkRequest.Builder builder;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
builder = new NetworkRequest.Builder();
//set the transport type do WIFI
builder.addTransportType(NetworkCapabilities.TRANSPORT_WIFI);
connectivityManager.requestNetwork(builder.build(), new ConnectivityManager.NetworkCallback() {
@Override
public void onAvailable(Network network) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
connectivityManager.bindProcessToNetwork(network);
} else {
//This method was deprecated in API level 23
ConnectivityManager.setProcessDefaultNetwork(network);
}
try {
//do a callback or something else to alert your code that it's ok to send the message through socket now
} catch (Exception e) {
e.printStackTrace();
}
connectivityManager.unregisterNetworkCallback(this);
}
});
}
}
편집하다
부울 메소드 isServerReachable ()은 예외를 발생시킵니다.
java.net.SocketException: socket failed: ENONET (Machine is not on the network)
public boolean isServerReachable(){
boolean isReachable;
try {
isReachable = InetAddress.getByName(mService.getHost().getHostAddress()).isReachable(5000);
} catch (Exception e) {
e.printStackTrace();
isReachable = false;
}
Log.i(TAG, "onServiceResolved: Server reachable? " + String.valueOf(isReachable));
Log.i(TAG, "onServiceResolved: Is Network online? " + String.valueOf(isNetworkOnline()));
if (isReachable){
return true;
} else {
return false;
}
}
나는 그것을 해결했지만 그것이 최선의 해결책이라고 확신하지 않으며 그것이 추가되면 더 나은 해결책을 받아 들일 것입니다.
이후 connectToWifi()
에 Thread.sleep(500);
호출 을 추가 하여 메서드 에서 for 루프를 수정했으며 wifiManager.enableNetwork()
이제 작동 중입니다.
for (WifiConfiguration tmp : wifiManager.getConfiguredNetworks()) {
if (tmp.SSID.equals("\"" + credentials.getSsid() + "\"")) {
netId = tmp.networkId;
boolean enabled = wifiManager.enableNetwork(netId, true);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
ie.printStackTrace();
}
mActivity.runOnUiThread(new Runnable() {
@Override
public void run() {
mActivity.onBackPressed();
}
});
return null;
}
}
이 기사는 인터넷에서 수집됩니다. 재 인쇄 할 때 출처를 알려주십시오.
침해가 발생한 경우 연락 주시기 바랍니다[email protected] 삭제
몇 마디 만하겠습니다