dct变换java_DCT, IDCT变换--C语言实现

这篇博客介绍了如何使用C语言实现离散余弦变换(DCT)和逆离散余弦变换(IDCT)。文章给出了DCT和IDCT的变换矩阵,以及计算过程的代码实现。通过示例展示了如何进行DCT和IDCT操作,并在最后验证了变换的正确性。
摘要由CSDN通过智能技术生成

变换矩阵,DCT = IDCT'

static const double MtxDCT[8][8] =

{

{0.3536, 0.3536, 0.3536, 0.3536, 0.3536, 0.3536, 0.3536, 0.3536},

{0.4904, 0.4157, 0.2778, 0.0975, -0.0975, -0.2778, -0.4157, -0.4904},

{0.4619, 0.1913, -0.1913, -0.4619, -0.4619, -0.1913, 0.1913, 0.4619},

{0.4157, -0.0975, -0.4904, -0.2778, 0.2778, 0.4904, 0.0975, -0.4157},

{0.3536, -0.3536, -0.3536, 0.3536, 0.3536, -0.3536, -0.3536, 0.3536},

{0.2778, -0.4904, 0.0975, 0.4157, -0.4157, -0.0975, 0.4904, -0.2778},

{0.1913, -0.4619, 0.4619, -0.1913, -0.1913, 0.4619, -0.4619, 0.1913},

{0.0975, -0.2778, 0.4157, -0.4904, 0.4904, -0.4157, 0.2778, -0.0975}

};

static const double MtxIDCT[8][8] =

{

{0.3536, 0.4904, 0.4619, 0.4157, 0.3536, 0.2778, 0.1913, 0.0975},

{0.3536, 0.4157, 0.1913, -0.0975, -0.3536, -0.4904, -0.4619, -0.2778},

{0.3536, 0.2778, -0.1913, -0.4904, -0.3536, 0.0975, 0.4619, 0.4157},

{0.3536, 0.0975, -0.4619, -0.2778, 0.3536, 0.4157, -0.1913, -0.4904},

{0.3536, -0.0975, -0.4619, 0.2778, 0.3536, -0.4157, -0.1913, 0.4904},

{0.3536, -0.2778, -0.1913, 0.4904, -0.3536, -0.0975, 0.4619, -0.4157},

{0.3536, -0.4157, 0.1913, 0.0975, -0.3536, 0.4904, -0.4619, 0.2778},

{0.3536, -0.4904, 0.4619, -0.4157, 0.3536, -0.2778, 0.1913, -0.0975}

};

变换方程:F(u,v) = G * f * G'

逆变换方程: f(x,y) = G' * F * G

其中 G = DCT,  G' = IDCT , G 与 G' 互为转置

计算:

class MCU

{

public:

int *operator[] (int row)

{

return data[row];

}

int data[8][8];

};

MCU MtxMulI2D(MCU &left, const double right[8][8])

{

MCU dctBuf;

for (int i = 0; i < 8; i++)

{

for (int j = 0; j < 8; j++)

{

double tempVal = 0.0;

for (int k = 0; k < 8; k++)

{

tempVal += left[i][k] * right[k][j];

}

dctBuf[i][j] = round(tempVal);

}

}

return dctBuf;

}

MCU MtxMulD2I(const double left[8][8], MCU &right)

{

MCU dctBuf;

for (int i = 0; i < 8; i++)

{

for (int j = 0; j < 8; j++)

{

double tempVal = 0.0;

for (int k = 0; k < 8; k++)

{

tempVal += left[i][k] * right[k][j];

}

dctBuf[i][j] = round(tempVal);

}

}

return dctBuf;

}

void DCT(MCU &block)

{

block = MtxMulD2I(MtxDCT, block);

block = MtxMulI2D(block, MtxIDCT);

}

void IDCT(MCU &block)

{

block = MtxMulD2I(MtxIDCT, block);

block = MtxMulI2D(block, MtxDCT);

}

测试:

int data2[8][8] =

{

{144,146,149,152,154,156,156,156},

{148,150,152,154,156,156,156,156},

{155,156,157,158,158,157,156,155},

{160,161,161,162,161,159,157,155},

{163,163,164,163,162,160,158,156},

{163,163,164,164,162,160,158,157},

{160,161,162,162,162,161,159,158},

{158,159,161,161,162,161,159,158}

};

int data1[8][8] =

{

{139, 144, 149, 153, 155, 155, 155, 155},

{144, 151, 153, 156, 159, 156, 156, 156},

{150, 155, 160, 163, 158, 156, 156, 156},

{159, 161, 162, 160, 160, 159, 159, 159},

{159, 160, 161, 162, 162, 155, 155, 155},

{161, 161, 161, 161, 160, 157, 157, 157},

{162, 162, 161, 163, 162, 157, 157, 157},

{162, 162, 161, 161, 163, 158, 158, 158}

};

void Show(MCU &block)

{

for (int i=0;i<8;i++)

{

for(int j=0;j<8;j++)

{

printf("%4d ", block[i][j]);

}

printf("\n");

}

}

int main(int argc, char *argv[])

{

MCU block;

for (int i=0;i<8;i++)

{

for(int j=0;j<8;j++)

{

block[i][j] = data1[i][j];

}

}

DCT(block); // 离散余弦变换

IDCT(block); // 逆离散余弦变换

Show(block);

return 0;

}

最后计算结果与原始数据相差不大

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值