Форум программистов, компьютерный форум CyberForum.ru

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 44, средняя оценка - 4.61
G-Cat
16 / 16 / 1
Регистрация: 15.03.2009
Сообщений: 94
#1

Черепашья графика - C++

07.02.2010, 19:18. Просмотров 5709. Ответов 23
Метки нет (Все метки)

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

Текст задачи:
Язык Лого, особенно популярный среди пользователей персональных компьютеров, сделал знаменитой черепашью графику. Представьте себе механическую черепаху, которая ползает по комнате под управлением программы на C++. Черепаха несет пишущее перо, которое может находиться в одной из двух позиций - нижней или верхней. Если перо в нижней позиции, черепаха вычерчивает траекторию движения, если в верхней, то черепаха передвигается свободно и ничего не вычерчивает. В этой задаче вы будете моделировать действия черепахи и создавать компьютерезированный эскиз пути.
Используя массив floor размером 20 на 20 с нулевым начальным условиями. Считывайте команды из содержащего их массива. Всё время отмечайте текущую позицию черепахи и положение пера - нижнее или верхнее. Предполагйте, что черепаха всегда стартует из позиции 0, 0 на полу с верхним положением пера. Ваша программа должна подавать команды черепахе в соответствии со следующими обозначениями:
1 - поднять перо
2 - опустить перо
3 - поворот направо
4 - поворот налево
5 - движение вперёд
6 - печать массива 20 на 20
9 - конец данных (контрольное значение)



вот моё решение задачи:
C++
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#include <iostream>
    using std::cout;
    using std::endl;
    using std::cin;
#include <iomanip>
    using std::setw;
 
void printFloor( bool [ 20 ][ 20 ] );
int move( bool [ 20 ][ 20 ], bool, bool );
 
int main( void )
{
    bool array[ 20 ][ 20 ] = { 0 };
    int press;
    bool feather = 0;
    bool way = 0;
    
    do
    {
        cout << "Enter a command:\n";
        cin >> press;
        
        switch( press )
        {
            case 1:
                feather = 0;//pero vverhu
                break;
            case 2:
                feather = 1;//pero vnizu
                break;
            case 3:
                way = 0;//vpravo
                break;
            case 4:
                way = 1;//vlevo
                break;
            case 5:
                move( array, way, feather );        
                break;
            case 6:
                printFloor( array );
                break;
            case 9:
                break;
            default:
                cout << "Wrong enter. Try again.\n";
                break;
        }
        
    }
    while( press != 9 );
 
    return 0;
}
 
int move( bool array[ 20 ][ 20 ], bool way, bool feather )
{
    static int nowX = 5;
    static int nowY = 5;
    
    static int lastX = 6;
    static int lastY = 5;
    
    if( way == 0 )//dvishenie vpravo
    {
        if( (nowX == lastX - 1) && (nowY == lastY) )
        {
            lastX = nowX;
            lastY = nowY;
            nowY++;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX) && (nowY == lastY + 1) )
        {
            lastX = nowX;
            lastY = nowY;
            nowX++;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX + 1) && (nowY == lastY) )
        {
            lastX = nowX;
            lastY = nowY;
            nowY--;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX) && (nowY == lastY - 1) )
        {
            lastX = nowX;
            lastY = nowY;
            nowX--;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
    }
    else//dvishenie vlevo
    {
        if( (nowX == lastX - 1) && (nowY == lastY) )
        {
            lastX = nowX;
            lastY = nowY;
            nowY--;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX) && (nowY == lastY - 1) )
        {
            lastX = nowX;
            lastY = nowY;
            nowX++;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX - 1) && (nowY == lastY) )
        {
            lastX = nowX;
            lastY = nowY;
            nowY++;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
        if( (nowX == lastX) && (nowY == lastY + 1) )
        {
            lastX = nowX;
            lastY = nowY;
            nowX--;
            if( feather == 1 )
                array[ nowX ][ nowY ] = 1;
            return 0;
        }
    }
}
 
void printFloor( bool array[ 20 ][ 20 ])
{
    for( int i = 0; i < 20; i++ )
    {
        for( int j = 0; j < 20; j++ )
        {
            if( array[ i ][ j ] == 1 )
                cout << setw( 2 ) << '*';
            else
                cout << setw( 2 ) << ' ';
        }
        
        cout << endl;
    }
}
Добавлено через 15 часов 49 минут
32 просмотра и ниодного комментария, это говорит об моей идеально решённой задаче?)))
неужели ни у кого нет других вариантов? да, код грамоздкий, но если чуть-чуть изменить условия задачи, а именно сделать не просто поворот на лево/поворот направо, а, грубо говоря, так:
3 - шаг влево
4 - шаг вправо
5 - шаг вверх
6 - шаг вниз
то задача решается проще пареной репы, одним switch'ом. Но тут то и интересность задачи, в её формулировке. Поэтому мне чисто интересно, возможно ли написать более простой и более понятный алгоритм в функции move()...может кто нибудь всё таки поделится идеями? Потому что мне кажется, что от этого алгоритма один шаг до создания игры "змейка"...кстати, следующая прога, которую я собираюсь писать.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
07.02.2010, 19:18     Черепашья графика
Посмотрите здесь:

C++ Графика
C++ ГРАФИКА
Графика в С++ C++
C++ Графика
Черепашья графика C++
C++ Графика[c++]
C++ [C++] Графика
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Turfur
0 / 0 / 0
Регистрация: 16.11.2014
Сообщений: 2
16.11.2014, 21:29     Черепашья графика #21
Сделал криво, в том смысле что массив создается в функции main() и передается по ссылке функции interpritate(), что не вполне логично. Хотел исправить, но лень. Ах да, чуть не забыл, поменял цифры на мнемонические буквы, что-бы было не так мозголомно тестить. Также в моей версии отображается сама черепаха символом @.
C
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
#include <stdio.h>
#include <stdbool.h>
#include <stdlib.h>
#define SIZE_OF_CANVAS 20
#define MAXPROG 501
 
enum direction {up,right,down,left};
 
static void move(enum direction dir, int * x, int * y, bool pero,char canvas[][SIZE_OF_CANVAS])
{
    if (pero)
        canvas[*x][*y] = '#';
    switch (dir)
    {
        case right:
            if ((*y+1) < SIZE_OF_CANVAS)
                (*y)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case left:
            if (*y > 0)
                (*y)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case up:
            if (*x > 0)
                (*x)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case down:
            if ((*x+1) < SIZE_OF_CANVAS)
                (*x)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
    }
}
 
static void show_canvas(char canvas[][SIZE_OF_CANVAS],int x,int y)
{
    int i,j;
    puts("");
    for (i = 0; i < SIZE_OF_CANVAS; i++)
    {
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            if ((i != x) || (j != y))
                putchar(canvas[i][j]);
            else
                putchar('@');
        putchar('\n');
    }
    puts("");
}
 
void interpritate(char * program, char canvas[][SIZE_OF_CANVAS])
{
    int x = 0,y = 0,i = 0;
    bool pero = false;
    enum direction dir = right;
    while (program[i] != '\n')
    {
        switch (program[i])
        {
            case 'p':
                pero = false; 
                break;
            case 'o':
                pero = true;
                break;
            case 'r':
                switch (dir)
                {
                    case right: dir = down; break;
                    case down: dir = left; break;
                    case left: dir = up; break;
                    case up: dir = right; break;
                }
                break;
            case 'l':
                switch (dir)
                {
                    case right: dir = up; break;
                    case down: dir = right; break;
                    case left: dir = down; break;
                    case up: dir = left; break;
                }
                break;
            case 'm':
                move(dir,&x,&y,pero,canvas);
                break;
            case 's':
                show_canvas(canvas,x,y);
                break;
            case 'e':
                show_canvas(canvas,x,y);
                return;
            default:
                printf("Syntax error\n");
                return;
        }
        i++;
    }
}
 
int main(int argc, char *argv[]) {
    char canvas[SIZE_OF_CANVAS][SIZE_OF_CANVAS],program[MAXPROG],i,j;
    for (i = 0; i < SIZE_OF_CANVAS; i++)
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            canvas[i][j] = ' ';
    printf("Enter program to turtle: \n");
    if (fgets(program,MAXPROG - 1,stdin) != NULL)
        interpritate(program,canvas);
    else
        printf("Error of input\n");
    system("pause");
    return 0;
}
Добавлено через 4 часа 2 минуты
Убрал пару статиков, ибо делалось в двух файлах. Теперь поле не из пробелов, а из точек. Добавлены команды видимости черепахи v и i, соответственно visible and invisible. Добавлены аргументы при m.
C
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
#include <stdio.h>
#include <stdbool.h>
#include <stdlib.h>
#define SIZE_OF_CANVAS 20
#define MAXPROG 501
 
enum direction {up,right,down,left};
 
void move(enum direction dir, int * x, int * y, bool pero,char canvas[][SIZE_OF_CANVAS])
{
    if (pero)
        canvas[*x][*y] = '#';
    switch (dir)
    {
        case right:
            if ((*y+1) < SIZE_OF_CANVAS)
                (*y)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case left:
            if (*y > 0)
                (*y)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case up:
            if (*x > 0)
                (*x)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case down:
            if ((*x+1) < SIZE_OF_CANVAS)
                (*x)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
    }
}
 
void moven(enum direction dir, int * x, int * y, bool pero,char canvas[][SIZE_OF_CANVAS],int n)
{
    int i;
    for (i = 0; i < n; i++)
        move(dir,x,y,pero,canvas);
}
 
void show_canvas(char canvas[][SIZE_OF_CANVAS],int x,int y, bool visible)
{
    int i,j;
    puts("");
    for (i = 0; i < SIZE_OF_CANVAS; i++)
    {
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            if ((visible == true) && (x == i) && (y == j))
                putchar('@');
            else
                putchar(canvas[i][j]);
        putchar('\n');
    }
    puts("");
}
 
void interpritate(char * program, char canvas[][SIZE_OF_CANVAS])
{
    int x = 0,y = 0,i = 0;
    bool pero = false,visible = true;
    enum direction dir = right;
    while (program[i] != '\n')
    {
        switch (program[i])
        {
            case 'p':
                pero = false; 
                break;
            case 'o':
                pero = true;
                break;
            case 'r':
                switch (dir)
                {
                    case right: dir = down; break;
                    case down: dir = left; break;
                    case left: dir = up; break;
                    case up: dir = right; break;
                }
                break;
            case 'l':
                switch (dir)
                {
                    case right: dir = up; break;
                    case down: dir = right; break;
                    case left: dir = down; break;
                    case up: dir = left; break;
                }
                break;
            case 'm':
                i++;
                if (atoi((char []){program[i],'\0'}) != 0)
                    moven(dir,&x,&y,pero,canvas,atoi((char []){program[i],'\0'}));
                else
                    puts("Incorrect argument of \"m\"");
                break;
            case 's':
                show_canvas(canvas,x,y,visible);
                break;
            case 'v':
                visible = true;
                break;
            case 'i':
                visible = false;
                break;
            case 'e':
                show_canvas(canvas,x,y,visible);
                return;
            default:
                printf("Syntax error\n");
                return;
        }
        i++;
    }
}
 
int main(int argc, char *argv[]) {
    char canvas[SIZE_OF_CANVAS][SIZE_OF_CANVAS],program[MAXPROG],i,j;
    for (i = 0; i < SIZE_OF_CANVAS; i++)
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            canvas[i][j] = '.';
    printf("Enter program to turtle: \n");
    if (fgets(program,MAXPROG - 1,stdin) != NULL)
        interpritate(program,canvas);
    else
        printf("Error of input\n");
    system("pause");
    return 0;
}
gunslinger
случайный прохожий
1117 / 735 / 187
Регистрация: 20.07.2013
Сообщений: 2,016
16.11.2014, 22:31     Черепашья графика #22
Можно считать, что черепаха рисует хвостом и при нахождении в текущей клетке (ячейке) касается предыдущей. Чтобы закрасить текущую, нужно сместиться в следующую.
При повороте можно допустить, что хвост касается границы между клетками, поэтому "левые" (не относящиеся к "пути следования", не путать с направлением) клетки не закрашиваются.
Turfur
0 / 0 / 0
Регистрация: 16.11.2014
Сообщений: 2
18.12.2014, 18:36     Черепашья графика #23
От нечего делать добавил в предыдущую версию возможность писать комментарии, заключенные в c...c и циклы в виде fa...e где a - односимвольное число, означающее кол-во итераций(хотел сделать многосимвольное, но мне лень). Теперь все это работает в REPL режиме, а выход символом q вместо последовательности команд. Код:
C
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
#include <stdio.h>
#include <stdbool.h>
#include <stdlib.h>
#define SIZE_OF_CANVAS 21
#define MAXPROG 501
#define SIZE_OF_RBODY 20
 
enum direction {up,right,down,left};
 
void move(enum direction dir, int * x, int * y, bool pero,char canvas[][SIZE_OF_CANVAS])
{
    if (pero)
        canvas[*x][*y] = '#';
    switch (dir)
    {
        case right:
            if ((*y+1) < SIZE_OF_CANVAS)
                (*y)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case left:
            if (*y > 0)
                (*y)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case up:
            if (*x > 0)
                (*x)--;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
        case down:
            if ((*x+1) < SIZE_OF_CANVAS)
                (*x)++;
            else
            {
                puts("Out of canvas!\n");
                return;
            }
            break;
    }
}
 
void moven(enum direction dir, int * x, int * y, bool pero,char canvas[][SIZE_OF_CANVAS],int n)
{
    int i;
    for (i = 0; i < n; i++)
        move(dir,x,y,pero,canvas);
}
 
void show_canvas(char canvas[][SIZE_OF_CANVAS],int x,int y, bool visible)
{
    int i,j;
    puts("");
    for (i = 0; i < SIZE_OF_CANVAS; i++)
    {
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            if ((visible == true) && (x == i) && (y == j))
                putchar('@');
            else
                putchar(canvas[i][j]);
        putchar('\n');
    }
    puts("");
}
 
void interpritate(char * program, char canvas[][SIZE_OF_CANVAS],int * x,int * y,bool * pero,bool * visible,enum direction * dir)
{
    int i = 0;
    while (program[i] != '\n')
    {
        switch (program[i])
        {
            case 'p':
                *pero = false; 
                break;
            case 'o':
                *pero = true;
                break;
            case 'r':
                switch (*dir)
                {
                    case right: *dir = down; break;
                    case down: *dir = left; break;
                    case left: *dir = up; break;
                    case up: *dir = right; break;
                }
                break;
            case 'l':
                switch (*dir)
                {
                    case right: *dir = up; break;
                    case down: *dir = right; break;
                    case left: *dir = down; break;
                    case up: *dir = left; break;
                }
                break;
            case 'm':
                i++;
                if (atoi((char []){program[i],'\0'}) != 0)
                    moven(*dir,x,y,*pero,canvas,atoi((char []){program[i],'\0'}));
                else
                    puts("Incorrect argument of \"m\"");
                break;
            case 's':
                show_canvas(canvas,*x,*y,*visible);
                break;
            case 'v':
                *visible = true;
                break;
            case 'i':
                *visible = false;
                break;
            case 'c':
                i++;
                while (program[i] != 'c')
                    i++;
                break;
            case 'f':
                i++;
                if (atoi((char []){program[i],'\0'}) != 0)
                {
                    char body[SIZE_OF_RBODY],n = atoi((char []){program[i],'\0'}),j = 0;
                    i++;
                    while (program[i] != 'e')
                    {
                        body[j] = program[i];
                        j++;i++;
                    }
                    body[j] = '\n';
                    body[j+1] = '\0';
                    for (j = 0; j < n; j++)
                        interpritate(body,canvas,x,y,pero,visible,dir);
                }
                else
                    puts("Incorrect argument of \"f\"");
                break;
            default:
                printf("Syntax error\n");
                return;
        }
        i++;
    }
}
 
int main(int argc, char *argv[]) {
    char canvas[SIZE_OF_CANVAS][SIZE_OF_CANVAS],program[MAXPROG],i,j;
    int x = 0,y = 0;
    bool pero = false,visible = true;
    enum direction dir= right;
    for (i = 0; i < SIZE_OF_CANVAS; i++)
        for (j = 0; j < SIZE_OF_CANVAS; j++)
            canvas[i][j] = '.';
    while (1)
    {
        printf("Enter program to turtle: \n");
        if (fgets(program,MAXPROG - 1,stdin) != NULL)
            if (strcmp(program,"q\n"))
                interpritate(program,canvas,&x,&y,&pero,&visible,&dir);
            else
                break;
        else
            printf("Error of input\n");
    }
    puts("Bye!");
    system("pause");
    return 0;
}
Полный набор команд:
p - поднять перо,
o - опустить перо,
r - повернуть направо,
l - повернуть налево,
ma - сдвинуться на a шагов вперед(a - односимвольное),
s - отобразить холст на момент вызова,
v - сделать черепаху видимой,
i - сделать черепаху невидимой,
fa..e - цикл из a итераций с телом вместо ..(a - односимвольное),
c..c - комментарии,
q - выход из программы. Работает только если подан в качестве единственного символа, иначе будет воспринят как синтаксическая ошибка.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
29.04.2016, 21:55     Черепашья графика
Еще ссылки по теме:

Графика C++
C++ Черепашья графика
C++ Черепашья графика (язык Лого)
C++ Графика в C++
C++ Черепашья графика

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

Или воспользуйтесь поиском по форуму:
Liss29
22 / 11 / 2
Регистрация: 18.11.2012
Сообщений: 461
Завершенные тесты: 1
29.04.2016, 21:55     Черепашья графика #24
Кликните здесь для просмотра всего текста
[CPP][static int nowX = 5;
static int nowY = 5;

static int lastX = 6;
static int lastY = 5;

if( way == 0 )//dvishenie vpravo
{
if( (nowX == lastX - 1) && (nowY == lastY) )
{
lastX = nowX;
lastY = nowY;
nowY++;
if( feather == 1 )
array[ nowX ][ nowY ] = 1;
return 0;
}
if( (nowX == lastX) && (nowY == lastY + 1) )
{
lastX = nowX;
lastY = nowY;
nowX++;
if( feather == 1 )
array[ nowX ][ nowY ] = 1;
return 0;
}
if( (nowX == lastX + 1) && (nowY == lastY) )
{
lastX = nowX;
lastY = nowY;
nowY--;
if( feather == 1 )
array[ nowX ][ nowY ] = 1;
return 0;
}
if( (nowX == lastX) && (nowY == lastY - 1) )
{
lastX = nowX;
lastY = nowY;
nowX--;
if( feather == 1 )
array[ nowX ][ nowY ] = 1;
return 0;
}
}
/CPP]
Дошёл до этой задачи у Дейтела и оказался в ступоре, если честно, если с выборам ещё понятно, что нужно применить либо
C++
1
else if
, либо
C++
1
switch
то с движением полная ..опа. Вот этот кусок кода мне не понятен, как это, Грубо говоря это координаты X и Y так? начало и конец соответственно, так? Это как в декартовой системе координат X это всё, что относится к координатам по горизонтали, а Y соответственно всё что относится к координтам по вертикали, так? Помогите разобраться для начала с этим или направте на статью где доходчиво описано это всё?
Yandex
Объявления
29.04.2016, 21:55     Черепашья графика
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru