linux下c程序执行时播放音乐

linux下c程序执行时播放音乐,第1张

/** 

*test.c 

*注意:这个例子在Ubuntu 12.04.1环境下编译运行成功。 

*/  

#include <stdio.h>  

#include <stdlib.h>  

#include <alsa/asoundlib.h>  

  

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

{  

    int i  

    int ret  

    int buf[128]  

    unsigned int val  

    int 做空dir=0  

    char *buffer  

    int size  

    snd_pcm_uframes_t frames  

    snd_pcm_uframes_t periodsize  搭胡猜

    snd_pcm_t *playback_handle//PCM设备句柄pcm.h  

    snd_pcm_hw_params_t *hw_params//硬件信息和PCM流配置  

    if (argc != 2) {  

        printf("error: alsa_play_test [music name]\n")  

        exit(1)  

    }  

    printf("play song %s by wolf\n", argv[1])  

    FILE *fp = fopen(argv[1], "rb")  

    if(fp == NULL)  

    return 0  

    fseek(fp, 100, SEEK_SET)  

      

    //1. 打开PCM,最后一个参数为0意味着标准配置  

    ret = snd_pcm_open(&playback_handle, "default", SND_PCM_STREAM_PLAYBACK, 0)  

    if (ret < 0) {  

        perror("snd_pcm_open")  

        exit(1)  

    }  

      

    //2. 分配snd_pcm_hw_params_t结构体  

    ret = snd_pcm_hw_params_malloc(&hw_params)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params_malloc")  

        exit(1)  

    }  

    //3. 初始化hw_params  

    ret = snd_pcm_hw_params_any(playback_handle, hw_params)  

    if (ret < 0) {  

    知型    perror("snd_pcm_hw_params_any")  

        exit(1)  

    }  

    //4. 初始化访问权限  

    ret = snd_pcm_hw_params_set_access(playback_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params_set_access")  

        exit(1)  

    }  

    //5. 初始化采样格式SND_PCM_FORMAT_U8,8位  

    ret = snd_pcm_hw_params_set_format(playback_handle, hw_params, SND_PCM_FORMAT_U8)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params_set_format")  

        exit(1)  

    }  

    //6. 设置采样率,如果硬件不支持我们设置的采样率,将使用最接近的  

    //val = 44100,有些录音采样频率固定为8KHz  

      

  

    val = 8000  

    ret = snd_pcm_hw_params_set_rate_near(playback_handle, hw_params, &val, &dir)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params_set_rate_near")  

        exit(1)  

    }  

    //7. 设置通道数量  

    ret = snd_pcm_hw_params_set_channels(playback_handle, hw_params, 2)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params_set_channels")  

        exit(1)  

    }  

      

    /* Set period size to 32 frames. */  

    frames = 32  

    periodsize = frames * 2  

    ret = snd_pcm_hw_params_set_buffer_size_near(playback_handle, hw_params, &periodsize)  

    if (ret < 0)   

    {  

         printf("Unable to set buffer size %li : %s\n", frames * 2, snd_strerror(ret))  

           

    }  

          periodsize /= 2  

  

    ret = snd_pcm_hw_params_set_period_size_near(playback_handle, hw_params, &periodsize, 0)  

    if (ret < 0)   

    {  

        printf("Unable to set period size %li : %s\n", periodsize,  snd_strerror(ret))  

    }  

                                    

    //8. 设置hw_params  

    ret = snd_pcm_hw_params(playback_handle, hw_params)  

    if (ret < 0) {  

        perror("snd_pcm_hw_params")  

        exit(1)  

    }  

      

     /* Use a buffer large enough to hold one period */  

    snd_pcm_hw_params_get_period_size(hw_params, &frames, &dir)  

                                  

    size = frames * 2 /* 2 bytes/sample, 2 channels */  

    buffer = (char *) malloc(size)  

    fprintf(stderr,  

            "size = %d\n",  

            size)  

      

    while (1)   

    {  

        ret = fread(buffer, 1, size, fp)  

        if(ret == 0)   

        {  

              fprintf(stderr, "end of file on input\n")  

              break  

        }   

        else if (ret != size)   

        {  

        }  

        //9. 写音频数据到PCM设备  

        while(ret = snd_pcm_writei(playback_handle, buffer, frames)<0)  

        {  

            usleep(2000)  

            if (ret == -EPIPE)  

            {  

                  /* EPIPE means underrun */  

                  fprintf(stderr, "underrun occurred\n")  

                  //完成硬件参数设置,使设备准备好  

                  snd_pcm_prepare(playback_handle)  

            }   

            else if (ret < 0)   

            {  

                  fprintf(stderr,  

                      "error from writei: %s\n",  

                      snd_strerror(ret))  

            }    

        }  

          

    }         

    //10. 关闭PCM设备句柄  

    snd_pcm_close(playback_handle)  

      

    return 0  

}  

  

//注意:编译的时候应该保持“gcc -o test test.c -L. -lasound”的格式,运行的时候应该保持"./test //clip2.wav"这种格式。

Linux学习起来很自由的,所以你现在需要的是一份源码

Linux下这类的源码很多,比如mplayer。

源码中代码量可能很大,源码中你需要的部分对应你的问题,会有以下相应的代码。

第一步 用C去播放一个音频文件(如MP3)需要一个解码器,因为mp3是一种网络祥羡颤格式,经过压缩的。所以源码中会有encoder,decoder这样的解码器代码,解码出可以直接播放的格式。当然也可以装上别人的播放器内核,这样你只需要一个调用就可以播放了,剩下的就只是界面的开发。

第二步 你可能听说过KDE,GNOME这样的图形界面系统。它们支持的谨败库分别是QT(c++),GTK(C),mplayer用的gtk。这两个库都是跨平台的,你可以把它们想象成MFC。有了这样的库就可以轻松做出界面了。派旦

然后把这两步的代码用数据结构关联起来。点击按钮,回调去调用解码器然后播放,你自己的播放器就成功播放了。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12498830.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-25
下一篇 2023-05-25

发表评论

登录后才能评论

评论列表(0条)

保存