001/******************************************************************************* 002 * This software is provided as a supplement to the authors' textbooks on digital 003 * image processing published by Springer-Verlag in various languages and editions. 004 * Permission to use and distribute this software is granted under the BSD 2-Clause 005 * "Simplified" License (see http://opensource.org/licenses/BSD-2-Clause). 006 * Copyright (c) 2006-2023 Wilhelm Burger, Mark J. Burge. All rights reserved. 007 * Visit https://imagingbook.com for additional details. 008 ******************************************************************************/ 009 010package imagingbook.spectral.dct; 011 012/** 013 * <p> 014 * Calculates the 1D DFT using tabulated cosine values for {@code float} or 015 * {@code double} data (see sub-classes {@link Dct1dDirect.Float} and 016 * {@link Dct1dDirect.Double}, respectively). This implementation is 017 * considerably faster than the naive version without cosine tables (see 018 * {@link Dct1dSlow}). Other optimizations are possible. Note that this class 019 * has no public constructor - instantiate sub-class {@link Dct1dDirect.Float} 020 * or {@link Dct1dDirect.Double} instead, as shown below. See Sec. 20.1 of [1] 021 * for additional details. 022 * </p> 023 * <p> 024 * Usage example (for {@code float} data): 025 * </p> 026 * <pre> 027 * float[] data = {1, 2, 3, 4, 5, 3, 0}; 028 * Dct1d.Float dct = new Dct1dDirect.Float(data.length); 029 * dct.forward(data); 030 * dct.inverse(data); 031 * ... 032 * </pre> 033 * <p> 034 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing – An Algorithmic 035 * Introduction</em>, 3rd ed, Springer (2022). 036 * </p> 037 * 038 * @author WB 039 * @version 2022/10/23 040 * @see Dct1dSlow 041 * @see Dct1dFast 042 */ 043public abstract class Dct1dDirect extends Dct1dImp { 044 045 final double CM0 = 1.0 / Math.sqrt(2); 046 final int M4; // = 4*M 047 final double s; // common scale factor 048 final double[] cosTable; // pre-calculated table of cosines 049 050 private Dct1dDirect(int M) { 051 super(M); 052 this.M4 = 4 * M; 053 this.s = Math.sqrt(2.0 / M); 054 this.cosTable = makeCosineTable(); 055 } 056 057 private double[] makeCosineTable() { 058 double[] table = new double[M4]; // we need a table of size 4*M 059 for (int j = 0; j < M4; j++) { // j is equivalent to (m * (2 * u + 1)) % 4M 060 //double phi = j * Math.PI / (2 * M); 061 double phi = 2 * j * Math.PI / M4; 062 table[j] = Math.cos(phi); 063 } 064 return table; 065 } 066 067 // ------------------------------------------------------------------------------ 068 069 /** 070 * One-dimensional DCT implementation using {@code float} data. 071 */ 072 public static class Float extends Dct1dDirect implements Dct1d.Float { 073 074 private final float[] tmp; // array to hold temporary data 075 076 /** 077 * Constructor. 078 * @param M the data size 079 */ 080 public Float(int M) { 081 super(M); 082 this.tmp = new float[M]; 083 } 084 085 @Override 086 public void forward(float[] g) { 087 checkSize(g); 088 if (g.length != M) 089 throw new IllegalArgumentException(); 090 float[] G = tmp; 091 for (int m = 0; m < M; m++) { 092 double cm = (m == 0) ? CM0 : 1.0; 093 double sum = 0; 094 for (int u = 0; u < M; u++) { 095 sum += g[u] * cm * cosTable[(m * (2 * u + 1)) % M4]; 096 } 097 G[m] = (float) (s * sum); 098 } 099 System.arraycopy(G, 0, g, 0, M); // copy G -> g 100 } 101 102 @Override 103 public void inverse(float[] G) { 104 checkSize(G); 105 if (G.length != M) 106 throw new IllegalArgumentException(); 107 float[] g = tmp; 108 for (int u = 0; u < M; u++) { 109 double sum = 0; 110 for (int m = 0; m < M; m++) { 111 double cm = (m == 0) ? CM0 : 1.0; 112 sum += cm * G[m] * cosTable[(m * (2 * u + 1)) % M4]; 113 } 114 g[u] = (float) (s * sum); 115 } 116 System.arraycopy(g, 0, G, 0, M); // copy g -> G 117 } 118 119 } 120 121 // ------------------------------------------------------------------------------ 122 123 /** 124 * One-dimensional DCT implementation using {@code double} data. 125 */ 126 public static class Double extends Dct1dDirect implements Dct1d.Double { 127 128 private final double[] tmp; // array to hold temporary data 129 130 /** 131 * Constructor. 132 * @param M the data size 133 */ 134 public Double(int M) { 135 super(M); 136 this.tmp = new double[M]; 137 } 138 139 @Override 140 public void forward(double[] g) { 141 checkSize(g); 142 if (g.length != M) 143 throw new IllegalArgumentException(); 144 double[] G = tmp; 145 for (int m = 0; m < M; m++) { 146 double cm = (m == 0) ? CM0 : 1.0; 147 double sum = 0; 148 for (int u = 0; u < M; u++) { 149 sum += g[u] * cm * cosTable[(m * (2 * u + 1)) % M4]; 150 } 151 G[m] = s * sum; 152 } 153 System.arraycopy(G, 0, g, 0, M); // copy G -> g 154 } 155 156 @Override 157 public void inverse(double[] G) { 158 checkSize(G); 159 if (G.length != M) 160 throw new IllegalArgumentException(); 161 double[] g = tmp; 162 for (int u = 0; u < M; u++) { 163 double sum = 0; 164 for (int m = 0; m < M; m++) { 165 double cm = (m == 0) ? CM0 : 1.0; 166 sum += cm * G[m] * cosTable[(m * (2 * u + 1)) % M4]; 167 } 168 g[u] = s * sum; 169 } 170 System.arraycopy(g, 0, G, 0, M); // copy g -> G 171 } 172 } 173 174}