基于MATLAB实现的PCM编码和解码过程

多媒体结课作业

(上传的平时上课用的代码,自己用)

  • PCM.m 是文件主程序
  • PCMcoding.m 是编码过程中用到的函数
  • PCMecoding.m 是解码过程中用到的函数

1. PCM.m

clear;
clc;
T=0.0005;
t=-0.01:T:0.01;
fs=2000;
sdt=1/fs;
t1=-0.01:sdt:0.01;
%% 抽样信号
xt=cos(2*pi*30*t)+sin(2*pi*120*t); %模拟原始信号
st=cos(2*pi*30*t1)+sin(2*pi*120*t1); %模拟抽样信号
max = max(abs(st)); %解码用到


% 原始信号
figure;
subplot(2,1,1);plot(t,xt);title('原始信号');grid on;
subplot(2,1,2);stem(t1,st,'.');title('抽样信号');grid on;


%% PCM编码
% PCM 编码,量化部分在函数PCMcoding中
pcm_encode = PCMcoding(xt); %传入原始信号进行编码
figure;
stairs(pcm_encode); %做出编码后的信号的阶梯图查看效果
axis([0 20 -0.1 1.1]);
title('PCM 编码');
grid on;

%% PCM解码部分
%由于在对抽样信号进行PCM编码时,对数据进行了归一化处理
%所以在程序一开始就对抽样信号的最大值进行了记录
%也就是max变量,
%在解码过程中以便恢复到原来的幅度
% PCM 译码
pcm_decode = PCMdecoding(pcm_encode, max);
%{
%要看效果解除注释即可
plot(t,pcm_decode,'r*-',t,xt,'b.-');
legend('PCM译码','原始信号');grid on;
%}
%% 失真度分析
mis=0; %mis就是最后的失真度
for i=1:length(t)
    dc=(st(i)-pcm_decode(i))^2/length(t);
    mis=mis+dc;
end
fprintf('失真度是:%.6f\n',mis);

2. PCMcoding.m

function code=PCMcoding(S)
% PCM编码部分
    z=sign(S);                                %判断S的正负
    MaxS=max(abs(S));                         %求S的最大值 
    S=abs(S/MaxS);                            %归一化
    Q=2048*S;                                 %量化处理
    code=zeros(length(S),8);                  %代码存储矩阵(全零)
    
    % 段落码判断程序
    for i=1:length(S)
        if (Q(i)>=128)&&(Q(i)<=2048)
            code(i,2)=1;            %在第五段与第八段之间,段位码第一位都为"1"
        end
        if (Q(i)>32)&&(Q(i)<128)||(Q(i)>=512)&&(Q(i)<=2048)
            code(i,3)=1;            %在第三四七八段内,段位码第二位为"1"
        end
        if (Q(i)>=16)&&(Q(i)<32)||(Q(i)>=64)&&(Q(i)<128)||(Q(i)>=256)&&(Q(i)<512)||(Q(i)>=1024)&&(Q(i)<=2048)
            code(i,4)=1;            %在二四六八段内,段位码第三位为"1"
        end
    end
    
    N=zeros(length(S));                              %段内码判断程序
    
    for i=1:length(S)
        N(i)=bin2dec(num2str(code(i,2:4)))+1;        %找到code位于第几段
    end
    
    a=[0,16,32,64,128,256,512,1024];                 %量化间隔
    b=[1,1,2,4,8,16,32,64];                          %除以16,得到每段的最小量化间隔
    for i=1:length(S)  
        q=ceil((Q(i)-a(N(i)))/b(N(i)));              %求出在段内的位置
        if q==0
            code(i,(5:8))=[0,0,0,0];                 %如果输入为零则输出"0"
        else k=num2str(dec2bin(q-1,4));              %编码段内码为二进制
            code(i,5)=str2num(k(1));
            code(i,6)=str2num(k(2));
            code(i,7)=str2num(k(3));
            code(i,8)=str2num(k(4));
        end
        %对符号位做标记
        if z(i)>0
            code(i,1)=1;
        elseif z(i)<0
            code(i,1)=0;
        end                                           %符号位的判断
    end
    code = reshape(code', 1, []);  %返回编码后的信号值
end

3. PCMdecoding.m

function s=PCMdecoding(encode, max)
% PCM解码部分
    encode=(reshape(encode',8,length(encode)/8))';
    l=size(encode,1);
    a=[0,16,32,64,128,256,512,1024];
    b=[1 1 2 4 8 16 32 64];
    c=[0 1.5:15.5];
    for i=1:l
        x=encode(i,1);
        T=bin2dec(num2str(encode(i,(2:4))))+1;
        Y=bin2dec(num2str(encode(i,(5:8))));
        if Y==0
            k(i)=a(T)/2048;
        else
            k(i)=(a(T)+b(T)*c(Y))/2048;
        end
        if x==0
            s(i)=-k(i);
        else
            s(i)=k(i);
        end
    end
    s = s*max;
end

结课时间:2021.10.20

  • 8
    点赞
  • 52
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 4
    评论
好的,关于PCM编码解码MATLAB实现,我可以给你一些指导。 首先,让我们明确一下PCM编码解码的基本原理: PCM编码是将模拟信号转换为数字信号的过程,可以分为采样、量化和编码三个步骤。采样是指在一定的时间间隔内对模拟信号进行取样;量化是将采样后的信号按照一定的精度进行量化,将其转换为离散的数值;编码是将量化后的数字信号转换为二进制码,以便于数字信号的传输和存储。 PCM解码是将数字信号转换为模拟信号的过程,可以分为解码、重构和滤波三个步骤。解码是将接收到的二进制码转换为相应的数字信号;重构是将数字信号转换为模拟信号;滤波是对重构后的信号进行滤波处理,以去除数字信号中的高频成分。 下面是一个简单的MATLAB代码,实现了基本的PCM编码解码过程: ``` % 采样频率 Fs = 8000; % 采样时间 t = 0:1/Fs:0.1; % 模拟信号 x = sin(2*pi*1000*t); % 量化精度 q = 8; % 量化电平数 L = 2^q; % 量化器输出范围 range = 2 / L; % 量化 xq = round(x / range) * range; % 编码 code = dec2bin(xq / range * (L/2), q); % 解码 xq2 = bin2dec(code) / (L/2) * range; % 重构 xr = interp1(0:length(xq2)-1, xq2, 0:1/q:length(xq2)-1/q, 'linear'); % 滤波 b = fir1(64, 2000/(Fs/2)); y = filter(b, 1, xr); % 绘图 subplot(2,1,1); plot(t, x); title('Original Signal'); subplot(2,1,2); plot(t, y); title('Reconstructed Signal'); ``` 这段代码实现了一个基于正弦波的PCM编码解码过程,可以作为你学习PCM编码解码的参考。如果你有其他问题或需要更详细的指导,请随时告诉我。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

亲爱的老吉先森

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值