#Model constructorclassHigherOrderNamingGame():def__init__(self,simplices,rule):#Structureself.simplices=simplicesself.nodes=list({nodeforsimplexinsimplicesfornodeinsimplex})self.N=len(self.nodes)#Timeself.t=0#Ruleself.rule=ruledefSetInitialConditions(self,beta,p,n_A,verbose=False):#Game parametersself.beta=betaself.p=p#Opinions of the nodes (vocabularies)self.opinions={}#Generating committed agentsN_p=int(self.N*self.p)#number of committed agents#Randomly picking N_p committed agentscommitted=random.sample(self.nodes,N_p)#Setting up a committed dictionary#self.is_committed = {n:False for n in self.nodes}forninself.nodes:ifnincommitted:#self.is_committed[n]=True#Assigning opinion "A" to committed agentsself.opinions[n]=frozenset(["A"])#Calculating the number of agents holding n_A (not committed) from the density in inputN_A=int(self.N*n_A)N_B=self.N-N_A-N_p#Creating a list of opinions to assignopinions_to_assign=['A']*N_A+['B']*N_B#Shuffling themrandom.shuffle(opinions_to_assign)#Agents left to be matched with opinionsnoncommitted=set(self.nodes)-set(committed)forn,oinzip(noncommitted,opinions_to_assign):self.opinions[n]=set(o)ifverbose:print('Setup Done.',self.N,'nodes,',"N_A:",N_A,"N_B:",N_B,"N_p:",N_p)defAgreeOnSimplex(self,simplex,said_word):#Updating the simplex on the agreed word forninsimplex:try:#There are also committed minorities which have frozensets!self.opinions[n].clear()self.opinions[n].add(said_word)exceptAttributeError:#It was committedpassdefListenersLearnWord(self,listeners,said_word):#Looping through the listenersforlistenerinlisteners:try:#Trying to learn...self.opinions[listener].add(said_word)exceptAttributeError:#It was committedpassdefplay_on_simplex(self,simplex):#Selecting speaker and listeners at randomrandom.shuffle(simplex)speaker=simplex[0]listeners=simplex[1:]#Selecting a random word to be saidsaid_word=random.choice(list(self.opinions[speaker]))words_of_listeners=[self.opinions[listener]forlistenerinlisteners]#Using the rule to get the words of listeners to be used for the agreementifself.rule=='union':words_of_listeners_by_rule=set.union(*[set(w)forwinwords_of_listeners])elifself.rule=='intersection':words_of_listeners_by_rule=set.intersection(*[set(w)forwinwords_of_listeners])#Trying to agree based on the rule and the communication efficiency betaif(said_wordinwords_of_listeners_by_rule)and(random.random()<=self.beta):self.AgreeOnSimplex(simplex,said_word)else:#No agreement, but the said word is learned by the listenersself.ListenersLearnWord(listeners,said_word)defget_densities(self):single_opinion_counter=collections.Counter([list(opinions)[0]foropinionsinself.opinions.values()iflen(opinions)==1])n_Ap=single_opinion_counter["A"]/self.Nn_B=single_opinion_counter["B"]/self.Nn_AB=1-n_Ap-n_Breturnn_Ap,n_B,n_ABdefrun(self,path,t_max=100,check_every=10,print_every=1):self.t_max=t_max#Opening file to save densities resultsdensities_path=path+'HONG_densities_N%i_beta%.4f_p%.2f.csv'%(self.N,self.beta,self.p)f=open(densities_path,'w')f.write('time,n_A+p,n_B,n_AB\n')whileself.t<=self.t_max:self.t+=1ifself.t%print_every==0: print('t=%i'%self.t)
######### THIS IS FOR PLAYING ALWAYS ON ALL SIMPLICES AT EACH TIME ##########Reshuffling the sequence of simplices#random.shuffle(self.simplices)#Looping over each simplex and playing#for simplex in self.simplices:#self.play_on_simplex(simplex)#############################################################################Playing on a random simplex simplex=random.choice(self.simplices)self.play_on_simplex(simplex)#Storing the values every check_every time steps:ifself.t%check_every==0:
n_Ap,n_B,n_AB=self.get_densities()line="%i,%.3f,%.3f,%.3f\n"%(self.t,n_Ap,n_B,n_AB)f.write(line)#Also checking if we reached the absorbing state:ifn_Ap==1orn_B==1:f.close()print('DONE! Reached the absorbing state.')returnNonef.close()print('DONE! Run out of time...')
13.3. Running on an homogeneous mixing 2-hypergraph#
fromitertoolsimportcombinationsdefHomMix2HG(N,k=2):nodes=range(N)simplices=list(combinations(nodes,k+1))#These are tuples, so I'll convert them to listssimplices=[list(simplex)forsimplexinsimplices]returnsimplices