Winmm.dll不返回fi的长度

2024-06-26 01:33:08 发布

您现在位置:Python中文网/ 问答频道 /正文

我正在尝试用python播放一个.mp3文件winmm.dll(ctypes.windll.winmm). 但是,当我尝试以毫秒为单位获取某个文件的长度时,得到的不是实际长度(05:23=323000 ms),而是3。status命令得到的时间格式为“m”,设置命令后时间格式不变。下面是一些说明问题的代码:

from ctypes import windll, c_buffer

fp = 'song.mp3'
alias = 'test'

buf = c_buffer(255)
r = windll.winmm.mciSendStringW(f'open "{fp}" alias {alias}', buf, 254, 0)
print(r)

buf = c_buffer(255)
r = windll.winmm.mciSendStringW(f'status {alias} time format', buf, 254, 0)
print(r, buf.value)

buf = c_buffer(255)
r = windll.winmm.mciSendStringW(f'set {alias} time format milliseconds', buf, 254, 0)
print(r)

buf = c_buffer(255)
r = windll.winmm.mciSendStringW(f'status {alias} time format', buf, 254, 0)
print(r, buf.value)

buf = c_buffer(255)
r = windll.winmm.mciSendStringW(f'status {alias} length', buf, 254, 0)
print(r, buf.value)

及其输出:

0
0 b'm'
0
0 b'm'
0 b'3'

提前感谢您的帮助!你知道吗


Tags: 文件命令formattimevaluebufferstatusalias
1条回答
网友
1楼 · 发布于 2024-06-26 01:33:08

正如我在评论中指出的,这是一个典型的未定义行为示例。有关详细信息,请查看[SO]: C function called from Python via ctypes returns incorrect value (@CristiFati's answer)
而且,您正在混合8位16位字符串。你知道吗

清单[Python 3.Docs]: ctypes - A foreign function library for Python。你知道吗

代码00.py:

#!/usr/bin/env python3

import sys
import ctypes as ct
from ctypes import wintypes as wt


def main(*argv):
    winmm_dll = ct.WinDLL("Winmm.dll")
    mciSendString  = winmm_dll.mciSendStringW
    mciSendString.argtypes = (wt.LPCWSTR, wt.LPWSTR, wt.UINT, wt.HANDLE)
    mciSendString.restype = wt.DWORD

    file_name = "./SampleAudio_0.4mb.mp3"  # Downloaded from https://www.sample-videos.com/audio/mp3/crowd-cheering.mp3
    alias_name = "test00"
    status_commands = [
        f"status {alias_name} time format",
        f"status {alias_name} length",
    ]

    commands = [f"open \"{file_name}\" alias {alias_name}"]
    commands.extend(status_commands)
    commands.append(f"set {alias_name} time format ms")
    commands.extend(status_commands)
    commands.append(f"close {alias_name}")

    out_buf_len = 0xFF
    out_buf = ct.create_unicode_buffer(out_buf_len)
    for command in commands:
        print(f"Sending {command}...")
        res = mciSendString(command, out_buf, out_buf_len, None)
        print(f"  Result: {res}\n  Output: {out_buf.value}")


if __name__ == "__main__":
    print("Python {0:s} {1:d}bit on {2:s}\n".format(" ".join(item.strip() for item in sys.version.split("\n")), 64 if sys.maxsize > 0x100000000 else 32, sys.platform))
    main(*sys.argv[1:])
    print("\nDone.")

输出

[cfati@CFATI-5510-0:e:\Work\Dev\StackOverflow\q059343461]> "e:\Work\Dev\VEnvs\py_064_03.07.03_test0\Scripts\python.exe" code00.py
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 22:22:05) [MSC v.1916 64 bit (AMD64)] 64bit on win32

Sending open "./SampleAudio_0.4mb.mp3" alias test00...
  Result: 0
  Output: 1
Sending status test00 time format...
  Result: 0
  Output: milliseconds
Sending status test00 length...
  Result: 0
  Output: 27746
Sending set test00 time format ms...
  Result: 0
  Output:
Sending status test00 time format...
  Result: 0
  Output: milliseconds
Sending status test00 length...
  Result: 0
  Output: 27746
Sending close test00...
  Result: 0
  Output:

Done.

相关问题 更多 >