-
Notifications
You must be signed in to change notification settings - Fork 0
/
lab3.c
111 lines (92 loc) · 4.21 KB
/
lab3.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
#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h>
#include <math.h>
#include <omp.h>
void swap(float *a, float *b) {
float temp = *a;
*a = *b;
*b = temp;
}
void combSort(float arr[], int n) {
int gap = n;
float shrink = 1.3;
int swapped = 1;
while (gap > 1 || swapped) {
if (gap > 1)
gap = (int)(gap / shrink);
swapped = 0;
//#pragma omp parallel for default(none) shared(arr, gap, n, swapped)
for (int i = 0; i < n - gap; i++) {
if (arr[i] > arr[i + gap]) {
swap(&arr[i], &arr[i + gap]);
swapped = 1;
}
}
}
}
int main(int argc, char* argv[]) {
int i, N;
unsigned int M;
struct timeval T1, T2;
long delta_ms;
N = atoi(argv[1]); // N равен первому параметру командной строки
M = atoi(argv[2]);
omp_set_num_threads(M);
gettimeofday(&T1, NULL); // запомнить текущее время T1
for (i = 0; i < 100; i++) { // 100 экспериментов
unsigned int k = i;
// //Заполнить массив исходных данных размером N
float* M1 = (float*)malloc(N * sizeof(float));
for (int j = 0; j < N; j++) {
M1[j] = rand_r(&k) % 320 + 1; // Случайные числа от 1 до 320
}
float* M2 = (float*)malloc((N/2)*sizeof(float));
for (int j = 0; j < N / 2; j++) {
M2[j] = (rand_r(&k) % (320 * 9)) + 320; // Случайные числа от 320 до 10*320
}
// Решить поставленную задачу, заполнить массив с результатами
// В массиве М1 к каждому элементу применить операцию гиперболический тангенс с последующим уменьшением на 1
#pragma omp parallel for default(none) shared(M1, N)
for (int j = 0; j < N; j++) {
M1[j] = tanh(M1[j]) - 1;
}
// В массиве М2 каждый элемент поочередно сложить с предыдущим (понадобится копия массива М2, из которого нужно будет брать операнды)
// к результату сложения применить операцию десятичный логарифм, возведенный в степень e (считать, что для начального элемента массива предыдущий элемент равен нулю)
float* M2_copy = (float*)malloc((N / 2) * sizeof(float));
#pragma omp parallel for default(none) shared(M2, M2_copy, N)
for(int j = 0; j < N/2; j++){
M2_copy[j] = M2[j];
}
#pragma omp parallel for default(none) shared(M2, M2_copy, N)
for (int j = 0; j < N / 2; j++) {
if (j != 0) {
M2[j] = M2_copy[j - 1] + M2[j];
}
M2[j] = pow(log10(M2[j]), M_E);
}
free(M2_copy);
#pragma omp parallel for default(none) shared(M2, M1, N)
for(int j = 0; j < N/2; j++){
M2[j] = M1[j]/M2[j];
}
// Отсортировать массив с результатами указанным методом
combSort(M2, N / 2);
// Рассчитать сумму синусов тех элементов массива М2, которые при делении на минимальный ненулевой элемент массива М2 дают четное число
float min = M2[0];
float x = 0.0f;
#pragma omp parallel for default(none) shared(M2, min, N) reduction(+:x)
for (int j = 1; j < N / 2; j++) {
if ((int)(M2[j] / min) % 2 == 0) {
x += sin(M2[j]);
}
}
printf("X: %f\n", x);
free(M1);
free(M2);
}
gettimeofday(&T2, NULL); // запомнить текущее время T2
delta_ms = (T2.tv_sec - T1.tv_sec) * 1000 + (T2.tv_usec - T1.tv_usec) / 1000;
printf("N=%d. Milliseconds passed=%ld\n", N, delta_ms);
return 0;
}