xnmedian.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. #include <stdlib.h>
  2. #include <grass/gis.h>
  3. #include <grass/raster.h>
  4. #include <grass/raster.h>
  5. #include <grass/calc.h>
  6. /**********************************************************************
  7. median(x1,x2,..,xn)
  8. return median of arguments
  9. **********************************************************************/
  10. static int icmp(const void *aa, const void *bb)
  11. {
  12. const CELL *a = aa;
  13. const CELL *b = bb;
  14. return *a - *b;
  15. }
  16. static int fcmp(const void *aa, const void *bb)
  17. {
  18. const FCELL *a = aa;
  19. const FCELL *b = bb;
  20. if (*a < *b)
  21. return -1;
  22. if (*a > *b)
  23. return 1;
  24. return 0;
  25. }
  26. static int dcmp(const void *aa, const void *bb)
  27. {
  28. const DCELL *a = aa;
  29. const DCELL *b = bb;
  30. if (*a < *b)
  31. return -1;
  32. if (*a > *b)
  33. return 1;
  34. return 0;
  35. }
  36. int f_nmedian(int argc, const int *argt, void **args)
  37. {
  38. static void *array;
  39. static int alloc;
  40. int size = argc * Rast_cell_size(argt[0]);
  41. int i, j;
  42. if (argc < 1)
  43. return E_ARG_LO;
  44. for (i = 1; i <= argc; i++)
  45. if (argt[i] != argt[0])
  46. return E_ARG_TYPE;
  47. if (size > alloc) {
  48. alloc = size;
  49. array = G_realloc(array, size);
  50. }
  51. switch (argt[0]) {
  52. case CELL_TYPE:
  53. {
  54. CELL *res = args[0];
  55. CELL **argv = (CELL **) &args[1];
  56. CELL *a = array;
  57. CELL a1;
  58. CELL *resc;
  59. for (i = 0; i < columns; i++) {
  60. int n = 0;
  61. for (j = 0; j < argc; j++) {
  62. if (IS_NULL_C(&argv[j][i]))
  63. continue;
  64. a[n++] = argv[j][i];
  65. }
  66. resc = &res[i];
  67. if (!n)
  68. SET_NULL_C(resc);
  69. else {
  70. qsort(a, n, sizeof(CELL), icmp);
  71. *resc = a[n / 2];
  72. if ((n & 1) == 0) {
  73. a1 = a[(n - 1) / 2];
  74. if (*resc != a1)
  75. *resc = (*resc + a1) / 2;
  76. }
  77. }
  78. }
  79. return 0;
  80. }
  81. case FCELL_TYPE:
  82. {
  83. FCELL *res = args[0];
  84. FCELL **argv = (FCELL **) &args[1];
  85. FCELL *a = array;
  86. FCELL a1;
  87. FCELL *resc;
  88. for (i = 0; i < columns; i++) {
  89. int n = 0;
  90. for (j = 0; j < argc; j++) {
  91. if (IS_NULL_F(&argv[j][i]))
  92. continue;
  93. a[n++] = argv[j][i];
  94. }
  95. resc = &res[i];
  96. if (!n)
  97. SET_NULL_F(resc);
  98. else {
  99. qsort(a, n, sizeof(FCELL), fcmp);
  100. *resc = a[n / 2];
  101. if ((n & 1) == 0) {
  102. a1 = a[(n - 1) / 2];
  103. if (*resc != a1)
  104. *resc = (*resc + a1) / 2;
  105. }
  106. }
  107. }
  108. return 0;
  109. }
  110. case DCELL_TYPE:
  111. {
  112. DCELL *res = args[0];
  113. DCELL **argv = (DCELL **) &args[1];
  114. DCELL *a = array;
  115. DCELL a1;
  116. DCELL *resc;
  117. for (i = 0; i < columns; i++) {
  118. int n = 0;
  119. for (j = 0; j < argc; j++) {
  120. if (IS_NULL_D(&argv[j][i]))
  121. continue;
  122. a[n++] = argv[j][i];
  123. }
  124. resc = &res[i];
  125. if (!n)
  126. SET_NULL_D(resc);
  127. else {
  128. qsort(a, n, sizeof(DCELL), dcmp);
  129. *resc = a[n / 2];
  130. if ((n & 1) == 0) {
  131. a1 = a[(n - 1) / 2];
  132. if (*resc != a1)
  133. *resc = (*resc + a1) / 2;
  134. }
  135. }
  136. }
  137. return 0;
  138. }
  139. default:
  140. return E_INV_TYPE;
  141. }
  142. }