参考链接:1. PS流的格式和解析总结 http://www.cnblogs.com/lihaiping/p/4181607.html
     2. TS科普5 PES包解析 https://blog.csdn.net/cabbage2008/article/details/49612011

PES包的解析(本代码主要解析了PTS和DTS, 需结合下图和代码中的PES包的伪代码看):

startcode(24) + streamid(8) + pes_len(16) + {header: flag1(8) + flag2(8, pts标识在这儿) + header_len(8)} + {header_data(header_len, 若前面的flag有数据, 数据就在这儿)} + pes_data(pes_len-3-header_len)

 #include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h> #define TAB44 " "
#define PRINTF_DEBUG #define MAX_PS_STARTCODE_LEN 4
#define MAX_PDTS_LEN 5
#define MAX_ES_NUMS 6
#define MAX_PDTS_STRING_LEN 12
#define MMIN_PS_HEADER_LEN 14 #define SCODE_PS_END 0x000001B9
#define SCODE_PS_HEADER 0x000001BA
#define SCODE_PS_SYSTEM_HEADER 0x000001BB
#define SCODE_PS_SYSTEM_MAP_HEADER 0x000001BC /**********************************************************************************************************
pack_header() {
pack_start_code 32 bits
'01' 2 bits
system_clock_reference_base[32..30] 3 bits
marker_bit 1 bit
system_clock_reference_base[29..15] 15 bits
marker_bit 1 bit
system_clock_reference_base[14..0] 15 bits
marker_bit 1 bit
system_clock_reference_extension 9 bits
marker_bit 1 bit
program_mux_rate 22 bits
marker_bit 1 bit
marker_bit 1 bit
reserved 5 bit
pack_stuffing_length 3 bits for (i=0; i<pack_stuffing_length; i++){
stuffing_byte 8 bits
} if (nextbits() == system_header_start_code) {
system_header()
}
}
**********************************************************************************************************/ /**********************************************************************************************************
system_header() {
system_header_start_code 32 bits
header_length 16 bits
marker_bit 1 bit
rate_bound 22 bits
marker_bit 1 bit
audio_bound 6 bits
fixed_flag 1 bit
CSPS_flag 1 bit
system_audio_lock_flag 1 bit
system_video_lock_flag 1 bit
marker_bit 1 bit
vedio_bound 5 bits
packet_rate_restriction_flag 1 bit
reserved_bits 7 bits while (nextbits() == '1') {
stream_id 8 bits
'11' 2 bits
P-STD_buffer_bound_scale 1 bit
P-STD_buffer_size_bound 13 bits
}
}
**********************************************************************************************************/ /**********************************************************************************************************
program_stream_map() {
packet_start_code_prefix 24 bits
map_stream_id 8 bits
program_stream_map_length 16 bits
current_next_indicator 1 bit
reserved 2 bits
program_stream_map_version 5 bits
reserved 7 bits
marker_bit 1 bit
program_stream_info_length 16 bits for (i=0;i<N;i++) {
descriptor()
} elementary_stream_map_length 16 bits for (i=0;i<N1;i++) {
stream_type 8 bits
elementary_stream_id 8 bits elementary_stream_info_length 16 bits for (i=0;i<N2;i++) {
descriptor()
}
} CRC_32 32 bits
} ** current_next_indicator: 当前下一个指示符字段, 1位字段. 置'1'时表示传送的节目流映射当前是可用的.
置'0'时表示传送的节目流映射还不可用, 但它将是下一个生效的表.
** program_stream_map_version: 5位字段, 表示整个节目流映射的版本号. 一旦节目流映射的定义发生变化,
该字段将递增1, 并对32取模. 在current_next_indicator为'1'时, 该字段应该是当前适用的节目流映射的版本号;
在current_next_indicator为'0'时, 该字段应该是下一个适用的节目流映射的版本号.
** stream_type: 流类型字段, 该字段只能标志包含在PES分组中的基本流且取值不能为0x05.
1. MPEG-4视频流: 0x10;
2. H.264视频流: 0x1B;
3. SVAC视频流: 0x80;
4. G.711音频流: 0x90;
5. G.722.1音频流: 0x92;
6. G.723.1音频流: 0x93;
7. G.729音频流: 0x99;
8. SVAC音频流: 0x9B.
因为节目映射流字段只有在关键帧打包的时候, 才会存在, 所以如果要判断PS打包的流编码类型, 就根据这个字段来判断.
** elementary_stream_map_length: 基本流映射长度字段. 指出在该节目流映射中的所有基本流信息的字节长度.
它只包括stream_type、elementary_stream_id和elementary_stream_info_length字段.
** elementary_stream_id: 基本流标识字段, 8位字段, 指出该基本流所在PES分组的PES分组标题中stream_id字段的值.
这个字段的定义, 其中0x(C0~DF)指音频, 0x(E0~EF)为视频.
**********************************************************************************************************/
typedef struct t_es_map
{
unsigned char streamType;
unsigned char esId;
unsigned short esInfoLen;
} T_ES_MAP; typedef struct t_ps_map
{
unsigned char curNextInd:, :, version:; unsigned short psInfoLen;
unsigned short esStreamMapLen; unsigned int esMapNum; T_ES_MAP esMaps[MAX_ES_NUMS];
} T_PS_MAP; /**********************************************************************************************************
PES_packet() {
packet_start_code_prefix 24 bits
stream_id 8 bits
PES_packet_length 16 bits if (stream_id != program_stream_map
&& stream_id != padding_stream
&& stream_id != private_stream_2
&& stream_id != ECM
&& stream_id != EMM
&& stream_id != program_stream_directory
&& stream_id != DSMCC_stream
&& stream_id != ITU-T Rec.H.222.1 type E stream) {
'10' 2 bits
PES_scrambling_control 2 bits
PES_priority 1 bit
data_alignment_indicator 1 bit
copyright 1 bit
original_or_copy 1 bit PTS_DTS_flags 2 bits
ESCR_flag 1 bit
ES_rate_flag 1 bit
DSM_trick_mode_flag 1 bit
additional_copy_info_flag 1 bit
PES_CRC_flag 1 bit
PES_extension_flag 1 bit PES_header_data_length 8 bits if (PTS_DTS_flags == '10') {
'0010' 4 bits
PTS[32..30] 3 bits
marker_bit 1 bit
PTS[29..15] 15 bits
marker_bit 1 bit
PTS[14..0] 15 bits
marker_bit 1 bit
} if (PTS_DTS_flags == '11') {
'0011' 4 bits
PTS[32..30] 3 bits
marker_bit 1 bit
PTS[29..15] 15 bits
marker_bit 1 bit
PTS[14..0] 15 bits
marker_bit 1 bit
'0001' 4 bits
PTS[32..30] 3 bits
marker_bit 1 bit
PTS[29..15] 15 bits
marker_bit 1 bits
PTS[14..0] 15 bits
marker_bit 1 bit
} if (ESCR_flag == '1') {
reserved 2 bits
ESCR_base[32..30] 3 bits
marker_bit 1 bit
ESCR_base[29..15] 15 bits
marker_bit 1 bit
ESCR_base[14..0] 15 bits
marker_bit 1 bit
ESCR_extension 9 bits
marker_bit 1 bit
} if (ES_rate_flag == '1') {
marker_bit 1 bit
ES_rate 22 bits
marker_bit 1 bit
} if (DSM_trick_mode_flag == '1') {
trick_mode_control 3 bits if (trick_mode_control == fast_forward) {
field_id 2 bits
intra_slice_refresh 1 bits
frequency_truncation 2 bits
} else if (trick_mode_control == slow_motion) {
rep_cntrl 5 bits
} else if (trick_mode _control == freeze_frame) {
field_id 2 bits
reserved 3 bits
} else if (trick_mode _control == fast_reverse) {
field_id 2 bits
intra_slice_refresh 1 bit
frequency_truncation 2 bits
} else if (trick_mode_control == slow_reverse) {
rep_cntrl 5 bits
} else {
reserved 5 bits
}
} if (additional_copy_info_flag =='1') {
marker_bit 1 bit
additional_copy_info 7 bits
} if (PES_CRC_flag == ‘1’) {
previous_PES_packet_CRC 16 bits
} if (PES_extension_flag == '1') {
PES_private_data_flag 1 bit
pack_header_field_flag 1 bit
program_packet_sequence_counter_flag 1 bit
P-STD_buffer_flag 1 bit
reserved 3 bits
PES_extension_flag_2 1 bit if (PES_private_data_flag == '1') {
PES_private_data 128 bits
} if (pack_header_field_flag == '1') {
pack_field_length 8 bits
pack_header()
} if (program_packet_sequence_counter_flag == '1') {
marker_bit 1 bit
program_packet_sequence_counter 7 bits
marker-bit 1 bit
MPEG1_MPEG2_indentifier 1 bit
original_stuff_length 6 bits
} if (P-STD_buffer_flag == '1') {
'01' 2 bits
P-STD_buffer_scale 1 bit
P-STD_buffer_size 13 bits
} if (PES_extension_flag_2 == '1') {
marker_bit 1 bit
PES_extension_field_length 7 bits for (i=0; i<PES_extension_field_length; i++) {
reserved 8 bits
}
}
} for (i=0; i<N1; i++) {
stuffing_byte 8 bits
} for (i=0; i<N2; i++) {
PES_packet_data_byte 8 bits
}
} else if (stream_id == program_stream_map
|| stream_id == private_stream_2
|| stream_id == ECM
|| stream_id == EMM
|| stream_id == program_stream_directory
|| stream_id == DSMCC_stream
|| stream_id == ITU-T Rec. H.222.1 type E stream ) {
for (i=0; i<PES_packet_length; i++) {
PES_packet_data_byte 8 bits
}
} else if (steam_id == padding_stream) {
for (i=0; i<PES_packet_length; i++) {
padding_byte 8 bits
}
}
} ** stream_id:
1011 1100 program_stream_map(0xBC)
1011 1101 private_stream_1(0xBD)
1011 1110 padding_stream(0xBE)
1011 1111 private_stream-2(0xBF)
110x xxxx GB/T XXXX.3或GB/T AAAA.3音频流编号xxxx(0xC0~0xDF)
1110 xxxx GB/T XXXX.2或GB/T AAAA.2视频流编号xxxx(0xE0~0xEF)
1111 0000 ECM_stream(0xF0)
1111 0001 EMM_stream(0xF1)
1111 0010 GB/T XXXX.1附录B或GB/T XXXX.6_DSMCC_stream(0xF2)
1111 0011 ISO/IEC_13522_stream(0xF3)
1111 0100 ITU-T Rec. H.222.1类型A
1111 0101 ITU-T Rec. H.222.1类型B
1111 0110 ITU-T Rec. H.222.1类型C
1111 0111 ITU-T Rec. H.222.1类型D
1111 1000 ITU-T Rec. H.222.1类型E
1111 1001 ancillary_stream(0xF9)
1111 1010…1111 1110 保留数据流
1111 1111 program_stream_directory(0xFF)
符号x表示值'0'或'1'均被允许且可产生相同的流类型. 流号码由x的取值决定.
**********************************************************************************************************/
typedef struct t_ps_pes
{
unsigned char streamId; long long pts;
long long dts; unsigned char ptsStr[MAX_PDTS_STRING_LEN+];
unsigned char dtsStr[MAX_PDTS_STRING_LEN+]; unsigned char pesHeaderLen;
} T_PS_PES; static void parsePsHeader(unsigned char* const psHeaderData)
{ } static void parsePsSystemHeader(unsigned char* const psSysHeaderData)
{ } static void parsePsSystemMapHeader(unsigned char* const psMapHeaderData)
{
int i = ; T_PS_MAP psMap = {}; unsigned char *data = NULL; data = psMapHeaderData; memset(&psMap, , sizeof(psMap)); psMap.curNextInd = (data[]>>) & 0x1;
psMap.version = data[] & 0x1f; data += ; psMap.psInfoLen = (data[] << ) | data[]; data += psMap.psInfoLen; psMap.esStreamMapLen = (data[] << ) | data[]; psMap.esMapNum = psMap.esStreamMapLen / ; for (i=; i<psMap.esMapNum; i++)
{
if (i == MAX_ES_NUMS)
{
printf("now just save %d es info!\n", MAX_ES_NUMS); break;
} psMap.esMaps[i].streamType = data[];
psMap.esMaps[i].esId = data[];
psMap.esMaps[i].esInfoLen = (data[] << ) | data[]; data += (+psMap.esMaps[i].esInfoLen);
} #ifdef PRINTF_DEBUG
int mNUm = ; if (psMap.esMapNum > MAX_ES_NUMS)
{
mNUm = MAX_ES_NUMS;
} for (i=; i<mNUm; i++)
{
printf("%s%sstreamNum: %d, streamType: %d, esId: %d\n", TAB44, TAB44, i, psMap.esMaps[i].streamType, psMap.esMaps[i].esId);
}
#endif
} static void getPdts(unsigned char *pdtsData, long long *pdts, unsigned char *pdtsString)
{
int hour = ;
int minute = ;
int second = ;
int msecond = ; long long pts = ;
long long pts2Ms = ; unsigned char ptsStr[MAX_PDTS_STRING_LEN+] = {}; /* 5个字节转33位的值 */
pts = (((pdtsData[]>>) & 0x7) << ) | (pdtsData[] << ) | (((pdtsData[]>>) & 0x7f) << ) | (pdtsData[] << ) | (pdtsData[]>> & 0x7f); /* 90KHz, 1000ms/90 */
pts2Ms = pts/; hour = pts2Ms/(**);
minute = (pts2Ms - hour * (**)) / (*);
second = (pts2Ms - hour * (**) - minute * (*)) / ;
msecond = pts2Ms - hour * (**) - minute * (*) - second * ; sprintf(ptsStr, "%02d:%02d:%02d:%03d", hour, minute, second, msecond); ptsStr[MAX_PDTS_STRING_LEN] = '\0'; memcpy(pdtsString, ptsStr, MAX_PDTS_STRING_LEN); *pdts = pts;
} /*********************************************************************************
startcode(24) + streamid(8) + pes_len(16) + {header: flag1(8) + flag2(8, pts标识在这儿) + header_len(8)} + {header_data(header_len, 若前面的flag有数据, 数据就在这儿)} + pes_data(pes_len-3-header_len)
**********************************************************************************/
static void parsePes(const unsigned char streamId, unsigned char* const pesData, const unsigned short pesLen)
{
unsigned char pts_dts_flag; static int audioNum = ;
static int videoNum = ;
static int privateNum = ;
static int paddingNum = ; unsigned char *data = NULL; unsigned char pts[MAX_PDTS_LEN+] = {};
unsigned char dts[MAX_PDTS_LEN+] = {}; T_PS_PES psPes = {}; data = pesData; memset(&psPes, 0x0, sizeof(psPes)); psPes.streamId = streamId; if (((streamId>=0xC0) && (streamId<=0xDF)) || ((streamId>=0xE0) && (streamId<=0xEF)))
{
pts_dts_flag = data[]>> & 0x3; psPes.pesHeaderLen = data[]; data += ; switch (pts_dts_flag)
{
case : /* 00, no pts, dts */
break; case : /* 10, only pts*/
memset(pts, 0x0, sizeof(pts)); memcpy(pts, data, MAX_PDTS_LEN); getPdts(pts, &psPes.pts, psPes.ptsStr); break; case : /* 11 pts & dts*/
memset(pts, 0x0, sizeof(pts));
memset(dts, 0x0, sizeof(dts)); memcpy(pts, data, MAX_PDTS_LEN);
memcpy(dts, data+MAX_PDTS_LEN, MAX_PDTS_LEN); getPdts(pts, &psPes.pts, psPes.ptsStr);
getPdts(dts, &psPes.dts, psPes.dtsStr); break; default:
break;
}
} #ifdef PRINTF_DEBUG
if ((streamId>=0xC0) && (streamId<=0xDF))
{
audioNum++; printf("%s%spes, Audio[%d], streamId: 0x%02X(%d), pesLength: %d, pesHeaderLen: %d", TAB44, TAB44, audioNum, streamId, streamId, pesLen, psPes.pesHeaderLen); if ( == pts_dts_flag)
{
printf(", pts: %s(%lld)", psPes.ptsStr, psPes.pts);
} if ( == pts_dts_flag)
{
printf(", pts: %s(%lld), dts: %s(%lld)", psPes.ptsStr, psPes.pts, psPes.dtsStr, psPes.dts);
} printf("\n");
}
else if ((streamId>=0xE0) && (streamId<=0xEF))
{
videoNum++; printf("%s%spes, Video[%d], streamId: 0x%02X(%d), pesLength: %d, pesHeaderLen: %d", TAB44, TAB44, videoNum, streamId, streamId, pesLen, psPes.pesHeaderLen); if ( == pts_dts_flag)
{
printf(", pts: %s(%lld)", psPes.ptsStr, psPes.pts);
} if ( == pts_dts_flag)
{
printf(", pts: %s(%lld), dts: %s(%lld)", psPes.ptsStr, psPes.pts, psPes.dtsStr, psPes.dts);
} printf("\n");
}
else if ((streamId==0xBD) || (streamId==0xBF))
{
privateNum++; printf("%s%spes, private[%d], streamId: 0x%02X(%d), pesLength: %d\n", TAB44, TAB44, privateNum, streamId, streamId, pesLen);
}
else if (streamId==0xBE)
{
paddingNum++; printf("%s%spes, padding[%d], streamId: 0x%02X(%d), pesLength: %d\n", TAB44, TAB44, privateNum, streamId, streamId, pesLen);
}
else
{
printf("%s%spes, streamId: 0x%02X(%d), pesLength: %d\n", TAB44, TAB44, streamId, streamId, pesLen);
}
#endif
} int main(int argc, char *argv[])
{
int readLen = ;
int pack_stuffing_length = ; unsigned int startCode = ; unsigned short pesPacketLen = ;
unsigned short psSystemHeaderLen = ;
unsigned short psSystemMapHeaderLen = ; unsigned char pesStreamId = ; unsigned char sCodeData[MAX_PS_STARTCODE_LEN+] = {};
unsigned char psData[MMIN_PS_HEADER_LEN-] = {}; unsigned char *pesData = NULL; FILE *fp = NULL; if ( != argc)
{
printf("Usage: flvparse **.mpg\n"); return -;
} fp = fopen(argv[], "rb");
if (!fp)
{
printf("open file[%s] error!\n", argv[]); return -;
} while ()
{
readLen = fread(&startCode, MAX_PS_STARTCODE_LEN, , fp);
if ( != readLen)
{
break;
} startCode = ntohl(startCode); #ifdef PRINTF_DEBUG
if (SCODE_PS_HEADER == startCode)
{
printf("+startCode: 0x%08X\n", startCode);
}
else
{
printf("%s+startCode: 0x%08X\n", TAB44, startCode);
}
#endif if (( != (startCode>> & 0xff)) && ( != (startCode>> & 0xff)) && ( != (startCode>> & 0xff)))
{
return -;
} switch (startCode)
{
case SCODE_PS_HEADER:
memset(psData, 0x0, sizeof(psData)); readLen = fread(psData, , MMIN_PS_HEADER_LEN-, fp);
if ((MMIN_PS_HEADER_LEN-) != readLen)
{
fclose(fp); return ;
} pack_stuffing_length = psData[MMIN_PS_HEADER_LEN-] & 0x7; fseek(fp, pack_stuffing_length, SEEK_CUR); break; case SCODE_PS_SYSTEM_HEADER:
if ( != fread(&psSystemHeaderLen, , , fp))
{
fclose(fp); return ;
} psSystemHeaderLen = ntohs(psSystemHeaderLen); fseek(fp, psSystemHeaderLen, SEEK_CUR); break; case SCODE_PS_SYSTEM_MAP_HEADER:
if ( != fread(&psSystemMapHeaderLen, , , fp))
{
fclose(fp); return ;
} psSystemMapHeaderLen = ntohs(psSystemMapHeaderLen); pesData = (unsigned char*)malloc(psSystemMapHeaderLen);
if (pesData)
{
memset(pesData, 0x0, pesPacketLen); if (psSystemMapHeaderLen != fread(pesData, , psSystemMapHeaderLen, fp))
{
fclose(fp); return ;
} parsePsSystemMapHeader(pesData); free(pesData); pesData = NULL;
} break; case SCODE_PS_END:
#ifdef PRINTF_DEBUG
printf("ps is end!\n");
#endif return ; break; /* pes pcaket */
default:
pesStreamId = startCode & 0xff; if ( != fread(&pesPacketLen, , , fp))
{
fclose(fp); return ;
} pesPacketLen = ntohs(pesPacketLen); pesData = (unsigned char*)malloc(pesPacketLen);
if (pesData)
{
memset(pesData, 0x0, pesPacketLen); if (pesPacketLen != fread(pesData, , pesPacketLen, fp))
{
fclose(fp); return ;
} parsePes(pesStreamId, pesData, pesPacketLen); free(pesData); pesData = NULL;
} break;
}
} fclose(fp); return ;
}

 最后如果您觉得本篇对您有帮助,可以打赏下,谢谢!!!

ps文件解析(纯c解析代码)的更多相关文章

  1. flv文件解析(纯c解析代码)

    参考链接: 1. FLV科普12 FLV脚本数据解析-Metadata Tag解析 https://blog.csdn.net/cabbage2008/article/details/50500021 ...

  2. mpeg4文件分析(纯c解析代码)

    参考链接: 1. MPEG4码流的帧率计算 https://blog.csdn.net/littlebee90/article/details/68924690                2. M ...

  3. h264文件分析(纯c解析代码)

    参考链接:1. 解析H264的SPS信息 https://blog.csdn.net/lizhijian21/article/details/80982403               2. h.2 ...

  4. ts文件分析(纯c解析代码)

    参考链接: 1. MPEG-2 TS码流分析 https://blog.csdn.net/zhubin215130/article/details/8958567 TS Header PAT PMT ...

  5. h265文件分析(纯c解析代码)

    参考链接: 1. HEVC码流解析 https://blog.csdn.net/CrystalShaw/article/details/80624804   2. HEVC编码结构:序列参数集SPS. ...

  6. mpeg2文件分析(纯c解析代码)

    参考链接: 1. MPEG-2码流结构分析 https://www.cnblogs.com/CoderTian/p/9246225.html(本文语法采用这里的截图,代码原创) 1. mpeg2的码流 ...

  7. mp4文件解析(纯c解析代码)

     参考链接:1. mp4文件格式解析 https://www.cnblogs.com/ranson7zop/p/7889272.html   2. MP4文件格式分析及分割实现(附源码) https: ...

  8. 【转】使用JavaParser获得Java代码中的类名、方法形参列表中的参数名以及统计总的文件个数与不能解析的文件个数

    遍历目录查找Java文件: public static void ergodicDir(File dir, HashSet<String> argNameSet, HashSet<S ...

  9. 原创:微信小程序调用PHP后台接口,解析纯html文本

    ---效果图片预览---    1.微信js动态传参:wx.request({        url: 'https://m.****.com/index.php/Home/Xiaoxxf/activ ...

随机推荐

  1. Bigger-Mai 养成计划,Python基础巩固二

    模块初识1.标准库2.第三方库import sys sys.path #自己的本文件名不可为sys.py#输出模块存储的环境变量sys.argv #打印脚本的相对路径sys.argv[2] #取第二个 ...

  2. Vue-admin工作整理(十七):Mock模拟Ajax请求

    思路:使用Mock拦截actions请求,通过 Mock.mock(/\/getUserInfo/, 'post', getUserInfo) 进行拦截标示,然后将内容返回 export const ...

  3. R语言并行运算示例 parallel 包

    library(parallel)#example 1cl <- makeCluster(getOption("cl.cores", 2))clusterApply(cl, ...

  4. Project Euler 66: Diophantine equation

    题目链接 思路: 连分数求佩尔方程最小特解 参考博客 模板: LL a[]; bool min_pell(LL d, LL &x, LL &y) { LL m = floor(sqrt ...

  5. 在Vue项目中使用Element UI:按需引入和完整引入

    下面操作在main.js文件中进行 完整引入: import Element from 'element-ui'; //样式文件,需单独引入 import 'element-ui/lib/theme- ...

  6. hql 函数大全

    序号  函数名称 说明 类型 支持 使用方法 备注 1 ABS(n) 取绝对值 数学函数 JPAQL HQL ABS(column_name[数字类型对象属性])   2 SQRT(n) 取平方根 数 ...

  7. flutter -------- ListView的使用

    学习了Flutter,来分享一下学习的一些常用的知识,先来说说ListView 案例效果: ListView是一个类似列的widget,它的内容对于其渲染框太长时会自动提供滚动. ListView 摘 ...

  8. WCF:一个棘手的问题

    前言 在做即时通信项目时,手上另一个项目的颠簸,即时通信项目一直是改改停停的,一些改动比较小,没有即时的签入,然后一段时间本地的项目代码与源代码存在不少区别,在这种情况下,因为新的需求添加,需要给WC ...

  9. .net core 获取本地ip及request请求端口

    1.获取ip和端口 string str = (Request.HttpContext.Connection.LocalIpAddress.MapToIPv4().ToString() + " ...

  10. Linux Shell基线配置高级操作

    一.输入解析类 1.1 echo解析tab和换行 问题描述:echo默认是原样输出字符串,并不解析\t和\n等反斜杠字符,如下图所示.我们希望echo能解析\t和\n等字符. 处理办法:可以使用-e指 ...