Отражение между нитями - PullRequest
       1

Отражение между нитями

3 голосов
/ 20 сентября 2011

Я работаю с установкой, в которой один поток настраивает множество потоков (служб), запускает их все вместе, чтобы имитировать работу системы, затем соединяет их всех в конце и обрабатывает завершение и т. Д. Мой тест выполняется как один из сервисов и общается с другими через JMS. Для одного из моих тестов мне нужен доступ к закрытой переменной, содержащейся в другом потоке. Я не могу изменить код, который выполняется в другом потоке, скажем, добавить метод доступа или заставить его отправлять переменную через JMS. У меня также нет способа передать ссылку на сервис, к которому я хочу получить доступ, в мой тестовый сервис из-за того, как фреймворк все настраивает.

Я знаю имя потока, в котором содержится класс, к которому мне нужен доступ, и я могу получить ссылку на поток, перечислив работающие потоки, но я не знаю, как извлечь что-либо из как только у меня это получится.

Есть ли какой-нибудь способ использовать рефлексию или другие методы, чтобы получить ссылку на класс в другом потоке?

РЕДАКТИРОВАТЬ : Вот пример ситуации, в которой я нахожусь:

import java.lang.reflect.Field;

public class Runner
{
    /**
     * Pretend this is my test class.
     */
    public static void main( String[] args )
    {
        // this is how my test starts up the system and runs the test
        runTest( TestService.class );
    }

    /**
     * Instantiate the test service and start up all of the threads in the
     * system. Doesn't return until test has completed.
     * 
     * @param testServiceClass
     *            the class that will run the test
     */
    static void runTest( Class<? extends Service> testServiceClass )
    {
        try
        {
            // setup the services
            Service testService =
                    testServiceClass.getConstructor( new Class<?>[] { String.class } )
                            .newInstance( "test service" );

            FixedService fixedService = new FixedService( "fixed service" );

            // start the services
            testService.start();
            fixedService.start();

            // wait for testService to signal that it is done
            System.out.println( "Started threads" );
            while ( !testService.isDone() )
            {
                try
                {
                    Thread.sleep( 1000 );
                }
                catch ( InterruptedException e )
                {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
            // stop the fixed service
            fixedService.stop();
            System.out.println( "TestService done, fixed service told to shutdown" );
        }
        catch ( Exception e )
        {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    /**
     * I cannot modify this class. Handling of thread start is similar to real
     * system.
     */
    abstract static class Service implements Runnable
    {
        protected boolean isDone = false;
        protected boolean stop = false;
        private Thread thisServiceThread;

        public Service( String name )
        {
            thisServiceThread = new Thread( this, name );
        }

        public boolean isDone()
        {
            return isDone;
        }

        public void start()
        {
            thisServiceThread.start();
        }

        public void stop()
        {
            this.stop = true;
        }
    }

    /**
     * I can modify this class. This is the class that actually runs my test.
     */
    static class TestService extends Service
    {
        public TestService( String name )
        {
            super( name );
        }

        @Override
        public void run()
        {
            System.out.println( "TestService: started" );

            // TODO: How can I access FixedService.getMe from where without
            // modifying FixedService?
            try
            {
                Field field = FixedService.class.getDeclaredField( "getMe" );
                field.setAccessible( true );
                System.out.println( field.get( null ) );
            }
            catch ( SecurityException e )
            {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            catch ( NoSuchFieldException e )
            {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            catch ( IllegalArgumentException e )
            {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            catch ( IllegalAccessException e )
            {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            System.out.println( "TestService: done" );
            isDone = true;
        }

    }

    /**
     * I cannot modify this class. This is part of the system being tested.
     */
    static class FixedService extends Service
    {
        private boolean getMe = false;

        public FixedService( String name )
        {
            super( name );
        }

        @Override
        public void run()
        {
            System.out.println( "FixedService: started" );

            // don't stop until signaled to do so
            while ( !stop )
            {
                try
                {
                    Thread.sleep( 1000 );
                }
                catch ( InterruptedException e )
                {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }

            System.out.println( "FixedService: gotMe? " + getMe );

            System.out.println( "FixedService: done" );
            isDone = true;
        }
    }
}

Ответы [ 2 ]

3 голосов
/ 20 сентября 2011

Как сказал Хемал Пандья, вам понадобится служебный объект, а не просто класс, если вы действительно хотите читать или манипулировать полем.

Предполагая, что вам нужно Object, это Runnable, установленный в потоке, возможно, с некоторыми очень грязными отражениями. Вы должны использовать хак для доступа к закрытому члену, чтобы получить поле target из потока, а затем использовать его снова, чтобы получить доступ к нужному полю в самом исполняемом файле.

Вот пример кода. Обратите внимание, что я действительно не рассматривал проблемы синхронизации потоков здесь (хотя я не уверен, что возможно даже правильно синхронизировать доступ такого рода)

import java.lang.reflect.Field;

public class SSCCE {
    static class T extends Thread {
        private int i;

        public T(int i) {
            this.i = i;
        }

        @Override
        public void run() {
            while(true) {
                System.out.println("T: " + i);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    // ignore
                }
            }
        }
    }

    static class R implements Runnable {
        private int i;

        public R(int i) {
            this.i = i;
        }

        @Override
        public void run() {
            while(true) {
                System.out.println("R: " + i);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    // ignore
                }
            }
        }
    }

    /**
     * @param args
     */
    public static void main(String[] args) {
        Thread t1 = new T(1);
        Thread t2 = new Thread(new R(2));

        t1.start();
        t2.start();

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            // ignore
        }

        setI(t1,3);
        setI(t2,4);
    }

    static void setI(Thread t, int newVal) {
        // Secret sauce here...
        try {
            Field fTarget = Thread.class.getDeclaredField("target");
            fTarget.setAccessible(true);
            Runnable r = (Runnable) fTarget.get(t);

            // This handles the case that the service overrides the run() method
            // in the thread instead of setting the target runnable
            if (r == null) r = t;

            Field fI = r.getClass().getDeclaredField("i");
            fI.setAccessible(true);
            fI.setInt(r, newVal);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}
0 голосов
/ 20 сентября 2011

Это не Thread, а Object, из которого вы должны получить переменную. Попробуйте придумать небольшой пример кода управляющих и контролируемых потоков, и он станет более понятным.

Доступ к частным пользователям является легкой частью.

РЕДАКТИРОВАТЬ: Используйте Map от Thread.getId() до Runnable. Если вы хотите проверить поток, определить работоспособный и использовать отражение. Возможно, вы не получите последнее значение из-за проблем с синхронизацией.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...