Language Models as Compilers: Simulating Pseudocode Execution Improves Algorithmic Reasoning in Language Models
Hyungjoo Chae, Yeonghyeon Kim, Seungone Kim, Kai Tzu-iunn Ong, Beong-woo Kwak, Moohyeon Kim, Seonghwan Kim, Taeyoon Kwon, Jiwan Chung, Youngjae Yu, Jinyoung Yeo
TL;DR
The paper tackles the challenge of algorithmic reasoning in large language models by introducing Think-and-Execute, a two-stage framework that first uncovers task-level reasoning logic and encodes it as pseudocode (Think), then tailors and executes this logic for each instance (Execute). By separating planning from execution and using a pseudocode representation, the approach enables reuse of logic across multiple instances and improves reasoning performance across seven Big-Bench Hard tasks, outperforming direct prompting, zero-shot CoT, and instance-specific Python code. Empirical results show the method transfers to small LMs like CodeLlama and that pseudocode generally describes the task logic better than natural language plans. Ablation and analysis reveal the importance of semantic content, pre-analysis, and code knowledge, underscoring the method’s reliance on task-level structure and code familiarity to enhance algorithmic reasoning.
Abstract
Algorithmic reasoning refers to the ability to understand the complex patterns behind the problem and decompose them into a sequence of reasoning steps towards the solution. Such nature of algorithmic reasoning makes it a challenge for large language models (LLMs), even though they have demonstrated promising performance in other reasoning tasks. Within this context, some recent studies use programming languages (e.g., Python) to express the necessary logic for solving a given instance/question (e.g., Program-of-Thought) as inspired by their strict and precise syntaxes. However, it is non-trivial to write an executable code that expresses the correct logic on the fly within a single inference call. Also, the code generated specifically for an instance cannot be reused for others, even if they are from the same task and might require identical logic to solve. This paper presents Think-and-Execute, a novel framework that decomposes the reasoning process of language models into two steps. (1) In Think, we discover a task-level logic that is shared across all instances for solving a given task and then express the logic with pseudocode; (2) In Execute, we further tailor the generated pseudocode to each instance and simulate the execution of the code. With extensive experiments on seven algorithmic reasoning tasks, we demonstrate the effectiveness of Think-and-Execute. Our approach better improves LMs' reasoning compared to several strong baselines performing instance-specific reasoning (e.g., CoT and PoT), suggesting the helpfulness of discovering task-level logic. Also, we show that compared to natural language, pseudocode can better guide the reasoning of LMs, even though they are trained to follow natural language instructions.
