马尔可夫链应用编程c语言,马尔可夫链算法(markov算法)的awk、C++、C语言实现代码...

1. 问题描述

马尔可夫链算法用于生成一段随机的英文,其思想非常简单。首先读入数据,然后将读入的数据分成前缀和后缀两部分,通过前缀来随机获取后缀,籍此产生一段可读的随机英文。

为了说明方便,假设我们有如下一段话:

代码如下:

Show your flowcharts and conceal your tables and I will be mystified. Show your tables and your flowcharts will be obvious.

假设前缀的长度为2,则我们处理输入以后得到如下数据,我们首先获取一个前缀,然后在前缀的后缀列表中随机选择一个单词,然后改变前缀,重复上述过程,这样,我们产生的句子将是可读的。

下面是处理过的数据:

代码如下:

前缀  后缀

show your  flowcharts tables

your flowcharts  and will

flowcharts and  conceal

flowcharts will  be

your tables  and and

will be  mystified. obvious.

be mystified.  show

be obvious.  (end)

处理这个文本的马尔可夫链算法将首先带引show your,然后随机取出flowcharts 或者table 两个单词,假设选择的是flowcharts, 则新的前缀就是your flowcharts,同理,选择table 时,新的前缀就是your table,有了新的前缀your flowcharts 以后,再次随即选择它的后缀,这次是在and 和 will 中随机选择,重复上述过程,就能够产生一段可读的文本。具体描述如下:

代码如下:

设置 w1 和 w2 为文本的前两个词

输出 w1 和 w2

循环:

随机地选出 w3,它是文本中 w1 w2 的后缀中的一个

打印 w3

把 w1 和 w2 分别换成 w2 和 w3

重复循环

2.awk 程序

马尔可夫链算法并不难,我们会在后面看到,用c语言来解决这个问题会相当麻烦,而用awk则只需要5分钟就能搞定。这简直就是一个演示awk优点的问题。

awk 中有关联数组,正好可以用来表示前缀和后缀的关系。程序如下:

# markov.awk: markov chain algorithm for 2-word prefixes

BEGIN { MAXGEN = 10000; NONWORD = "n"; w1 = w2 = NONWORD }

{ for (i = 1; i = 1

p = statetab[w1,w2,r]

if (p == NONWORD)

exit

print p

w1 = w2 # advance chain

w2 = p

}

}

3. C++ 程序

该问题的主要难点就在于通过前缀随机的获取后缀,在C++ 中,我们可以借助map 来实现前缀和后缀的对应关系,以此得到较高的开发效率。

/* Copyright (C) 1999 Lucent Technologies */

/* Excerpted from 'The Practice of Programming' */

/* by Brian W. Kernighan and Rob Pike */

#include

#include

#include

#include

#include

#include

using namespace std;

const int NPREF = 2;

const char NONWORD[] = "n"; // cannot appear as real line: we remove newlines

const int MAXGEN = 10000; // maximum words generated

typedef deque Prefix;

map statetab; // prefix -> suffixes

void build(Prefix&, istream&);

void generate(int nwords);

void add(Prefix&, const string&);

// markov main: markov-chain random text generation

int main(void)

{

int nwords = MAXGEN;

Prefix prefix; // current input prefix

srand(time(NULL));

for (int i = 0; i < NPREF; i++)

add(prefix, NONWORD);

build(prefix, cin);

add(prefix, NONWORD);

generate(nwords);

return 0;

}

// build: read input words, build state table

void build(Prefix& prefix, istream& in)

{

string buf;

while (in >> buf)

add(prefix, buf);

}

// add: add word to suffix deque, update prefix

void add(Prefix& prefix, const string& s)

{

if (prefix.size() == NPREF) {

statetab[prefix].push_back(s);

prefix.pop_front();

}

prefix.push_back(s);

}

// generate: produce output, one word per line

void generate(int nwords)

{

Prefix prefix;

int i;

for (i = 0; i < NPREF; i++)

add(prefix, NONWORD);

for (i = 0; i < nwords; i++) {

vector& suf = statetab[prefix];

const string& w = suf[rand() % suf.size()];

if (w == NONWORD)

break;

cout pref[i]) != 0)

break;

if (i == NPREF) /* found it */

return sp;

}

if (create) {

sp = (State *) emalloc(sizeof(State));

for (i = 0; i < NPREF; i++)

sp->pref[i] = prefix[i];

sp->suf = NULL;

sp->next = statetab[h];

statetab[h] = sp;

}

return sp;

}

/* addsuffix: add to state. suffix must not change later */

void addsuffix(State *sp, char *suffix)

{

Suffix *suf;

suf = (Suffix *) emalloc(sizeof(Suffix));

suf->word = suffix;

suf->next = sp->suf;

sp->suf = suf;

}

/* add: add word to suffix list, update prefix */

void add(char *prefix[NPREF], char *suffix)

{

State *sp;

sp = lookup(prefix, 1); /* create if not found */

addsuffix(sp, suffix);

/* move the words down the prefix */

memmove(prefix, prefix+1, (NPREF-1)*sizeof(prefix[0]));

prefix[NPREF-1] = suffix;

}

/* build: read input, build prefix table */

void build(char *prefix[NPREF], FILE *f)

{

char buf[100], fmt[10];

/* create a format string; %s could overflow buf */

sprintf(fmt, "%%%ds", sizeof(buf)-1);

while (fscanf(f, fmt, buf) != EOF)

add(prefix, estrdup(buf));

}

/* generate: produce output, one word per line */

void generate(int nwords)

{

State *sp;

Suffix *suf;

char *prefix[NPREF], *w;

int i, nmatch;

for (i = 0; i < NPREF; i++) /* reset initial prefix */

prefix[i] = NONWORD;

for (i = 0; i < nwords; i++) {

sp = lookup(prefix, 0);

if (sp == NULL)

eprintf("internal error: lookup failed");

nmatch = 0;

for (suf = sp->suf; suf != NULL; suf = suf->next)

if (rand() % ++nmatch == 0) /* prob = 1/nmatch */

w = suf->word;

if (nmatch == 0)

eprintf("internal error: no suffix %d %s", i, prefix[0]);

if (strcmp(w, NONWORD) == 0)

break;

printf("%sn", w);

memmove(prefix, prefix+1, (NPREF-1)*sizeof(prefix[0]));

prefix[NPREF-1] = w;

}

}

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Matlab中实现马尔科夫链蒙特卡洛(MCMC)算法代码,可以参考以下步骤: 1. 首先,你需要定义你的马尔科夫链。这可以通过一个状态转移矩阵来实现,其中每个元素表示从一个状态转移到另一个状态的概率。 2. 接下来,你需要选择一个初始状态。这个初始状态可以是随机选择的或者是根据问题的特定要求选择的。 3. 然后,你可以使用马尔科夫链的转移矩阵来生成下一个状态。你可以使用随机数生成函数(如rand)来从转移概率中进行采样。 4. 重复步骤3,直到达到你设定的迭代次数或者满足某个收敛准则。 下面是一个简单的马尔科夫链蒙特卡洛算法在Matlab中的伪代码: ``` % 定义马尔科夫链转移矩阵 transition_matrix = [0.2, 0.8; 0.6,0.4]; % 设定初始状态 initial_state = 1; % 设定迭代次数 iterations = 1000; % 初始化状态序列 state_sequence = zeros(1, iterations); % 生成状态序列 current_state = initial_state; for i = 1:iterations % 根据转移概率选择下一个状态 next_state = rand < transition_matrix(current_state, 2); % 更新当前状态 current_state = next_state; % 记录状态序列 state_sequence(i) = current_state; end % 绘制状态序列 plot(state_sequence); ``` 请注意,以上代码只是一个简单的示例,实际应用中可能需要根据具体问题进行调整和扩展。你可以根据你的具体需求和问题来修改代码,并根据需要添加额外的功能和处理步骤。<span class="em">1</span><span class="em">2</span> #### 引用[.reference_title] - *1* [基于MCMC马尔科夫-蒙特卡洛抽样matlab仿真+操作视频 ](https://download.csdn.net/download/ccsss22/85629717)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [Metropolis-Hastings 蒙特卡洛马尔可夫链算法附matlab代码.zip](https://download.csdn.net/download/m0_60703264/87707504)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值