NSD отсутствует NSDManager в DiscoveryListener-callback при вызове resolService - PullRequest
0 голосов
/ 31 мая 2018

Я хочу обмениваться данными между устройствами Android, поэтому следует использовать NSD для поиска других устройств.Я следовал примеру на Android Developers

Служба успешно зарегистрирована в сети серверным устройством, клиентское устройство распознает службу и введите метод обратного вызова discoveryListener для onServiceFound ().Затем я инициализирую resolListener и вызываю resolService (), но приложение вылетает с

java.lang.NullPointerException: Attempt to invoke virtual method 'void android.net.nsd.NsdManager.resolveService(android.net.nsd.NsdServiceInfo, android.net.nsd.NsdManager$ResolveListener)' on a null object reference
                  at de.niklasdahlheimer.oboeapitest.oboeapitest.NSDFinder$1.onServiceFound(NSDFinder.java:59)
                  at android.net.nsd.NsdManager$ServiceHandler.handleMessage(NsdManager.java:333)
                  at android.os.Handler.dispatchMessage(Handler.java:102)
                  at android.os.Looper.loop(Looper.java:154)
                  at android.os.HandlerThread.run(HandlerThread.java:61)

Мой mNsdManager не равен null в классе NSDFinder, но почему-то он равен нулю в методах обратного вызова DiscoveryListener,Похоже, у меня проблема с разными потоками или недоразумение при работе с контекстами.Должен ли мой класс NSDFinder расширять сервис или что-то еще?Я не хочу, чтобы NSD-код в основной активности содержал хорошо структурированный код.


(сокращенная) основная операция

public class MainActivity extends AppCompatActivity {

    //Variable Declarations
    NSDFinder my_nsdfinder;

    @Override
    protected void onCreate(Bundle savedInstanceState) {

        button_discoverserver =(Button) findViewById(R.id.button_discoverServer);



        //Variables Instantiations
        my_nsdfinder = new NSDFinder();


                    button_discoverserver.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                my_nsdfinder.initializeDiscoveryListener();
                my_nsdfinder.startDiscovery(getApplicationContext());
                Toast.makeText(getApplicationContext(),"Suche nach Servern...",Toast.LENGTH_LONG).show();
            }
        });

    } 
}

NSDFinder класс (NSDFinder).java)

public class NSDFinder{
    String TAG = "NSDFINDER";
    NsdManager mNsdManager;
    NsdManager.DiscoveryListener mDiscoveryListener;
    String mServiceName;
    String SERVICE_TYPE = "_http._tcp.";
    Context mContext;
    int mDiscoveryActive = 0;
    FragmentManager m_DialogManager;
    NsdManager.ResolveListener mResolveListener;
    NsdServiceInfo mService;
    int mServiceport;
    InetAddress mServicehostAdress;


    public void initializeDiscoveryListener() {
        // Instantiate a new DiscoveryListener
        mDiscoveryListener = new NsdManager.DiscoveryListener() {

            // Called as soon as service discovery begins.
            @Override
            public void onDiscoveryStarted(String regType) {
                Log.d(TAG, "Service discovery started");
                mDiscoveryActive = 1;

            }

            @Override
            public void onServiceFound(NsdServiceInfo service) {
                // A service was found! Do something with it.
                Log.d(TAG, "Service discovery success\n" + service);

                if (!service.getServiceType().equals(SERVICE_TYPE)) {
                    // Service type is the string containing the protocol and
                    // transport layer for this service.
                    Log.d(TAG, "Unknown Service Type: " + service.getServiceType());
                } else if (service.getServiceName().equals(mServiceName)) {
                    // The name of the service tells the user what they'd be
                    // connecting to. It could be "Bob's Chat App".
                    Log.d(TAG, "Same machine: " + mServiceName);
                } else if (service.getServiceName().contains("LTCTool")){
                    Toast.makeText(mContext,"LTCTool Server gefunden! "+service,Toast.LENGTH_LONG).show();
                    initializeResolveListener();
                    mNsdManager.resolveService(service,mResolveListener);
                    //mNsdManager.stopServiceDiscovery(mDiscoveryListener);
                }
            }

            @Override
            public void onServiceLost(NsdServiceInfo service) {
                // When the network service is no longer available.
                // Internal bookkeeping code goes here.
                Log.e(TAG, "service lost" + service);
            }

            @Override
            public void onDiscoveryStopped(String serviceType) {
                Log.i(TAG, "Discovery stopped: " + serviceType);
                mDiscoveryActive = 0;
            }

            @Override
            public void onStartDiscoveryFailed(String serviceType, int errorCode) {
                Log.e(TAG, "Discovery failed: Error code:" + errorCode);
                mNsdManager.stopServiceDiscovery(this);
            }

            @Override
            public void onStopDiscoveryFailed(String serviceType, int errorCode) {
                Log.e(TAG, "Discovery failed: Error code:" + errorCode);
                mNsdManager.stopServiceDiscovery(this);
            }
        };
    }

    public void startDiscovery(Context _c){
        mContext = _c;
        NsdManager mNsdManager = (NsdManager) _c.getSystemService(Context.NSD_SERVICE);
        mNsdManager.discoverServices(SERVICE_TYPE, mNsdManager.PROTOCOL_DNS_SD, mDiscoveryListener);

    }


    public void initializeResolveListener() {
        Log.d(TAG, "start initalizing ResolveListener");
        mResolveListener = new NsdManager.ResolveListener() {

            @Override
            public void onResolveFailed(NsdServiceInfo serviceInfo, int errorCode) {
                // Called when the resolve fails. Use the error code to debug.
                Log.e(TAG, "Resolve failed" + errorCode);
            }

            @Override
            public void onServiceResolved(NsdServiceInfo serviceInfo) {
                Log.e(TAG, "Resolve Succeeded. " + serviceInfo);

                if (serviceInfo.getServiceName().equals(mServiceName)) {
                    Log.d(TAG, "Same IP.");
                    return;
                }
                mService = serviceInfo;
                mServiceport = mService.getPort();
                mServicehostAdress = mService.getHost();
                Toast.makeText(mContext,"Dienstinformationen aufgelöst! Port: "+mServiceport+" Adresse: "+mServicehostAdress,Toast.LENGTH_LONG).show();
            }
        };

        Log.d(TAG, "ResolveListener initialized");
    }

}

1 Ответ

0 голосов
/ 16 июня 2018

После нескольких месяцев написания кода на Java я снова смотрю на этот вопрос.

Теперь ответ ясен.В

public void startDiscovery(Context _c)

я вызвал

NsdManager mNsdManager = (NsdManager) _c.getSystemService(Context.NSD_SERVICE);

, который вместо инициализации переменной-члена класса объявил новую, локальную переменную в функции startDiscovery().Так что mNsdManager присутствовал только в startDiscovery().Изменение строки на

mNsdManager = (NsdManager) _c.getSystemService(Context.NSD_SERVICE);

, вероятно, решило бы проблему.

Возможно, этот пост когда-нибудь поможет другому "новичку".

Рад видеть, что иногда "загадочные проблемы "будут решаться временем и индивидуальным прогрессом:)

...