使用libmpg123加alsa实现MP3的播放/暂停,切换,模式选择,C语言5

note:前面的播放/暂停都是使用互斥锁加条件变量阻塞线程,现在播放/暂停是结束线程,记录暂停的位置,播放时从暂停的位置继续播放。关键点在于记录暂停位置的函数pause_position = mpg123_tell(mh);播放重新定位到这个位置:mpg123_seek(mh, pause_position, SEEK_SET);

以下为我的c语言例程源码:

#include <alsa/asoundlib.h>
#include <mpg123.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#include <signal.h>

#define PCM_DEVICE "default"
#define MAX_SONGS 3

// 歌曲列表
char *songs[MAX_SONGS] = {
      "/mnt/hgfs/share/talk_anymore.mp3",
    "/mnt/hgfs/share/poxiao.mp3",
    "/mnt/hgfs/share/attention.mp3"
};

int current_song = 0;
int play_mode = 0;  // 0: 单曲播放, 1: 单曲循环, 2: 列表循环
int is_paused = 0;  // 播放暂停标志
int stop_playback = 0;  // 停止播放标志
int is_playing = 0;  // 是否正在播放标志
char *music_mode[3] = {"单曲模式", "单曲循环", "列表循环"};

pthread_mutex_t mutex;
pthread_cond_t cond;

mpg123_handle *mh;
snd_pcm_t *pcm_handle;
pthread_t play_thread;
off_t pause_position = 0; // 记录暂停位置

// 清理函数,释放资源
void cleanup() {
    if (mh) {
        mpg123_close(mh);
        mpg123_delete(mh);
    }
    if (pcm_handle) {
        snd_pcm_close(pcm_handle);
    }
    mpg123_exit();
}

// 设置ALSA设备
int setup_alsa(int rate, int channels) {
    int err;
    snd_pcm_hw_params_t *params;

    if ((err = snd_pcm_open(&pcm_handle, PCM_DEVICE, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
        fprintf(stderr, "无法打开PCM设备 %s: %s\n", PCM_DEVICE, snd_strerror(err));
        return -1;
    }

    snd_pcm_hw_params_alloca(&params);
    snd_pcm_hw_params_any(pcm_handle, params);
    snd_pcm_hw_params_set_access(pcm_handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);
    snd_pcm_hw_params_set_format(pcm_handle, params, SND_PCM_FORMAT_S16_LE);
    snd_pcm_hw_params_set_channels(pcm_handle, params, channels);
    snd_pcm_hw_params_set_rate(pcm_handle, params, rate, 0);
    if ((err = snd_pcm_hw_params(pcm_handle, params)) < 0) {
        fprintf(stderr, "无法设置PCM硬件参数: %s\n", snd_strerror(err));
        return -1;
    }

    if ((err = snd_pcm_prepare(pcm_handle)) < 0) {
        fprintf(stderr, "无法准备PCM设备: %s\n", snd_strerror(err));
        return -1;
    }

    return 0;
}

// 播放歌曲的线程函数
void *play_song_thread(void *arg) {
    int song_index = *(int *)arg;
    free(arg);  // 释放分配的内存
    int err;
    size_t done;
    unsigned char *audio;
    int channels, encoding;
    long rate;

    // 打开MP3文件
    if (mpg123_open(mh, songs[song_index]) != MPG123_OK) {
        fprintf(stderr, "无法打开文件 %s: %s\n", songs[song_index], mpg123_strerror(mh));
        is_playing = 0;
        return NULL;
    }

    // 获取MP3格式
    if (mpg123_getformat(mh, &rate, &channels, &encoding) != MPG123_OK) {
        fprintf(stderr, "无法获取文件格式: %s\n", mpg123_strerror(mh));
        mpg123_close(mh);
        is_playing = 0;
        return NULL;
    }

    // 配置ALSA
    if (setup_alsa(rate, channels) != 0) {
        mpg123_close(mh);
        is_playing = 0;
        return NULL;
    }

    // 设置输出格式为当前格式
    mpg123_format_none(mh);
    mpg123_format(mh, rate, channels, encoding);

    // 如果暂停位置不为0,跳转到暂停位置
    if (pause_position > 0) {
        mpg123_seek(mh, pause_position, SEEK_SET);
        pause_position = 0;  // 重置暂停位置
    }

    // 分配音频缓冲区
    size_t buffer_size = mpg123_outblock(mh);
    audio = (unsigned char*) malloc(buffer_size);
    if (!audio) {
        fprintf(stderr, "无法分配音频缓冲区\n");
        snd_pcm_close(pcm_handle);
        mpg123_close(mh);
        is_playing = 0;
        return NULL;
    }

    printf("当前播放歌曲是%s   ,歌曲模式是%s \r\n", songs[song_index], music_mode[play_mode]);
    is_playing = 1;

    while (!stop_playback) {
        pthread_mutex_lock(&mutex);
        while (is_paused && !stop_playback) {
            pthread_cond_wait(&cond, &mutex);
        }
        pthread_mutex_unlock(&mutex);

        if (stop_playback) break;

        int ret = mpg123_read(mh, audio, buffer_size, &done);
        if (ret == MPG123_OK || ret == MPG123_DONE) {
            // 写入PCM数据
            if ((err = snd_pcm_writei(pcm_handle, audio, done / (channels * 2))) < 0) { // 2 bytes per sample for S16_LE
                fprintf(stderr, "snd_pcm_writei error: %s\n", snd_strerror(err));
                snd_pcm_prepare(pcm_handle);
            }
        }

        if (ret == MPG123_DONE) {
            // 根据播放模式处理
            pthread_mutex_lock(&mutex);
            if (play_mode == 1) { // 单曲循环
                mpg123_seek(mh, 0, SEEK_SET);
                printf("单曲循环\r\n");
                pthread_mutex_unlock(&mutex);
                continue;
            } else if (play_mode == 2) { // 列表循环
                current_song = (current_song + 1) % MAX_SONGS;
                pthread_mutex_unlock(&mutex);
                free(audio);  // 释放音频缓冲区
                snd_pcm_drop(pcm_handle);
                mpg123_close(mh);
                is_playing = 0;
                start_song(current_song);
                printf("列表循环\r\n");
                return NULL;
            } else { // 单曲播放
                pthread_mutex_unlock(&mutex);
                printf("单曲结束\r\n");
                break; // 退出循环
            }
        } else if (ret != MPG123_OK) {
            fprintf(stderr, "mpg123_read error: %s\n", mpg123_strerror(mh));
            break;
        }
    }

    printf("退出线程了\r\n");
    free(audio);
    snd_pcm_drain(pcm_handle);
    snd_pcm_close(pcm_handle);
    mpg123_close(mh);
    is_playing = 0;
    return NULL;
}

// 开始播放歌曲
void start_song(int song_index) {
    if (is_playing) {
        stop_playback = 1;  // 停止当前播放的歌曲
        pthread_cond_signal(&cond); // 唤醒播放线程以便其检查停止标志
        pthread_join(play_thread, NULL);
        stop_playback = 0;
    }

    // 分配内存以传递给线程
    int *arg = malloc(sizeof(int));
    if (!arg) {
        fprintf(stderr, "无法分配内存\n");
        return;
    }
    *arg = song_index;

    // 重置暂停标志
    is_paused = 0;

    // 创建新的播放线程
    if (pthread_create(&play_thread, NULL, play_song_thread, arg) != 0) {
        fprintf(stderr, "无法创建播放线程\n");
        free(arg);
        return;
    }
}

// 切换播放/暂停状态
void toggle_play_pause() {
    pthread_mutex_lock(&mutex);
    if (is_playing) {  // 如果当前正在播放
        if (is_paused) {
            // 恢复播放
            is_paused = 0;
            pthread_cond_signal(&cond);
            printf("继续播放\n");
        } else {
            // 暂停播放并记录当前播放位置
            pause_position = mpg123_tell(mh);
            stop_playback = 1;  // 终止当前播放线程
            pthread_cond_signal(&cond);  // 唤醒线程以便其检查停止标志
            pthread_join(play_thread, NULL);  // 等待线程结束
            stop_playback = 0;
            is_paused = 1;
            printf("暂停播放, 位置: %ld\n", pause_position);
        }
    } else {
        start_song(current_song);  // 如果没有播放,开始播放当前歌曲
    }
    pthread_mutex_unlock(&mutex);
}

// 切换播放模式
void switch_play_mode() {
    play_mode = (play_mode + 1) % 3;
    printf("播放模式切换为: %s\n", music_mode[play_mode]);
}

int main() {
    // 初始化MPG123库
    if (mpg123_init() != MPG123_OK) {
        fprintf(stderr, "无法初始化MPG123库\n");
        return 1;
    }

    // 创建MPG123句柄
    if ((mh = mpg123_new(NULL, NULL)) == NULL) {
        fprintf(stderr, "无法创建MPG123句柄\n");
        mpg123_exit();
        return 1;
    }

    pthread_mutex_init(&mutex, NULL);
    pthread_cond_init(&cond, NULL);

    // 开始播放第一首歌曲
    start_song(current_song);

    // 控制循环
    char command;
    while (1) {
        printf("输入 'p' 播放/暂停, 'n' 下一首, 'm' 切换模式, 'q' 退出: ");
        command = getchar();
        getchar(); // 处理换行符
        if (command == 'p') {
            toggle_play_pause();
        } else if (command == 'n') {
            pause_position = 0;
            current_song = (current_song + 1) % MAX_SONGS;
            start_song(current_song);
        } else if (command == 'm') {
            switch_play_mode();
        } else if (command == 'q') {
            break;
        }
    }

    stop_playback = 1;
    pthread_cond_signal(&cond);
    pthread_join(play_thread, NULL);

    cleanup();
    pthread_mutex_destroy(&mutex);
    pthread_cond_destroy(&cond);

    return 0;
}

<think>我们使用libmpg123进行MP3解码,然后使用ALSA直接输出PCM数据。 步骤: 1. 初始化libmpg123 2. 打开MP3文件并获取解码参数(采样率、声道数等) 3. 初始化ALSA,设置输出参数(与解码参数匹配) 4. 循环读取解码后的PCM数据,通过ALSA接口播放 5. 清理资源 注意:需要安装libmpg123-dev和libasound2-dev 编译命令:gcc mpg123_alsa.c -o player -lmpg123 -lasound 下面是一个示例代码:</think>### 使用 libmpg123ALSA 在 C 语言播放 MP3 文件 #### 完整代码示例 ```c #include <mpg123.h> #include <alsa/asoundlib.h> #include <stdio.h> #define BUFFER_SIZE 4096 int main(int argc, char *argv[]) { if (argc != 2) { fprintf(stderr, "用法: %s <mp3文件>\n", argv[0]); return 1; } mpg123_handle *mh = NULL; int err; unsigned char *buffer; size_t done; int channels, encoding; long rate; snd_pcm_t *pcm_handle; snd_pcm_hw_params_t *hw_params; // 1. 初始化libmpg123 if (mpg123_init() != MPG123_OK) { fprintf(stderr, "mpg123初始化失败\n"); return 1; } // 2. 创建解码器实例 mh = mpg123_new(NULL, &err); if (!mh) { fprintf(stderr, "创建解码器失败: %s\n", mpg123_plain_strerror(err)); return 1; } // 3. 打开MP3文件 if (mpg123_open(mh, argv[1]) != MPG123_OK) { fprintf(stderr, "文件打开失败: %s\n", mpg123_strerror(mh)); return 1; } // 4. 获取音频格式信息 if (mpg123_getformat(mh, &rate, &channels, &encoding) != MPG123_OK) { fprintf(stderr, "获取格式失败: %s\n", mpg123_strerror(mh)); return 1; } // 5. 设置输出格式(确保支持) mpg123_format_none(mh); mpg123_format(mh, rate, channels, encoding); // 6. 初始化ALSA if (snd_pcm_open(&pcm_handle, "default", SND_PCM_STREAM_PLAYBACK, 0) < 0) { fprintf(stderr, "无法打开PCM设备\n"); return 1; } // 7. 配置ALSA硬件参数 snd_pcm_hw_params_malloc(&hw_params); snd_pcm_hw_params_any(pcm_handle, hw_params); snd_pcm_hw_params_set_access(pcm_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED); snd_pcm_hw_params_set_format(pcm_handle, hw_params, SND_PCM_FORMAT_S16_LE); // 16位小端 snd_pcm_hw_params_set_channels(pcm_handle, hw_params, channels); snd_pcm_hw_params_set_rate_near(pcm_handle, hw_params, (unsigned int*)&rate, 0); // 8. 应用ALSA参数 if (snd_pcm_hw_params(pcm_handle, hw_params) < 0) { fprintf(stderr, "无法设置硬件参数\n"); return 1; } snd_pcm_hw_params_free(hw_params); // 9. 准备PCM设备 if (snd_pcm_prepare(pcm_handle) < 0) { fprintf(stderr, "无法准备PCM接口\n"); return 1; } // 10. 分配解码缓冲区 buffer = (unsigned char*)malloc(BUFFER_SIZE * sizeof(unsigned char)); if (!buffer) { fprintf(stderr, "内存分配失败\n"); return 1; } // 11. 解码并播放循环 printf("正在播放: %s (%.1fkHz, %d声道)\n", argv[1], rate/1000.0, channels); while (mpg123_read(mh, buffer, BUFFER_SIZE, &done) == MPG123_OK) { if (snd_pcm_writei(pcm_handle, buffer, done / (channels * 2)) < 0) { snd_pcm_prepare(pcm_handle); // 处理欠载 } } // 12. 等待播放完成并清理 snd_pcm_drain(pcm_handle); snd_pcm_close(pcm_handle); free(buffer); mpg123_close(mh); mpg123_delete(mh); mpg123_exit(); printf("播放完成\n"); return 0; } ``` #### 关键步骤说明 1. **初始化 libmpg123** - `mpg123_init()` 初始化解码库 - `mpg123_new()` 创建解码器实例 - `mpg123_open()` 打开 MP3 文件 2. **获取音频格式** - `mpg123_getformat()` 获取采样率 $f_s$、声道数和编码格式 - 格式设置公式:$ \text{帧大小} = \text{声道数} \times \text{采样深度/8} $ 3. **ALSA 配置** - `snd_pcm_open()` 打开默认 PCM 设备 - `snd_pcm_hw_params_set_format()` 设置 16 位小端格式 - `snd_pcm_hw_params_set_rate_near()` 设置采样率 4. **解码播放循环** - `mpg123_read()` 解码 MP3 数据为 PCM - `snd_pcm_writei()` 写入 ALSA 缓冲区 - 处理欠载:`snd_pcm_prepare()` 恢复播放状态 #### 编译与运行 ```bash # 安装依赖 sudo apt install libmpg123-dev libasound2-dev # 编译 gcc mp3_player.c -o player -lmpg123 -lasound # 运行 ./player your_song.mp3 ``` #### 常见问题解决 1. **ALSA 配置错误** ```c // 检查设备支持格式 snd_pcm_hw_params_set_format(pcm_handle, hw_params, SND_PCM_FORMAT_S16_LE); // 验证采样率支持 unsigned int actual_rate = rate; snd_pcm_hw_params_set_rate_near(pcm_handle, hw_params, &actual_rate, 0); if (rate != actual_rate) { printf("警告: 采样率从 %ld 调整为 %u\n", rate, actual_rate); } ``` 2. **播放中断处理** ```c // 在播放循环中添恢复机制 snd_pcm_state_t state = snd_pcm_state(pcm_handle); if (state == SND_PCM_STATE_XRUN) { snd_pcm_recover(pcm_handle, -EPIPE, 0); } else if (state == SND_PCM_STATE_SUSPENDED) { snd_pcm_resume(pcm_handle); } ``` 3. **性能优化** - 增缓冲区大小:`#define BUFFER_SIZE 8192` - 使用双缓冲技术减少延迟 - 预解码部分数据填充缓冲区
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

hqb_newfarmer

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

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

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

打赏作者

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

抵扣说明:

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

余额充值