Форум программистов, компьютерный форум, киберфорум
Java SE (J2SE)
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.95/56: Рейтинг темы: голосов - 56, средняя оценка - 4.95
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32

Добавить еще один поток, который выводит на экран другое сообщение каждые 7 секунд

25.07.2014, 01:56. Показов 10810. Ответов 16
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Есть задача:
Напишите программу, которая каждую секунду отображает на экране данные о времени, прошедшем от начала сессии, а другой ее поток выводит сообщение каждые 5 секунд. Предусмотрите возможность ежесекундного оповещения потока, воспроизводящего сообщение, потоком, отсчитывающим время. Не внося изменений в код потока-"хронометра" , добавьте еще один поток, который выводит на экран другое сообщение каждые 7 секунд.
Предполагается использование методов wait(), notifyAll().

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


Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
public class test {
    public static void main (String [] args) throws InterruptedException{
        Chronometr my=new Chronometr();
        Runnable t1=new Messenger(5, my);
        new Thread(t1, "t1").start();
        for (int i=0;i<10;i++) {
            synchronized(my) {
                Thread.sleep(1000);
                Chronometr.time++;
                System.out.println(Chronometr.time);
                my.notifyAll();
                }
            }
        }
    }
    
class Chronometr {
    static int time=0;
    public void timeCount() throws InterruptedException {
         
    }
}   
class Messenger implements Runnable{
    int time;
    public Chronometr ch;
    Messenger(int time, Chronometr ch) {
        this.time=time;
        this.ch=ch;
    }
    public void waitForTime() {
        synchronized (ch) { 
            
            try{
                while (Chronometr.time%time!=0)
                    ch.wait();
                System.out.println("Thread "+this.time);
            }
            catch(InterruptedException e) {}
        } 
    }
    public void run() {
        waitForTime();
    }
}
0
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
25.07.2014, 01:56
Ответы с готовыми решениями:

Выводит на экран каждые 5 секунд попеременно строки из первого и второго файлов
Написать shell-процедуру, которая: Читает содержимое первого файла, передаваемого в качестве первого параметра; Читает содержимое...

Нарисовать графический объект звезда, который меняет цвет через каждые 5 секунд
Нарисовать графический объект звезда, который меняет цвет через каждые 5 секунд.

Резидентный обработчик прерывания, который каждые 5 секунд меняет цвет символов некоторого сообщения
Помогите решить задачу пожалуйста на ассемблере. Завтра нужно сдавать, а я совсем не шарю Разработать резидентный обработчик...

16
871 / 721 / 304
Регистрация: 15.04.2013
Сообщений: 2,047
Записей в блоге: 5
25.07.2014, 16:31
Hbadger,
Как вариант, или вам надо именно на основе вашего кода?
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
    public static void main(String[] args) throws InterruptedException {
        Messenger m1 = new Messenger(1, 1);
        Messenger m5 = new Messenger(5, 2);
        Messenger m7 = new Messenger(7, 3);
        m1.run();
        m5.run();
        m7.run();
    }
 
    static class Messenger implements Runnable {
        int time;
        int id;
 
        /**
         * @param time time in seconds
         * @param id   id of thread
         */
        Messenger(int time, int id) {
            this.time = time * 1000;
            this.id = id;
        }
 
        @Override
        public void run() {
 
            Timer timer = new Timer();
            timer.schedule(new SayHello(id), 0, time);
        }
 
        class SayHello extends TimerTask {
            int id;
 
            SayHello(int id) {
                this.id = id;
            }
 
            @Override
            public void run() {
                System.out.printf("I'm thread with id=%d\n", id);
            }
        }
    }
0
Эксперт Java
 Аватар для turbanoff
4094 / 3828 / 745
Регистрация: 18.05.2010
Сообщений: 9,331
Записей в блоге: 12
25.07.2014, 19:00
Hbadger, у вас в принципе всё правильно сделано.
Проблема в том, что второй поток не успевает захватить монитор - его всё время обгоняет главный поток. Вынесите Thread.sleep из блока synchronized, это должно помочь
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
27.07.2014, 14:47  [ТС]
turbanoff, да-да. с этим я уже разобралась. но теперь возникла другая проблема. дело в том, что поток, который должен выводить сообщение каждые 5 секунд, по факту делает это только один раз. дело, я так поняла, в том, что он запускается на выполнение, выводит сообщение, когда условие в while становится false и завершается. мне же необходимо, что после вывода сообщения он уходил опять в wait-set и просыпался, через следующие 5 секунд. я пробовала поместить этот поток во внешний цикл, но это приводить лишь к тому, что сообщение, которое он должен выводить по прошествии 5 секунд выводится бесконечное количество раз. затем я добавила булеву переменную, которая должна сигнализировать каждому из потоков, что другой поток завершил свои действия и пробуждать его.
теперь мой код выглядит вот так:
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
public class test {
    public static void main (String [] args) throws InterruptedException{
        Chronometr my=new Chronometr();
        Runnable t1=new Messenger(5, my);
        
        new Thread(t1, "t1").start();
        
        for (int i=0;i<10;i++) {
            synchronized(my) {
                while (my.flag)
                    my.wait();
                Chronometr.time+=1;
                System.out.println(Chronometr.time);
    
                my.flag=false;
                my.notifyAll();
                
                
                }
            Thread.sleep(1000);
            }
        
        
        }
    }
    
class Chronometr {
    static int time=0;
    boolean flag=true;
    public void timeCount() throws InterruptedException {
         
    }
}   
class Messenger implements Runnable{
    int time;
    public Chronometr ch;
    Messenger(int time, Chronometr ch) {
        this.time=time;
        this.ch=ch;
    }
    public void waitForTime() {
        
        synchronized (ch) { 
            
            try{
            
            while (ch.flag) 
                ch.wait();
            
            if(Chronometr.time%this.time==0) 
            {
                System.out.println("Thread "+this.time);
                ch.flag=false;
                ch.notifyAll();
            }   
            ch.flag=false;
            ch.notifyAll();
            }
    
            catch(InterruptedException e) {}
        }
    } 
    
    public void run()  {
    waitForTime();
    }
}

Результат: основной поток выполняет свои действия 2 раза (т.е. на экран выводятся цифры 1 и 2) и все. при этом программа работу не завершает, т.к. второй поток висит в wait-set.

Добавлено через 6 минут
XRoy, спасибо, но меня интересует именно мой код. дело в том, что я совсем новичек в java и, например, класса Timer я пока что не знаю вовсе. Я, конечно, вижу, что с его использованием мороки будет гораздо меньше и вполне могу с его использованием по быстрому разобраться, но мне кажется на этапе обучения как раз важно выполнять задания с тем багажом знаний, который авторы задания посчитали достаточным для его реализации. Если вам не сложно, посмотрите, пожалуйста, мой исправленный (но по прежнему некорректно работающий) код в комментарии ниже.
0
Эксперт Java
 Аватар для turbanoff
4094 / 3828 / 745
Регистрация: 18.05.2010
Сообщений: 9,331
Записей в блоге: 12
27.07.2014, 16:58
Цитата Сообщение от Hbadger Посмотреть сообщение
я пробовала поместить этот поток во внешний цикл, но это приводить лишь к тому, что сообщение, которое он должен выводить по прошествии 5 секунд выводится бесконечное количество раз
Была правильная идея. Надо добавить еще один внешний цикл в Messenger. Покажите тот старый код?
Проще всего сделать ваш второй поток - демоном, тогда JVM сама завершит его после того как основной поток закончит выполняться.
Как-то так:
Java
1
2
3
    public void waitForTime() {
        Thread.currentThread().setDaemon(true);
        //...
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
27.07.2014, 18:34  [ТС]
turbanoff, сразу прошу прощения за "много текста", для себя я разобралась с этой задачей на том уровне знаний, которым владею сейчас. Если у вас будет время одолеть мою писанину и код, который к ней прилагается, буду благодарна за ваши замечания.
Я добавила внешний цикл в мессенджер, добавила булеву переменную, по которой цикл в мессенджере и соотвественно сам поток должен завершаться, после того, как завершиться основной поток, где выполняется хронометр. Главная проблема была в том, что флажок, по которому просыпался поток с месенджером был объявлен в классе Chronometr и, соотвественно, если я хотела запустить несколько потоков с несколькими мессенджерами сразу, то они вызывались из wait set одновременно и только один из них получал монитор объекта, в результате при запуске уже двух потоков с мессенджерами на выходе получалась полная каша.
Я пересесла переменную-флажок с класс Messenger и теперь каждый мессенджер имеет свой уникальный флажок, по которому он выходит из режима ожидания. Реализацию отсчета времени и оповещения потоков мессенджеров полностью перенесла в класс Chronometr. Теперь метод countTime(), отсчитывает время и каждую секунду изменяет флажки, которые отвечают за пробуждение потоков мессенджеров с последующим их оповещением. Я реализовала этот метод для 2-х потоков мессенджеров, но, я так понимаю, можно его реализовать для любого количества таких потоков с использованием vararg.
Ниже привожу последний вариант кода:

Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
public class test {
    public static void main (String [] args) throws InterruptedException{
        Chronometr my=new Chronometr();
        Runnable t1=new Messenger(5, my);
        Runnable t2=new Messenger(1, my);
        new Thread(t1, "t1").start();
        new Thread(t2, "t2").start();
        my.countTime((Messenger)t1, (Messenger)t2, 10);
        }
    }
    
 
class Chronometr {
public int time=0;
public void countTime (Messenger m, Messenger m1, int period) {
    for (int i=0; i<period; i++){
        synchronized(this) {
            time+=1;
            System.out.println(time);
            if (i==period-1) Messenger.finish=true;
            m.flag=false;
            this.notify();
            m1.flag=false;
            this.notify();
            }
    try {   Thread.sleep(1000);}
    catch (InterruptedException e) {};
        }
}
}   
class Messenger implements Runnable{
    private int time;
    public Chronometr ch;
    public static boolean finish=false;
    public boolean flag=true;
    Messenger(int time, Chronometr ch) {
        this.time=time;
        this.ch=ch;
    }
    public void waitForTime() {
        while(true){
        synchronized (ch) { 
            try{
                while (flag) 
                    ch.wait();
                if (finish) return;
                if(ch.time%this.time==0) 
                {
                    System.out.println("Thread "+this.time);
                    flag=true;
                }   
                flag=true;
            }
            catch(InterruptedException e) {}
        }}
    } 
    public void run()  {
        waitForTime();
        System.out.println("The end");
        }
}
0
Эксперт Java
 Аватар для turbanoff
4094 / 3828 / 745
Регистрация: 18.05.2010
Сообщений: 9,331
Записей в блоге: 12
27.07.2014, 19:07
Цитата Сообщение от Hbadger Посмотреть сообщение
если я хотела запустить несколько потоков с несколькими мессенджерами сразу, то они вызывались из wait set одновременно и только один из них получал монитор объекта, в результате при запуске уже двух потоков с мессенджерами на выходе получалась полная каша.
Так мессенджеры же должны проверять - действительно ли пришло их время. Если нет - опять вызывать wait. И потом бы монитор перехватил бы другой мессенджер. И всё бы работало хорошо.
0
51 / 51 / 42
Регистрация: 06.09.2013
Сообщений: 188
27.07.2014, 19:08
Hbadger, а где брала задачу?
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
27.07.2014, 19:43  [ТС]
Yoghurt_92, Арнольд, Гослинг, Холмс "Язык программирования Java", 2003 год. Старенькая, но по форме подачи материала мне пока больше всего понравилась.

Добавлено через 3 минуты
turbanoff, вот как-то так, да:
Java
1
2
3
4
5
6
7
8
9
10
11
12
public void waitForTime() { 
while(true){
        synchronized (ch) { 
            try{
                if (ch.time%this.time==0) 
                    System.out.println("Thread "+this.time);
                    ch.wait();
                if (finish) return;
                }
            catch (InterruptedException e) {}
            
        }}
0
51 / 51 / 42
Регистрация: 06.09.2013
Сообщений: 188
27.07.2014, 21:24
Hbadger, а можно ссылку?
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
27.07.2014, 22:11  [ТС]
Yoghurt_92, у меня она в бумажном варианте, но думаю, учитывая возраст книги, кто-то её уже давно выложил в сеть))
0
51 / 51 / 42
Регистрация: 06.09.2013
Сообщений: 188
27.07.2014, 22:15
Hbadger, есть старые варианты относящиеся к 97 году... мне она стала любопытна из-за наличия задач, вся литература которую я читаю без таковых, я как-то изучал С++ по Дейтелям, так там после каждой главы огромное количество задач, очень понравилась их книга... а тут такого не наблюдаю...
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
27.07.2014, 22:29  [ТС]
Yoghurt_92, да, именно задачи меня в ней и "покорили")) их немного, но они застявляют подумать. до этого начинала читать Шилдта, где задач нет. В итоге оказалось, что многие вещи, которые мне были типа понятны из книги Шилдта при попытке применить их на практике к задачам из этой книги оказались темным лесом.
п.с. и вправду, именно 2003 года издание не удается найти.
0
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 7
27.07.2014, 23:11
в книге 97 года нет этой задачи( ну или я не внимательно смотрел
я тоже в шилде прочитал потоки,вроде все понятно,но эту задачу написать не смог,хотя условие по-моему не очень хорошо сформулировано
0
51 / 51 / 42
Регистрация: 06.09.2013
Сообщений: 188
28.07.2014, 16:12
Hbadger, я с потоками более-менее разобрался, но все ровно, хотел попрактиковаться... Ладно, срасибо за ответ)

Добавлено через 16 часов 38 минут
silllen, "Флэнаган Д. - Java в примерах. Справочник (2-е издание) - 2003" нашел хороший справочник, там куча примеров и есть задачи, даже на GUI
1
0 / 0 / 0
Регистрация: 25.07.2014
Сообщений: 32
29.07.2014, 12:45  [ТС]
Yoghurt_92, silllen, в 2015 году должно выйти новое издание книги Гослинга - теперь уже с реалиями java 8, ждем-с.
0
51 / 51 / 42
Регистрация: 06.09.2013
Сообщений: 188
29.07.2014, 16:03
Hbadger, ну я надеюсь к 2015 уже освоить это там уже надо будет читать что-то посерьезнее
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
29.07.2014, 16:03
Помогаю со студенческими работами здесь

Сетевой экран блокируеться Авастом каждые 30 секунд....
Аваст постоянно блокирует 4 адреса....

Сообщение через каждые 30 секунд
Ребят, помогите, пожалуйста, как сделать в программе, чтобы через каждые 30 секунд выдавался вопрос продолжать или нет работу программы (y...

Вывод системного времени на экран через каждые 5 секунд
Здравствуйте. Необходимо написать программу на Ассемблере, которая через каждые 5 секунд будет отображать системное время на экране....

не создает еще один поток
пытаюсь создать еще один поток(четвертый,помимо 2-х форм,если это важно) вот собственно листинг,где запускается поток unit Unit5; ...

Резидентная программа: вывод строки символов на экран каждые 45 секунд
Доброго времени суток. Дано задание, написать резидентную программу,которая вывод строку символов на экран каждые 45 секунд. code ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
17
Ответ Создать тему
Новые блоги и статьи
YAFU@home — распределённые вычисления для математики. На CPU
Programma_Boinc 20.01.2026
YAFU@home — распределённые вычисления для математики. На CPU YAFU@home — это BOINC-проект, который занимается факторизацией больших чисел и исследованием aliquot-последовательностей. Звучит. . .
http://iceja.net/ математические сервисы
iceja 20.01.2026
Обновила свой сайт http:/ / iceja. net/ , приделала Fast Fourier Transform экстраполяцию сигналов. Однако предсказывает далеко не каждый сигнал (см ограничения http:/ / iceja. net/ fourier/ docs ). Также. . .
http://iceja.net/ сервер решения полиномов
iceja 18.01.2026
Выкатила http:/ / iceja. net/ сервер решения полиномов (находит действительные корни полиномов методом Штурма). На сайте документация по API, но скажу прямо VPS слабенький и 200 000 полиномов. . .
Расчёт переходных процессов в цепи постоянного тока
igorrr37 16.01.2026
/ * Дана цепь постоянного тока с R, L, C, k(ключ), U, E, J. Программа составляет систему уравнений по 1 и 2 законам Кирхгофа, решает её и находит: токи, напряжения и их 1 и 2 производные при t = 0;. . .
Восстановить юзерскрипты Greasemonkey из бэкапа браузера
damix 15.01.2026
Если восстановить из бэкапа профиль Firefox после переустановки винды, то список юзерскриптов в Greasemonkey будет пустым. Но восстановить их можно так. Для этого понадобится консольная утилита. . .
Сукцессия микоризы: основная теория в виде двух уравнений.
anaschu 11.01.2026
https:/ / rutube. ru/ video/ 7a537f578d808e67a3c6fd818a44a5c4/
WordPad для Windows 11
Jel 10.01.2026
WordPad для Windows 11 — это приложение, которое восстанавливает классический текстовый редактор WordPad в операционной системе Windows 11. После того как Microsoft исключила WordPad из. . .
Classic Notepad for Windows 11
Jel 10.01.2026
Old Classic Notepad for Windows 11 Приложение для Windows 11, позволяющее пользователям вернуть классическую версию текстового редактора «Блокнот» из Windows 10. Программа предоставляет более. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru