列表

详情


NC19029. Lambda Calculus

描述

The lambda calculus is a simple language that is often used to study the theory of programming languages. This language only consists of variable references, functions that take a single argument, and applications of functions. The lambda calculus consists of a language of lambda terms. The syntax of the lambda calculus defines that all valid lambda terms can be built by applying the following three rules:
  1. a variable x, which is a nonempty string other than lambda, is itself a lambda term;
  2. if t is a valid lambda term, and x is a variable, then (lambda (x) t) is a lambda term (called a lambda abstraction), here x has at least one occurrence in t;
  3. if t and s are lambda terms, then (t s) is a lambda term (called an application).
Nothing else is a lambda term. Thus a lambda term is valid if and only if it can be obtained by repeated application of these three rules. Note that the parentheses in the second and the third rule can NOT be omitted.
A lambda abstraction (lambda (x) t) is a definition of an anonymous function that is capable of taking a single input x and substituting it into the term t. The abstraction binds the variable x in term t, so we call it a lambda binding of variable x. 
An application (t s) represents the application of a function t to an input s, that is, it represents the act of calling function t on input s to produce t(s). 
To see how this works, consider the lambda calculus extended with arithmetic operators. In that language, a lambda abstraction (lambda (x) x + 5), in which x is bound, defines a function f(x) = x + 5. And an application ((lambda (x) x + 5) 3) applies function f(x) = x + 5 to input 3 and produces f(3) = 3 + 5 = 8. 
We say that a variable occurs free in an lambda term t if it has some occurrence in t that is not inside some lambda binding of the same variable. For example,
  • x occurs free in x; 
  • x does not occur free in y;
  • x does not occur free in (lambda (x) (x y));
  • x occurs free in (lambda (y) (x y));
  • x occurs free in ((lambda (x) x) (x y)), which is an application that produces (x y) in which x occurs free; 
  • x occurs free in (lambda (y) (lambda (z) (x (y z)))). 
Now you are given a lambda term t, you need to find all distinct variables that occur free in t. 

输入描述

The first line of the input contains a positive integer T denoting the number of test cases. Then T lines follow, each line contains a nonempty string denoting a lambda term. 
Each variable in the the lambda terms contains only latin letters and the hyphen, and is no longer than 20 characters. 
It is guaranteed that all lambda terms in the input are valid, and the total length of all lambda terms will not exceed 107
There might be some extra spaces in the input as long as they do not cause any misunderstanding.

输出描述

For each test case, output first the case number and then all distinct variables that occur free in the lambda term in a single line. These variables should be printed in lexicographic order, and there should be a single space between each two adjacent variables. There should be a space after the colon in each test case.

示例1

输入:

6
x
y
(lambda (x) (x y)) 
(lambda (y) (x y)) 
( (lambda (x) x) (x y) )
( lambda (y) ( lambda (z) (x (y z)) ) )

输出:

Case #1: x
Case #2: y
Case #3: y
Case #4: x
Case #5: x y
Case #6: x

说明:

The free variables of a term are those variables not bound by a lambda abstraction. The set of free variables of an expression is defined inductively:

1. The free variables of x are just x;
2. The set of free variables of (lambda (x) t) is the set of free variables of t, but with x removed;
3. The set of free variables of (t s) is the union

We can define it with the context-free grammar:
      LcExp ::= Variable
            ::= (lambda (Variable) LcExp) 
            ::= (LcExp LcExp)
where a variable is any string other than lambda.


原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++ 解法, 执行用时: 25ms, 内存消耗: 9380K, 提交时间: 2021-09-29 00:58:53

#include <functional>
#include <iostream>
#include <memory>
#include <set>
#include <string>
#include <utility>
#include <vector>

void assert(bool f) {
  if (!f) while(1);
}

std::set<std::string> freeVar;

struct Parser {
  const std::string& str;
  int i = 0;
  std::multiset<std::string> context;

  Parser(const std::string& str) : str(str) {}

  void space() {
    while (i < str.length() && str[i] == ' ') i++;
  }

  std::string var() {
    space();
    std::string v;
    while (i < str.length() && str[i] != '(' && str[i] != ')' && str[i] != ' ') {
      v += str[i++];
    }
    space();
    return v;
  }

  void app() {
    space();
    term();
    space();
    term();
    space();
  }

  void term() {
    if (str[i] == '(') {
      i++;
      space();
      int old = i;
      std::string v = var();
      if (v == "lambda") {
        assert(str[i++] == '(');
        space();
        v = var();
        assert(str[i++] == ')');
        space();
        context.insert(v);
        term();
        context.erase(context.find(v));
        assert(str[i++] == ')');
        space();
      } else {
        i = old;
        app();
        assert(str[i++] == ')');
        space();
      }
    } else {
      std::string v = var();
      assert(v.length() != 0);
      if (!context.count(v)) {
        freeVar.insert(v);
      }
    }
  }

  void run() {
    space();
    term();
    space();
    assert(i == str.length());
  }
};

int main() {
  std::ios::sync_with_stdio(false);
  int T;
  std::cin >> T;
  std::cin.ignore();
  for (int kase = 1; kase <= T; kase++) {
    std::string code;
    std::getline(std::cin, code);
    freeVar.clear();
    auto p = Parser(code);
    p.run();
    std::cout << "Case #" << kase << ":";
    for (auto& v : freeVar) std::cout << ' ' << v;
    if (freeVar.empty()) std::cout << ' ';
    std::cout << '\n';
  }
  return 0;
}

C++14(g++5.4) 解法, 执行用时: 75ms, 内存消耗: 24028K, 提交时间: 2018-10-07 14:14:13

#include<bits/stdc++.h>
using namespace std;

const int maxn=1e7+10;

int n;
char s[maxn];

set<string>ans;
set<pair<string,int> >SET;

string getname(int &start){
	string t="";
	while(start<n&&(s[start]>='a'&&s[start]<='z'||
				s[start]>='A'&&s[start]<='Z'||
				s[start]=='-'))t.push_back(s[start++]);
	return t;
}

void eatspace(int &start){
	while(start<n&&s[start]==' ')start++;
}

void DFS(int deep,int &start){
	while(start<n){
		eatspace(start);
		if(start>=n)return;
		if(s[start]=='('){
			start++;
			DFS(deep+1,start);
		}
		else if(s[start]==')'){
			start++;
			return;
		}
		else{
			string name=getname(start);
			if(name=="lambda"){
				eatspace(start);
				if(start>=n)return;
				start++;// (
				eatspace(start);
				if(start>=n)return;
				string know=getname(start);
				SET.insert(make_pair(know,deep));
				eatspace(start);
				if(start>=n)return;
				start++;// )
				DFS(deep,start);
				SET.erase(make_pair(know,deep));
			}
			else{
				int flag=0;
				for(int i=deep;i>=0;i--)if(SET.count(make_pair(name,i))){flag=1;break;}
				if(!flag)ans.insert(name);
			}
		}
	}
}


void gao(int kase){
	SET.clear();
	ans.clear();
	fgets(s,maxn,stdin);

	n=strlen(s)-1;
	int start=0;
	DFS(0,start);
	printf("Case #%d:",kase);
	if(!ans.size())printf(" ");
	for(string t:ans)printf(" %s",t.c_str());
	printf("\n");
}

int main(){
	int T;
	scanf("%d\n",&T);
	for(int i=1;i<=T;i++)gao(i);
	return 0;
}

上一题